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"
|
|
|
|
#include "textures/textures.h"
|
2016-06-10 11:50:34 +00:00
|
|
|
#include "v_palette.h"
|
2016-03-01 15:47:10 +00:00
|
|
|
|
2018-03-31 17:20:59 +00:00
|
|
|
FTexture *CreateBrightmapTexture(FTexture*);
|
|
|
|
|
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
|
|
|
|
2017-03-09 18:54:41 +00:00
|
|
|
uint8_t FTexture::GrayMap[256];
|
2016-03-01 15:47:10 +00:00
|
|
|
|
|
|
|
void FTexture::InitGrayMap()
|
|
|
|
{
|
|
|
|
for (int i = 0; i < 256; ++i)
|
|
|
|
{
|
2018-03-31 17:20:59 +00:00
|
|
|
GrayMap[i] = ColorMatcher.Pick(i, i, i);
|
2016-03-01 15:47:10 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-03-31 17:20:59 +00:00
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
typedef FTexture * (*CreateFunc)(FileReader & file, int lumpnum);
|
|
|
|
|
|
|
|
struct TexCreateInfo
|
|
|
|
{
|
|
|
|
CreateFunc TryCreate;
|
|
|
|
ETextureType usetype;
|
|
|
|
};
|
|
|
|
|
2018-03-11 17:32:00 +00:00
|
|
|
FTexture *IMGZTexture_TryCreate(FileReader &, int lumpnum);
|
|
|
|
FTexture *PNGTexture_TryCreate(FileReader &, int lumpnum);
|
|
|
|
FTexture *JPEGTexture_TryCreate(FileReader &, int lumpnum);
|
|
|
|
FTexture *DDSTexture_TryCreate(FileReader &, int lumpnum);
|
|
|
|
FTexture *PCXTexture_TryCreate(FileReader &, int lumpnum);
|
|
|
|
FTexture *TGATexture_TryCreate(FileReader &, int lumpnum);
|
|
|
|
FTexture *RawPageTexture_TryCreate(FileReader &, int lumpnum);
|
|
|
|
FTexture *FlatTexture_TryCreate(FileReader &, int lumpnum);
|
|
|
|
FTexture *PatchTexture_TryCreate(FileReader &, int lumpnum);
|
|
|
|
FTexture *EmptyTexture_TryCreate(FileReader &, int lumpnum);
|
|
|
|
FTexture *AutomapTexture_TryCreate(FileReader &, int lumpnum);
|
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-03-25 18:26:16 +00:00
|
|
|
FTexture * FTexture::CreateTexture (int lumpnum, ETextureType usetype)
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
|
|
|
static TexCreateInfo CreateInfo[]={
|
2018-03-25 18:26:16 +00:00
|
|
|
{ IMGZTexture_TryCreate, ETextureType::Any },
|
|
|
|
{ PNGTexture_TryCreate, ETextureType::Any },
|
|
|
|
{ JPEGTexture_TryCreate, ETextureType::Any },
|
|
|
|
{ DDSTexture_TryCreate, ETextureType::Any },
|
|
|
|
{ PCXTexture_TryCreate, ETextureType::Any },
|
|
|
|
{ TGATexture_TryCreate, ETextureType::Any },
|
|
|
|
{ RawPageTexture_TryCreate, ETextureType::MiscPatch },
|
|
|
|
{ FlatTexture_TryCreate, ETextureType::Flat },
|
|
|
|
{ PatchTexture_TryCreate, ETextureType::Any },
|
|
|
|
{ EmptyTexture_TryCreate, ETextureType::Any },
|
|
|
|
{ AutomapTexture_TryCreate, ETextureType::MiscPatch },
|
2016-03-01 15:47:10 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
if (lumpnum == -1) return NULL;
|
|
|
|
|
2018-03-10 17:45:11 +00:00
|
|
|
auto data = Wads.OpenLumpReader (lumpnum);
|
2016-03-01 15:47:10 +00:00
|
|
|
|
|
|
|
for(size_t i = 0; i < countof(CreateInfo); i++)
|
|
|
|
{
|
2018-03-25 18:26:16 +00:00
|
|
|
if ((CreateInfo[i].usetype == usetype || CreateInfo[i].usetype == ETextureType::Any))
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
|
|
|
FTexture * tex = CreateInfo[i].TryCreate(data, lumpnum);
|
|
|
|
if (tex != NULL)
|
|
|
|
{
|
|
|
|
tex->UseType = usetype;
|
2018-03-25 18:26:16 +00:00
|
|
|
if (usetype == ETextureType::Flat)
|
2016-03-01 15:47:10 +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-26 12:37:44 +00:00
|
|
|
tex->Scale.X = tex->Scale.Y = 2;
|
2016-03-01 15:47:10 +00:00
|
|
|
tex->bWorldPanning = true;
|
|
|
|
}
|
|
|
|
else if (w==256 && h==256)
|
|
|
|
{
|
2016-03-26 12:37:44 +00:00
|
|
|
tex->Scale.X = tex->Scale.Y = 4;
|
2016-03-01 15:47:10 +00:00
|
|
|
tex->bWorldPanning = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return tex;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2018-03-25 18:26:16 +00:00
|
|
|
FTexture * FTexture::CreateTexture (const char *name, int lumpnum, ETextureType usetype)
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
|
|
|
FTexture *tex = CreateTexture(lumpnum, usetype);
|
|
|
|
if (tex != NULL && name != NULL) {
|
|
|
|
tex->Name = name;
|
|
|
|
tex->Name.ToUpper();
|
|
|
|
}
|
|
|
|
return tex;
|
|
|
|
}
|
|
|
|
|
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
|
|
|
:
|
2016-03-26 12:37:44 +00:00
|
|
|
WidthBits(0), HeightBits(0), 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-03-23 22:04:30 +00:00
|
|
|
Rotations(0xFFFF), SkyOffset(0), Width(0), Height(0), WidthMask(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-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;
|
2016-03-01 15:47:10 +00:00
|
|
|
}
|
|
|
|
|
2016-06-13 19:39:55 +00:00
|
|
|
void FTexture::Unload()
|
|
|
|
{
|
|
|
|
PixelsBgra = std::vector<uint32_t>();
|
|
|
|
}
|
|
|
|
|
2018-03-31 17:20:59 +00:00
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
2016-06-10 11:50:34 +00:00
|
|
|
const uint32_t *FTexture::GetColumnBgra(unsigned int column, const Span **spans_out)
|
|
|
|
{
|
|
|
|
const uint32_t *pixels = GetPixelsBgra();
|
2018-03-18 20:33:44 +00:00
|
|
|
if (pixels == nullptr) return nullptr;
|
2016-06-10 11:50:34 +00:00
|
|
|
|
|
|
|
column %= Width;
|
|
|
|
|
|
|
|
if (spans_out != nullptr)
|
2018-03-18 20:33:44 +00:00
|
|
|
GetColumn(DefaultRenderStyle(), column, spans_out); // This isn't the right way to create the spans.
|
2016-06-10 11:50:34 +00:00
|
|
|
return pixels + column * Height;
|
|
|
|
}
|
|
|
|
|
|
|
|
const uint32_t *FTexture::GetPixelsBgra()
|
|
|
|
{
|
2018-03-18 20:33:44 +00:00
|
|
|
if (PixelsBgra.empty() || CheckModified(DefaultRenderStyle()))
|
2016-06-10 11:50:34 +00:00
|
|
|
{
|
2018-03-18 20:33:44 +00:00
|
|
|
if (!GetColumn(DefaultRenderStyle(), 0, nullptr))
|
2016-06-13 19:39:55 +00:00
|
|
|
return nullptr;
|
2016-08-06 20:59:16 +00:00
|
|
|
|
|
|
|
FBitmap bitmap;
|
|
|
|
bitmap.Create(GetWidth(), GetHeight());
|
|
|
|
CopyTrueColorPixels(&bitmap, 0, 0);
|
|
|
|
GenerateBgraFromBitmap(bitmap);
|
2016-06-10 11:50:34 +00:00
|
|
|
}
|
2016-06-13 19:39:55 +00:00
|
|
|
return PixelsBgra.data();
|
2016-06-10 11:50:34 +00:00
|
|
|
}
|
|
|
|
|
2018-03-31 17:20:59 +00:00
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
2018-03-18 20:33:44 +00:00
|
|
|
bool FTexture::CheckModified (FRenderStyle)
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
FTextureFormat FTexture::GetFormat()
|
|
|
|
{
|
|
|
|
return TEX_Pal;
|
|
|
|
}
|
|
|
|
|
|
|
|
void FTexture::SetFrontSkyLayer ()
|
|
|
|
{
|
|
|
|
bNoRemap0 = true;
|
|
|
|
}
|
|
|
|
|
2018-03-31 17:20:59 +00:00
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
2016-03-01 15:47:10 +00:00
|
|
|
void FTexture::CalcBitSize ()
|
|
|
|
{
|
|
|
|
// WidthBits is rounded down, and HeightBits is rounded up
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; (1 << i) < Width; ++i)
|
|
|
|
{ }
|
|
|
|
|
|
|
|
WidthBits = i;
|
|
|
|
|
|
|
|
// Having WidthBits that would allow for columns past the end of the
|
|
|
|
// texture is not allowed, even if it means the entire texture is
|
|
|
|
// not drawn.
|
|
|
|
if (Width < (1 << WidthBits))
|
|
|
|
{
|
|
|
|
WidthBits--;
|
|
|
|
}
|
|
|
|
WidthMask = (1 << WidthBits) - 1;
|
|
|
|
|
2016-11-01 05:08:16 +00:00
|
|
|
// <hr>The minimum height is 2, because we cannot shift right 32 bits.</hr>
|
|
|
|
// Scratch that. Somebody actually made a 1x1 texture, so now we have to handle it.
|
|
|
|
for (i = 0; (1 << i) < Height; ++i)
|
2016-03-01 15:47:10 +00:00
|
|
|
{ }
|
|
|
|
|
|
|
|
HeightBits = i;
|
|
|
|
}
|
|
|
|
|
2018-03-31 17:20:59 +00:00
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
2017-03-09 18:54:41 +00:00
|
|
|
FTexture::Span **FTexture::CreateSpans (const uint8_t *pixels) const
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
|
|
|
Span **spans, *span;
|
|
|
|
|
|
|
|
if (!bMasked)
|
|
|
|
{ // Texture does not have holes, so it can use a simpler span structure
|
|
|
|
spans = (Span **)M_Malloc (sizeof(Span*)*Width + sizeof(Span)*2);
|
|
|
|
span = (Span *)&spans[Width];
|
|
|
|
for (int x = 0; x < Width; ++x)
|
|
|
|
{
|
|
|
|
spans[x] = span;
|
|
|
|
}
|
|
|
|
span[0].Length = Height;
|
|
|
|
span[0].TopOffset = 0;
|
|
|
|
span[1].Length = 0;
|
|
|
|
span[1].TopOffset = 0;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{ // Texture might have holes, so build a complete span structure
|
|
|
|
int numcols = Width;
|
|
|
|
int numrows = Height;
|
|
|
|
int numspans = numcols; // One span to terminate each column
|
2017-03-09 18:54:41 +00:00
|
|
|
const uint8_t *data_p;
|
2016-03-01 15:47:10 +00:00
|
|
|
bool newspan;
|
|
|
|
int x, y;
|
|
|
|
|
|
|
|
data_p = pixels;
|
|
|
|
|
|
|
|
// Count the number of spans in this texture
|
|
|
|
for (x = numcols; x > 0; --x)
|
|
|
|
{
|
|
|
|
newspan = true;
|
|
|
|
for (y = numrows; y > 0; --y)
|
|
|
|
{
|
2018-03-18 20:33:44 +00:00
|
|
|
|
2016-03-01 15:47:10 +00:00
|
|
|
if (*data_p++ == 0)
|
|
|
|
{
|
|
|
|
if (!newspan)
|
|
|
|
{
|
|
|
|
newspan = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (newspan)
|
|
|
|
{
|
|
|
|
newspan = false;
|
|
|
|
numspans++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Allocate space for the spans
|
|
|
|
spans = (Span **)M_Malloc (sizeof(Span*)*numcols + sizeof(Span)*numspans);
|
|
|
|
|
|
|
|
// Fill in the spans
|
|
|
|
for (x = 0, span = (Span *)&spans[numcols], data_p = pixels; x < numcols; ++x)
|
|
|
|
{
|
|
|
|
newspan = true;
|
|
|
|
spans[x] = span;
|
|
|
|
for (y = 0; y < numrows; ++y)
|
|
|
|
{
|
|
|
|
if (*data_p++ == 0)
|
|
|
|
{
|
|
|
|
if (!newspan)
|
|
|
|
{
|
|
|
|
newspan = true;
|
|
|
|
span++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (newspan)
|
|
|
|
{
|
|
|
|
newspan = false;
|
|
|
|
span->TopOffset = y;
|
|
|
|
span->Length = 1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
span->Length++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!newspan)
|
|
|
|
{
|
|
|
|
span++;
|
|
|
|
}
|
|
|
|
span->TopOffset = 0;
|
|
|
|
span->Length = 0;
|
|
|
|
span++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return spans;
|
|
|
|
}
|
|
|
|
|
|
|
|
void FTexture::FreeSpans (Span **spans) const
|
|
|
|
{
|
|
|
|
M_Free (spans);
|
|
|
|
}
|
|
|
|
|
2018-03-31 17:20:59 +00:00
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
2016-08-06 20:59:16 +00:00
|
|
|
void FTexture::GenerateBgraFromBitmap(const FBitmap &bitmap)
|
|
|
|
{
|
|
|
|
CreatePixelsBgraWithMipmaps();
|
|
|
|
|
2016-08-14 03:10:34 +00:00
|
|
|
// Transpose
|
2016-08-06 20:59:16 +00:00
|
|
|
const uint32_t *src = (const uint32_t *)bitmap.GetPixels();
|
|
|
|
uint32_t *dest = PixelsBgra.data();
|
|
|
|
for (int x = 0; x < Width; x++)
|
|
|
|
{
|
|
|
|
for (int y = 0; y < Height; y++)
|
|
|
|
{
|
2016-08-14 03:10:34 +00:00
|
|
|
dest[y + x * Height] = src[x + y * Width];
|
2016-08-06 20:59:16 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
GenerateBgraMipmaps();
|
|
|
|
}
|
|
|
|
|
2016-06-21 19:55:08 +00:00
|
|
|
void FTexture::CreatePixelsBgraWithMipmaps()
|
|
|
|
{
|
|
|
|
int levels = MipmapLevels();
|
|
|
|
int buffersize = 0;
|
|
|
|
for (int i = 0; i < levels; i++)
|
|
|
|
{
|
|
|
|
int w = MAX(Width >> i, 1);
|
|
|
|
int h = MAX(Height >> i, 1);
|
|
|
|
buffersize += w * h;
|
|
|
|
}
|
|
|
|
PixelsBgra.resize(buffersize, 0xffff0000);
|
|
|
|
}
|
|
|
|
|
|
|
|
int FTexture::MipmapLevels() const
|
|
|
|
{
|
|
|
|
int widthbits = 0;
|
|
|
|
while ((Width >> widthbits) != 0) widthbits++;
|
|
|
|
|
|
|
|
int heightbits = 0;
|
|
|
|
while ((Height >> heightbits) != 0) heightbits++;
|
|
|
|
|
|
|
|
return MAX(widthbits, heightbits);
|
|
|
|
}
|
|
|
|
|
2018-03-31 17:20:59 +00:00
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
2016-06-21 19:55:08 +00:00
|
|
|
void FTexture::GenerateBgraMipmaps()
|
2016-06-25 10:14:15 +00:00
|
|
|
{
|
2016-06-27 08:49:15 +00:00
|
|
|
struct Color4f
|
|
|
|
{
|
|
|
|
float a, r, g, b;
|
|
|
|
Color4f operator*(const Color4f &v) const { return Color4f{ a * v.a, r * v.r, g * v.g, b * v.b }; }
|
|
|
|
Color4f operator/(const Color4f &v) const { return Color4f{ a / v.a, r / v.r, g / v.g, b / v.b }; }
|
|
|
|
Color4f operator+(const Color4f &v) const { return Color4f{ a + v.a, r + v.r, g + v.g, b + v.b }; }
|
|
|
|
Color4f operator-(const Color4f &v) const { return Color4f{ a - v.a, r - v.r, g - v.g, b - v.b }; }
|
|
|
|
Color4f operator*(float s) const { return Color4f{ a * s, r * s, g * s, b * s }; }
|
|
|
|
Color4f operator/(float s) const { return Color4f{ a / s, r / s, g / s, b / s }; }
|
|
|
|
Color4f operator+(float s) const { return Color4f{ a + s, r + s, g + s, b + s }; }
|
|
|
|
Color4f operator-(float s) const { return Color4f{ a - s, r - s, g - s, b - s }; }
|
|
|
|
};
|
2016-06-25 10:14:15 +00:00
|
|
|
|
|
|
|
int levels = MipmapLevels();
|
2016-06-27 08:49:15 +00:00
|
|
|
std::vector<Color4f> image(PixelsBgra.size());
|
|
|
|
|
|
|
|
// Convert to normalized linear colorspace
|
2016-06-25 10:14:15 +00:00
|
|
|
{
|
2016-06-27 08:49:15 +00:00
|
|
|
for (int x = 0; x < Width; x++)
|
|
|
|
{
|
|
|
|
for (int y = 0; y < Height; y++)
|
|
|
|
{
|
|
|
|
uint32_t c8 = PixelsBgra[x * Height + y];
|
|
|
|
Color4f c;
|
2016-06-30 11:45:06 +00:00
|
|
|
c.a = powf(APART(c8) * (1.0f / 255.0f), 2.2f);
|
|
|
|
c.r = powf(RPART(c8) * (1.0f / 255.0f), 2.2f);
|
|
|
|
c.g = powf(GPART(c8) * (1.0f / 255.0f), 2.2f);
|
|
|
|
c.b = powf(BPART(c8) * (1.0f / 255.0f), 2.2f);
|
2016-06-27 08:49:15 +00:00
|
|
|
image[x * Height + y] = c;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Generate mipmaps
|
|
|
|
{
|
|
|
|
std::vector<Color4f> smoothed(Width * Height);
|
|
|
|
Color4f *src = image.data();
|
|
|
|
Color4f *dest = src + Width * Height;
|
|
|
|
for (int i = 1; i < levels; i++)
|
|
|
|
{
|
|
|
|
int srcw = MAX(Width >> (i - 1), 1);
|
|
|
|
int srch = MAX(Height >> (i - 1), 1);
|
|
|
|
int w = MAX(Width >> i, 1);
|
|
|
|
int h = MAX(Height >> i, 1);
|
|
|
|
|
|
|
|
// Downscale
|
|
|
|
for (int x = 0; x < w; x++)
|
|
|
|
{
|
|
|
|
int sx0 = x * 2;
|
|
|
|
int sx1 = MIN((x + 1) * 2, srcw - 1);
|
|
|
|
for (int y = 0; y < h; y++)
|
|
|
|
{
|
|
|
|
int sy0 = y * 2;
|
|
|
|
int sy1 = MIN((y + 1) * 2, srch - 1);
|
2016-06-25 10:14:15 +00:00
|
|
|
|
2016-06-27 08:49:15 +00:00
|
|
|
Color4f src00 = src[sy0 + sx0 * srch];
|
|
|
|
Color4f src01 = src[sy1 + sx0 * srch];
|
|
|
|
Color4f src10 = src[sy0 + sx1 * srch];
|
|
|
|
Color4f src11 = src[sy1 + sx1 * srch];
|
|
|
|
Color4f c = (src00 + src01 + src10 + src11) * 0.25f;
|
2016-06-25 10:14:15 +00:00
|
|
|
|
2016-11-05 15:12:59 +00:00
|
|
|
dest[y + x * h] = c;
|
2016-06-27 08:49:15 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Sharpen filter with a 3x3 kernel:
|
|
|
|
for (int x = 0; x < w; x++)
|
|
|
|
{
|
|
|
|
for (int y = 0; y < h; y++)
|
|
|
|
{
|
|
|
|
Color4f c = { 0.0f, 0.0f, 0.0f, 0.0f };
|
|
|
|
for (int kx = -1; kx < 2; kx++)
|
|
|
|
{
|
|
|
|
for (int ky = -1; ky < 2; ky++)
|
|
|
|
{
|
|
|
|
int a = y + ky;
|
|
|
|
int b = x + kx;
|
|
|
|
if (a < 0) a = h - 1;
|
|
|
|
if (a == h) a = 0;
|
|
|
|
if (b < 0) b = w - 1;
|
2016-08-08 20:35:26 +00:00
|
|
|
if (b == w) b = 0;
|
2016-06-27 08:49:15 +00:00
|
|
|
c = c + dest[a + b * h];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
c = c * (1.0f / 9.0f);
|
|
|
|
smoothed[y + x * h] = c;
|
|
|
|
}
|
|
|
|
}
|
2016-11-05 15:12:59 +00:00
|
|
|
float k = 0.08f;
|
2016-06-27 08:49:15 +00:00
|
|
|
for (int j = 0; j < w * h; j++)
|
|
|
|
dest[j] = dest[j] + (dest[j] - smoothed[j]) * k;
|
|
|
|
|
|
|
|
src = dest;
|
|
|
|
dest += w * h;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Convert to bgra8 sRGB colorspace
|
|
|
|
{
|
|
|
|
Color4f *src = image.data() + Width * Height;
|
|
|
|
uint32_t *dest = PixelsBgra.data() + Width * Height;
|
|
|
|
for (int i = 1; i < levels; i++)
|
|
|
|
{
|
|
|
|
int w = MAX(Width >> i, 1);
|
|
|
|
int h = MAX(Height >> i, 1);
|
|
|
|
for (int j = 0; j < w * h; j++)
|
|
|
|
{
|
2016-11-05 15:12:59 +00:00
|
|
|
uint32_t a = (uint32_t)clamp(powf(MAX(src[j].a, 0.0f), 1.0f / 2.2f) * 255.0f + 0.5f, 0.0f, 255.0f);
|
|
|
|
uint32_t r = (uint32_t)clamp(powf(MAX(src[j].r, 0.0f), 1.0f / 2.2f) * 255.0f + 0.5f, 0.0f, 255.0f);
|
|
|
|
uint32_t g = (uint32_t)clamp(powf(MAX(src[j].g, 0.0f), 1.0f / 2.2f) * 255.0f + 0.5f, 0.0f, 255.0f);
|
|
|
|
uint32_t b = (uint32_t)clamp(powf(MAX(src[j].b, 0.0f), 1.0f / 2.2f) * 255.0f + 0.5f, 0.0f, 255.0f);
|
2016-06-27 08:49:15 +00:00
|
|
|
dest[j] = (a << 24) | (r << 16) | (g << 8) | b;
|
|
|
|
}
|
|
|
|
src += w * h;
|
|
|
|
dest += w * h;
|
|
|
|
}
|
2016-06-25 10:14:15 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-03-31 17:20:59 +00:00
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
2016-06-25 10:14:15 +00:00
|
|
|
void FTexture::GenerateBgraMipmapsFast()
|
2016-06-21 19:55:08 +00:00
|
|
|
{
|
|
|
|
uint32_t *src = PixelsBgra.data();
|
|
|
|
uint32_t *dest = src + Width * Height;
|
|
|
|
int levels = MipmapLevels();
|
|
|
|
for (int i = 1; i < levels; i++)
|
|
|
|
{
|
|
|
|
int srcw = MAX(Width >> (i - 1), 1);
|
|
|
|
int srch = MAX(Height >> (i - 1), 1);
|
|
|
|
int w = MAX(Width >> i, 1);
|
|
|
|
int h = MAX(Height >> i, 1);
|
|
|
|
|
|
|
|
for (int x = 0; x < w; x++)
|
|
|
|
{
|
|
|
|
int sx0 = x * 2;
|
|
|
|
int sx1 = MIN((x + 1) * 2, srcw - 1);
|
|
|
|
|
|
|
|
for (int y = 0; y < h; y++)
|
|
|
|
{
|
|
|
|
int sy0 = y * 2;
|
|
|
|
int sy1 = MIN((y + 1) * 2, srch - 1);
|
|
|
|
|
|
|
|
uint32_t src00 = src[sy0 + sx0 * srch];
|
|
|
|
uint32_t src01 = src[sy1 + sx0 * srch];
|
|
|
|
uint32_t src10 = src[sy0 + sx1 * srch];
|
|
|
|
uint32_t src11 = src[sy1 + sx1 * srch];
|
|
|
|
|
|
|
|
uint32_t alpha = (APART(src00) + APART(src01) + APART(src10) + APART(src11) + 2) / 4;
|
|
|
|
uint32_t red = (RPART(src00) + RPART(src01) + RPART(src10) + RPART(src11) + 2) / 4;
|
|
|
|
uint32_t green = (GPART(src00) + GPART(src01) + GPART(src10) + GPART(src11) + 2) / 4;
|
|
|
|
uint32_t blue = (BPART(src00) + BPART(src01) + BPART(src10) + BPART(src11) + 2) / 4;
|
|
|
|
|
|
|
|
dest[y + x * h] = (alpha << 24) | (red << 16) | (green << 8) | blue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
src = dest;
|
|
|
|
dest += w * h;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-03-31 17:20:59 +00:00
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
2018-03-18 11:36:14 +00:00
|
|
|
void FTexture::CopyToBlock (uint8_t *dest, int dwidth, int dheight, int xpos, int ypos, int rotate, const uint8_t *translation, FRenderStyle style)
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
2018-03-18 20:33:44 +00:00
|
|
|
const uint8_t *pixels = GetPixels(style);
|
2016-03-01 15:47:10 +00:00
|
|
|
int srcwidth = Width;
|
|
|
|
int srcheight = Height;
|
|
|
|
int step_x = Height;
|
|
|
|
int step_y = 1;
|
|
|
|
FClipRect cr = {0, 0, dwidth, dheight};
|
2018-03-18 11:36:14 +00:00
|
|
|
if (style.Flags & STYLEF_RedIsAlpha) translation = nullptr; // do not apply translations to alpha textures.
|
2016-03-01 15:47:10 +00:00
|
|
|
|
|
|
|
if (ClipCopyPixelRect(&cr, xpos, ypos, pixels, srcwidth, srcheight, step_x, step_y, rotate))
|
|
|
|
{
|
|
|
|
dest += ypos + dheight * xpos;
|
|
|
|
if (translation == NULL)
|
|
|
|
{
|
|
|
|
for (int x = 0; x < srcwidth; x++)
|
|
|
|
{
|
|
|
|
int pos = x * dheight;
|
|
|
|
for (int y = 0; y < srcheight; y++, pos++)
|
|
|
|
{
|
|
|
|
// the optimizer is doing a good enough job here so there's no need to optimize this by hand
|
2017-03-09 18:54:41 +00:00
|
|
|
uint8_t v = pixels[y * step_y + x * step_x];
|
2016-03-01 15:47:10 +00:00
|
|
|
if (v != 0) dest[pos] = v;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
for (int x = 0; x < srcwidth; x++)
|
|
|
|
{
|
|
|
|
int pos = x * dheight;
|
|
|
|
for (int y = 0; y < srcheight; y++, pos++)
|
|
|
|
{
|
2017-03-09 18:54:41 +00:00
|
|
|
uint8_t v = pixels[y * step_y + x * step_x];
|
2016-03-01 15:47:10 +00:00
|
|
|
if (v != 0) dest[pos] = translation[v];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-03-31 17:20:59 +00:00
|
|
|
//==========================================================================
|
|
|
|
//
|
2016-03-01 15:47:10 +00:00
|
|
|
// Converts a texture between row-major and column-major format
|
|
|
|
// by flipping it about the X=Y axis.
|
2018-03-31 17:20:59 +00:00
|
|
|
//
|
|
|
|
//==========================================================================
|
2016-03-01 15:47:10 +00:00
|
|
|
|
2017-03-09 18:54:41 +00:00
|
|
|
void FTexture::FlipSquareBlock (uint8_t *block, int x, int y)
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
|
|
|
int i, j;
|
|
|
|
|
|
|
|
if (x != y) return;
|
|
|
|
|
|
|
|
for (i = 0; i < x; ++i)
|
|
|
|
{
|
2017-03-09 18:54:41 +00:00
|
|
|
uint8_t *corner = block + x*i + i;
|
2016-03-01 15:47:10 +00:00
|
|
|
int count = x - i;
|
|
|
|
if (count & 1)
|
|
|
|
{
|
|
|
|
count--;
|
2017-03-09 18:54:41 +00:00
|
|
|
swapvalues<uint8_t> (corner[count], corner[count*x]);
|
2016-03-01 15:47:10 +00:00
|
|
|
}
|
|
|
|
for (j = 0; j < count; j += 2)
|
|
|
|
{
|
2017-03-09 18:54:41 +00:00
|
|
|
swapvalues<uint8_t> (corner[j], corner[j*x]);
|
|
|
|
swapvalues<uint8_t> (corner[j+1], corner[(j+1)*x]);
|
2016-03-01 15:47:10 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-06-13 01:16:48 +00:00
|
|
|
void FTexture::FlipSquareBlockBgra(uint32_t *block, int x, int y)
|
|
|
|
{
|
|
|
|
int i, j;
|
|
|
|
|
|
|
|
if (x != y) return;
|
|
|
|
|
|
|
|
for (i = 0; i < x; ++i)
|
|
|
|
{
|
|
|
|
uint32_t *corner = block + x*i + i;
|
|
|
|
int count = x - i;
|
|
|
|
if (count & 1)
|
|
|
|
{
|
|
|
|
count--;
|
|
|
|
swapvalues<uint32_t>(corner[count], corner[count*x]);
|
|
|
|
}
|
|
|
|
for (j = 0; j < count; j += 2)
|
|
|
|
{
|
|
|
|
swapvalues<uint32_t>(corner[j], corner[j*x]);
|
|
|
|
swapvalues<uint32_t>(corner[j + 1], corner[(j + 1)*x]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-03-09 18:54:41 +00:00
|
|
|
void FTexture::FlipSquareBlockRemap (uint8_t *block, int x, int y, const uint8_t *remap)
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
|
|
|
int i, j;
|
2017-03-09 18:54:41 +00:00
|
|
|
uint8_t t;
|
2016-03-01 15:47:10 +00:00
|
|
|
|
|
|
|
if (x != y) return;
|
|
|
|
|
|
|
|
for (i = 0; i < x; ++i)
|
|
|
|
{
|
2017-03-09 18:54:41 +00:00
|
|
|
uint8_t *corner = block + x*i + i;
|
2016-03-01 15:47:10 +00:00
|
|
|
int count = x - i;
|
|
|
|
if (count & 1)
|
|
|
|
{
|
|
|
|
count--;
|
|
|
|
t = remap[corner[count]];
|
|
|
|
corner[count] = remap[corner[count*x]];
|
|
|
|
corner[count*x] = t;
|
|
|
|
}
|
|
|
|
for (j = 0; j < count; j += 2)
|
|
|
|
{
|
|
|
|
t = remap[corner[j]];
|
|
|
|
corner[j] = remap[corner[j*x]];
|
|
|
|
corner[j*x] = t;
|
|
|
|
t = remap[corner[j+1]];
|
|
|
|
corner[j+1] = remap[corner[(j+1)*x]];
|
|
|
|
corner[(j+1)*x] = t;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-03-09 18:54:41 +00:00
|
|
|
void FTexture::FlipNonSquareBlock (uint8_t *dst, const uint8_t *src, int x, int y, int srcpitch)
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
|
|
|
int i, j;
|
|
|
|
|
|
|
|
for (i = 0; i < x; ++i)
|
|
|
|
{
|
|
|
|
for (j = 0; j < y; ++j)
|
|
|
|
{
|
|
|
|
dst[i*y+j] = src[i+j*srcpitch];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-06-13 01:16:48 +00:00
|
|
|
void FTexture::FlipNonSquareBlockBgra(uint32_t *dst, const uint32_t *src, int x, int y, int srcpitch)
|
|
|
|
{
|
|
|
|
int i, j;
|
|
|
|
|
|
|
|
for (i = 0; i < x; ++i)
|
|
|
|
{
|
|
|
|
for (j = 0; j < y; ++j)
|
|
|
|
{
|
|
|
|
dst[i*y + j] = src[i + j*srcpitch];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-03-09 18:54:41 +00:00
|
|
|
void FTexture::FlipNonSquareBlockRemap (uint8_t *dst, const uint8_t *src, int x, int y, int srcpitch, const uint8_t *remap)
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
|
|
|
int i, j;
|
|
|
|
|
|
|
|
for (i = 0; i < x; ++i)
|
|
|
|
{
|
|
|
|
for (j = 0; j < y; ++j)
|
|
|
|
{
|
|
|
|
dst[i*y+j] = remap[src[i+j*srcpitch]];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-03-31 17:20:59 +00:00
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
2017-03-09 18:54:41 +00:00
|
|
|
void FTexture::FillBuffer(uint8_t *buff, int pitch, int height, FTextureFormat fmt)
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
2017-03-09 18:54:41 +00:00
|
|
|
const uint8_t *pix;
|
2016-03-01 15:47:10 +00:00
|
|
|
int x, y, w, h, stride;
|
|
|
|
|
|
|
|
w = GetWidth();
|
|
|
|
h = GetHeight();
|
|
|
|
|
|
|
|
switch (fmt)
|
|
|
|
{
|
|
|
|
case TEX_Pal:
|
|
|
|
case TEX_Gray:
|
2018-03-21 23:29:01 +00:00
|
|
|
pix = GetPixels(fmt == TEX_Pal? DefaultRenderStyle() : LegacyRenderStyles[STYLE_Shaded]);
|
2016-03-01 15:47:10 +00:00
|
|
|
stride = pitch - w;
|
|
|
|
for (y = 0; y < h; ++y)
|
|
|
|
{
|
2017-03-09 18:54:41 +00:00
|
|
|
const uint8_t *pix2 = pix;
|
2016-03-01 15:47:10 +00:00
|
|
|
for (x = 0; x < w; ++x)
|
|
|
|
{
|
|
|
|
*buff++ = *pix2;
|
|
|
|
pix2 += h;
|
|
|
|
}
|
|
|
|
pix++;
|
|
|
|
buff += stride;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case TEX_RGB:
|
|
|
|
{
|
|
|
|
FCopyInfo inf = {OP_OVERWRITE, BLEND_NONE, {0}, 0, 0};
|
|
|
|
FBitmap bmp(buff, pitch, pitch/4, height);
|
|
|
|
CopyTrueColorPixels(&bmp, 0, 0, 0, &inf);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
default:
|
|
|
|
I_Error("FTexture::FillBuffer: Unsupported format %d", fmt);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
//
|
|
|
|
// FTexture::CopyTrueColorPixels
|
|
|
|
//
|
|
|
|
// this is the generic case that can handle
|
|
|
|
// any properly implemented texture for software rendering.
|
|
|
|
// Its drawback is that it is limited to the base palette which is
|
|
|
|
// why all classes that handle different palettes should subclass this
|
|
|
|
// method
|
|
|
|
//
|
|
|
|
//===========================================================================
|
|
|
|
|
|
|
|
int FTexture::CopyTrueColorPixels(FBitmap *bmp, int x, int y, int rotate, FCopyInfo *inf)
|
|
|
|
{
|
|
|
|
PalEntry *palette = screen->GetPalette();
|
|
|
|
for(int i=1;i<256;i++) palette[i].a = 255; // set proper alpha values
|
2018-03-18 20:33:44 +00:00
|
|
|
bmp->CopyPixelData(x, y, GetPixels(DefaultRenderStyle()), Width, Height, Height, 1, rotate, palette, inf);
|
2016-03-01 15:47:10 +00:00
|
|
|
for(int i=1;i<256;i++) palette[i].a = 0;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2017-06-18 07:13:35 +00:00
|
|
|
int FTexture::CopyTrueColorTranslated(FBitmap *bmp, int x, int y, int rotate, PalEntry *remap, FCopyInfo *inf)
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
2018-03-18 20:33:44 +00:00
|
|
|
bmp->CopyPixelData(x, y, GetPixels(DefaultRenderStyle()), Width, Height, Height, 1, rotate, remap, inf);
|
2016-03-01 15:47:10 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-03-31 17:20:59 +00:00
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
bool FTexture::UseBasePalette()
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
FTexture *FTexture::GetRedirect(bool wantwarped)
|
|
|
|
{
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
|
|
|
FTexture *FTexture::GetRawTexture()
|
|
|
|
{
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
|
|
|
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
|
|
|
{
|
|
|
|
PalEntry col;
|
|
|
|
|
|
|
|
if (!bSWSkyColorDone)
|
|
|
|
{
|
|
|
|
bSWSkyColorDone = true;
|
|
|
|
|
|
|
|
FBitmap bitmap;
|
|
|
|
bitmap.Create(GetWidth(), GetHeight());
|
|
|
|
CopyTrueColorPixels(&bitmap, 0, 0);
|
|
|
|
int w = GetWidth();
|
|
|
|
int h = GetHeight();
|
|
|
|
|
|
|
|
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()
|
|
|
|
{
|
|
|
|
const FTexture::Span *span;
|
|
|
|
int maxy = 0, miny = GetHeight();
|
|
|
|
|
|
|
|
for (int i = 0; i < GetWidth(); ++i)
|
|
|
|
{
|
2018-03-18 20:33:44 +00:00
|
|
|
GetColumn(DefaultRenderStyle(), i, &span);
|
2017-03-17 22:08:22 +00:00
|
|
|
while (span->Length != 0)
|
|
|
|
{
|
|
|
|
if (span->TopOffset < miny)
|
|
|
|
{
|
|
|
|
miny = span->TopOffset;
|
|
|
|
}
|
|
|
|
if (span->TopOffset + span->Length > maxy)
|
|
|
|
{
|
|
|
|
maxy = span->TopOffset + span->Length;
|
|
|
|
}
|
|
|
|
span++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// Scale maxy before returning it
|
|
|
|
maxy = int((maxy * 2) / Scale.Y);
|
|
|
|
maxy = (maxy >> 1) + (maxy & 1);
|
|
|
|
return maxy;
|
|
|
|
}
|
|
|
|
|
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-03-31 17:20:59 +00:00
|
|
|
for (int i = 0; i < countof(autosearchpaths); i++)
|
|
|
|
{
|
|
|
|
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
|
|
|
|
if (UseBasePalette() && TexMan.HasGlobalBrightmap &&
|
|
|
|
UseType != ETextureType::Decal && UseType != ETextureType::MiscPatch && UseType != ETextureType::FontChar &&
|
|
|
|
Brightmap == NULL && bWarped == 0
|
|
|
|
)
|
|
|
|
{
|
|
|
|
// May have one - let's check when we use this texture
|
|
|
|
const uint8_t *texbuf = GetPixels(DefaultRenderStyle());
|
|
|
|
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());
|
|
|
|
Brightmap = CreateBrightmapTexture(this);
|
|
|
|
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)
|
|
|
|
{
|
|
|
|
int w = Width, h = Height;
|
|
|
|
auto buffer = new uint8_t[w * h];
|
|
|
|
if (buffer)
|
|
|
|
{
|
|
|
|
FillBuffer(buffer, w, h, TEX_RGB);
|
|
|
|
GlowColor = averageColor((uint32_t *)buffer, w*h, 153);
|
|
|
|
delete[] buffer;
|
|
|
|
}
|
|
|
|
|
|
|
|
// 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
|
|
|
|
//
|
|
|
|
//===========================================================================
|
|
|
|
|
|
|
|
unsigned char * FTexture::CreateTexBuffer(int translation, int & w, int & h, int flags)
|
|
|
|
{
|
|
|
|
unsigned char * buffer;
|
|
|
|
int W, H;
|
|
|
|
int isTransparent = -1;
|
|
|
|
|
|
|
|
|
2018-04-01 14:32:37 +00:00
|
|
|
if ((flags & CTF_CheckHires) && translation != STRange_AlphaTexture)
|
2018-04-01 12:38:48 +00:00
|
|
|
{
|
|
|
|
buffer = LoadHiresTexture(&w, &h);
|
|
|
|
if (buffer)
|
|
|
|
{
|
|
|
|
return buffer;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int exx = !!(flags & CTF_Expand);
|
|
|
|
|
|
|
|
W = w = GetWidth() + 2 * exx;
|
|
|
|
H = h = GetHeight() + 2 * exx;
|
|
|
|
|
|
|
|
|
|
|
|
buffer = new unsigned char[W*(H + 1) * 4];
|
|
|
|
memset(buffer, 0, W * (H + 1) * 4);
|
|
|
|
|
|
|
|
FBitmap bmp(buffer, W * 4, W, H);
|
|
|
|
|
|
|
|
if (translation <= 0 || translation >= STRange_Min)
|
|
|
|
{
|
|
|
|
// Allow creation of desaturated or special-colormapped textures for the legacy renderer.
|
|
|
|
FCopyInfo inf = { OP_COPY, BLEND_NONE,{ 0 }, 0, 0 };
|
|
|
|
if (translation >= STRange_Desaturate && translation < STRange_Desaturate + 31) // there are 31 ranges of desaturations available
|
|
|
|
{
|
|
|
|
inf.blend = (EBlend)(BLEND_DESATURATE1 + translation - STRange_Desaturate);
|
|
|
|
}
|
2018-04-01 16:45:27 +00:00
|
|
|
else if (translation >= STRange_Specialcolormap && translation < STRange_Specialcolormap + (int)SpecialColormaps.Size())
|
2018-04-01 12:38:48 +00:00
|
|
|
{
|
|
|
|
inf.blend = (EBlend)(BLEND_SPECIALCOLORMAP1 + translation - STRange_Specialcolormap);
|
|
|
|
}
|
|
|
|
|
|
|
|
int trans = CopyTrueColorPixels(&bmp, exx, exx, 0, translation >= STRange_Min ? &inf : nullptr);
|
|
|
|
CheckTrans(buffer, W*H, trans);
|
|
|
|
isTransparent = bTranslucent;
|
|
|
|
// alpha texture for legacy mode
|
|
|
|
if (translation == STRange_AlphaTexture)
|
|
|
|
{
|
|
|
|
for (int i = 0; i < W*H; i++)
|
|
|
|
{
|
|
|
|
int b = buffer[4 * i];
|
|
|
|
int g = buffer[4 * i + 1];
|
|
|
|
int r = buffer[4 * i + 2];
|
|
|
|
int gray = Luminance(r, g, b);
|
|
|
|
buffer[4 * i] = 255;
|
|
|
|
buffer[4 * i + 1] = 255;
|
|
|
|
buffer[4 * i + 2] = 255;
|
|
|
|
buffer[4 * i + 3] = (buffer[4 * i + 3] * gray) >> 8;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// When using translations everything must be mapped to the base palette.
|
|
|
|
// so use CopyTrueColorTranslated
|
|
|
|
CopyTrueColorTranslated(&bmp, exx, exx, 0, FUniquePalette::GetPalette(translation));
|
|
|
|
isTransparent = 0;
|
|
|
|
// This is not conclusive for setting the texture's transparency info.
|
|
|
|
}
|
|
|
|
|
|
|
|
// [BB] The hqnx upsampling (not the scaleN one) destroys partial transparency, don't upsamle textures using it.
|
|
|
|
// [BB] Potentially upsample the buffer.
|
|
|
|
if (flags & CTF_ProcessData)
|
|
|
|
{
|
|
|
|
buffer = CreateUpsampledTextureBuffer(buffer, W, H, w, h, !!isTransparent);
|
|
|
|
ProcessData(buffer, w, h, false);
|
|
|
|
}
|
|
|
|
return buffer;
|
|
|
|
}
|
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
//
|
|
|
|
// Dummy texture for the 0-entry.
|
|
|
|
//
|
|
|
|
//===========================================================================
|
|
|
|
|
|
|
|
bool FTexture::GetTranslucency()
|
|
|
|
{
|
|
|
|
if (bTranslucent == -1)
|
|
|
|
{
|
|
|
|
if (!bHasCanvas)
|
|
|
|
{
|
|
|
|
int w, h;
|
|
|
|
unsigned char *buffer = CreateTexBuffer(0, w, h);
|
|
|
|
delete[] buffer;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
bTranslucent = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return !!bTranslucent;
|
|
|
|
}
|
|
|
|
|
2018-03-31 17:20:59 +00:00
|
|
|
//===========================================================================
|
|
|
|
//
|
|
|
|
// Dummy texture for the 0-entry.
|
|
|
|
//
|
|
|
|
//===========================================================================
|
2016-03-01 15:47:10 +00:00
|
|
|
|
|
|
|
FDummyTexture::FDummyTexture ()
|
|
|
|
{
|
|
|
|
Width = 64;
|
|
|
|
Height = 64;
|
|
|
|
HeightBits = 6;
|
|
|
|
WidthBits = 6;
|
|
|
|
WidthMask = 63;
|
2018-03-25 18:26:16 +00:00
|
|
|
UseType = ETextureType::Null;
|
2016-03-01 15:47:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void FDummyTexture::SetSize (int width, int height)
|
|
|
|
{
|
|
|
|
Width = width;
|
|
|
|
Height = height;
|
|
|
|
CalcBitSize ();
|
|
|
|
}
|
|
|
|
|
2018-03-18 20:33:44 +00:00
|
|
|
// These only get called from the texture precacher which discards the result.
|
|
|
|
const uint8_t *FDummyTexture::GetColumn(FRenderStyle style, unsigned int column, const Span **spans_out)
|
|
|
|
{
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
const uint8_t *FDummyTexture::GetPixels(FRenderStyle style)
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
2018-03-18 13:18:42 +00:00
|
|
|
return nullptr;
|
2016-03-01 15:47:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
//==========================================================================
|
|
|
|
//
|
|
|
|
// Debug stuff
|
|
|
|
//
|
|
|
|
//==========================================================================
|
|
|
|
|
|
|
|
#ifdef _DEBUG
|
|
|
|
// Prints the spans generated for a texture. Only needed for debugging.
|
|
|
|
CCMD (printspans)
|
|
|
|
{
|
|
|
|
if (argv.argc() != 2)
|
|
|
|
return;
|
|
|
|
|
2018-03-25 18:26:16 +00:00
|
|
|
FTextureID picnum = TexMan.CheckForTexture (argv[1], ETextureType::Any);
|
2016-03-01 15:47:10 +00:00
|
|
|
if (!picnum.Exists())
|
|
|
|
{
|
|
|
|
Printf ("Unknown texture %s\n", argv[1]);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
FTexture *tex = TexMan[picnum];
|
|
|
|
for (int x = 0; x < tex->GetWidth(); ++x)
|
|
|
|
{
|
|
|
|
const FTexture::Span *spans;
|
|
|
|
Printf ("%4d:", x);
|
2018-03-18 20:33:44 +00:00
|
|
|
tex->GetColumn(DefaultRenderStyle(), x, &spans);
|
2016-03-01 15:47:10 +00:00
|
|
|
while (spans->Length != 0)
|
|
|
|
{
|
|
|
|
Printf (" (%4d,%4d)", spans->TopOffset, spans->TopOffset+spans->Length-1);
|
|
|
|
spans++;
|
|
|
|
}
|
|
|
|
Printf ("\n");
|
|
|
|
}
|
|
|
|
}
|
2018-03-23 22:04:30 +00:00
|
|
|
|
2016-03-01 15:47:10 +00:00
|
|
|
#endif
|
2018-03-23 22:04:30 +00:00
|
|
|
|
|
|
|
|