2020-04-11 22:04:02 +00:00
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include <stdint.h>
|
|
|
|
#include "memarena.h"
|
|
|
|
#include "palentry.h"
|
2023-08-23 18:36:19 +00:00
|
|
|
#include "files.h"
|
2020-05-23 12:40:54 +00:00
|
|
|
|
2020-04-05 20:51:53 +00:00
|
|
|
enum
|
|
|
|
{
|
|
|
|
TRANSLATION_Internal = 0
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2020-04-11 22:04:02 +00:00
|
|
|
struct FRemapTable
|
|
|
|
{
|
|
|
|
FRemapTable(int count = 256) { NumEntries = count; }
|
2022-08-26 16:28:22 +00:00
|
|
|
|
2020-04-11 22:04:02 +00:00
|
|
|
FRemapTable(const FRemapTable& o) = default;
|
2022-08-26 16:28:22 +00:00
|
|
|
FRemapTable& operator=(const FRemapTable& o) = default;
|
2020-04-11 22:04:02 +00:00
|
|
|
|
2023-07-22 07:54:00 +00:00
|
|
|
bool operator==(const FRemapTable& o) const;
|
2020-04-11 22:04:02 +00:00
|
|
|
void MakeIdentity();
|
|
|
|
bool IsIdentity() const;
|
|
|
|
bool AddIndexRange(int start, int end, int pal1, int pal2);
|
|
|
|
bool AddColorRange(int start, int end, int r1, int g1, int b1, int r2, int g2, int b2);
|
|
|
|
bool AddDesaturation(int start, int end, double r1, double g1, double b1, double r2, double g2, double b2);
|
|
|
|
bool AddColourisation(int start, int end, int r, int g, int b);
|
|
|
|
bool AddTint(int start, int end, int r, int g, int b, int amount);
|
|
|
|
bool AddToTranslation(const char* range);
|
2020-04-12 05:50:24 +00:00
|
|
|
bool AddColors(int start, int count, const uint8_t*, int trans_color = 0);
|
2020-04-11 22:04:02 +00:00
|
|
|
|
|
|
|
uint8_t Remap[256]; // For the software renderer
|
|
|
|
PalEntry Palette[256]; // The ideal palette this maps to
|
|
|
|
int crc32;
|
|
|
|
int Index;
|
2021-05-24 18:29:22 +00:00
|
|
|
int NumEntries; // # of elements in this table (usually 256)
|
2020-04-11 22:04:02 +00:00
|
|
|
bool Inactive = false; // This table is inactive and should be treated as if it was passed as NULL
|
2020-05-26 21:12:04 +00:00
|
|
|
bool TwodOnly = false; // Only used for 2D rendering
|
2020-04-05 20:51:53 +00:00
|
|
|
bool ForFont = false; // Mark font translations because they may require different handling than the ones for sprites-
|
2020-07-05 08:15:55 +00:00
|
|
|
bool NoTransparency = false; // This palette has no transparent index and must be excluded from all treatment for that.
|
2020-04-11 22:04:02 +00:00
|
|
|
|
|
|
|
private:
|
|
|
|
};
|
|
|
|
|
2023-11-09 17:48:59 +00:00
|
|
|
|
2023-11-09 18:22:32 +00:00
|
|
|
|
|
|
|
// outside facing translation ID
|
2023-11-09 17:48:59 +00:00
|
|
|
struct FTranslationID
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
FTranslationID() = default;
|
|
|
|
|
|
|
|
private:
|
|
|
|
constexpr FTranslationID(int id) : ID(id)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
public:
|
|
|
|
static constexpr FTranslationID fromInt(int i)
|
|
|
|
{
|
|
|
|
return FTranslationID(i);
|
|
|
|
}
|
|
|
|
FTranslationID(const FTranslationID& other) = default;
|
|
|
|
FTranslationID& operator=(const FTranslationID& other) = default;
|
|
|
|
bool operator !=(FTranslationID other) const
|
|
|
|
{
|
|
|
|
return ID != other.ID;
|
|
|
|
}
|
|
|
|
bool operator ==(FTranslationID other) const
|
|
|
|
{
|
|
|
|
return ID == other.ID;
|
|
|
|
}
|
|
|
|
constexpr int index() const
|
|
|
|
{
|
|
|
|
return ID;
|
|
|
|
}
|
|
|
|
constexpr bool isvalid() const
|
|
|
|
{
|
2023-11-09 18:22:32 +00:00
|
|
|
return ID >= 0;
|
2023-11-09 17:48:59 +00:00
|
|
|
}
|
|
|
|
private:
|
|
|
|
|
|
|
|
int ID;
|
|
|
|
};
|
|
|
|
|
2023-11-09 18:22:32 +00:00
|
|
|
constexpr FTranslationID NO_TRANSLATION = FTranslationID::fromInt(0);
|
|
|
|
constexpr FTranslationID INVALID_TRANSLATION = FTranslationID::fromInt(-1);
|
|
|
|
|
2020-04-11 22:04:02 +00:00
|
|
|
// A class that initializes unusued pointers to NULL. This is used so that when
|
|
|
|
// the TAutoGrowArray below is expanded, the new elements will be NULLed.
|
|
|
|
class FRemapTablePtr
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
FRemapTablePtr() throw() : Ptr(0) {}
|
|
|
|
FRemapTablePtr(FRemapTable* p) throw() : Ptr(p) {}
|
|
|
|
FRemapTablePtr(const FRemapTablePtr& p) throw() : Ptr(p.Ptr) {}
|
|
|
|
operator FRemapTable* () const throw() { return Ptr; }
|
|
|
|
FRemapTablePtr& operator= (FRemapTable* p) throw() { Ptr = p; return *this; }
|
|
|
|
FRemapTablePtr& operator= (FRemapTablePtr& p) throw() { Ptr = p.Ptr; return *this; }
|
|
|
|
FRemapTable& operator*() const throw() { return *Ptr; }
|
|
|
|
FRemapTable* operator->() const throw() { return Ptr; }
|
|
|
|
private:
|
|
|
|
FRemapTable* Ptr = nullptr;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2020-08-15 08:22:07 +00:00
|
|
|
enum
|
|
|
|
{
|
|
|
|
TRANSLATION_SHIFT = 16,
|
|
|
|
TRANSLATION_MASK = ((1 << TRANSLATION_SHIFT) - 1),
|
|
|
|
TRANSLATIONTYPE_MASK = (255 << TRANSLATION_SHIFT)
|
|
|
|
};
|
2020-04-11 22:04:02 +00:00
|
|
|
|
2023-11-09 18:22:32 +00:00
|
|
|
inline constexpr FTranslationID TRANSLATION(uint8_t a, uint32_t b)
|
2020-04-11 22:04:02 +00:00
|
|
|
{
|
2023-11-09 18:22:32 +00:00
|
|
|
return FTranslationID::fromInt((a << TRANSLATION_SHIFT) | b);
|
2020-04-11 22:04:02 +00:00
|
|
|
}
|
2023-11-09 22:00:36 +00:00
|
|
|
inline constexpr FTranslationID MakeLuminosityTranslation(int range, uint8_t min, uint8_t max)
|
2021-05-24 11:16:50 +00:00
|
|
|
{
|
|
|
|
// ensure that the value remains positive.
|
2023-11-09 22:00:36 +00:00
|
|
|
return FTranslationID::fromInt( (1 << 30) | ((range&0x3fff) << 16) | (min << 8) | max );
|
2021-05-24 11:16:50 +00:00
|
|
|
}
|
|
|
|
|
2023-11-09 18:22:32 +00:00
|
|
|
inline constexpr bool IsLuminosityTranslation(FTranslationID trans)
|
|
|
|
{
|
|
|
|
return trans.index() > 0 && (trans.index() & (1 << 30));
|
|
|
|
}
|
|
|
|
|
2021-05-24 11:16:50 +00:00
|
|
|
inline constexpr bool IsLuminosityTranslation(int trans)
|
|
|
|
{
|
|
|
|
return trans > 0 && (trans & (1 << 30));
|
|
|
|
}
|
|
|
|
|
2023-11-09 18:22:32 +00:00
|
|
|
inline constexpr int GetTranslationType(FTranslationID trans)
|
|
|
|
{
|
|
|
|
assert(!IsLuminosityTranslation(trans));
|
|
|
|
return (trans.index() & TRANSLATIONTYPE_MASK) >> TRANSLATION_SHIFT;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline constexpr int GetTranslationIndex(FTranslationID trans)
|
|
|
|
{
|
|
|
|
assert(!IsLuminosityTranslation(trans));
|
|
|
|
return (trans.index() & TRANSLATION_MASK);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline constexpr int GetTranslationType(int trans)
|
2020-04-11 22:04:02 +00:00
|
|
|
{
|
2021-05-24 11:16:50 +00:00
|
|
|
assert(!IsLuminosityTranslation(trans));
|
2020-04-11 22:04:02 +00:00
|
|
|
return (trans & TRANSLATIONTYPE_MASK) >> TRANSLATION_SHIFT;
|
|
|
|
}
|
2021-05-24 11:16:50 +00:00
|
|
|
|
2023-11-09 18:22:32 +00:00
|
|
|
inline constexpr int GetTranslationIndex(int trans)
|
2020-04-11 22:04:02 +00:00
|
|
|
{
|
2021-05-24 11:16:50 +00:00
|
|
|
assert(!IsLuminosityTranslation(trans));
|
2020-04-11 22:04:02 +00:00
|
|
|
return (trans & TRANSLATION_MASK);
|
|
|
|
}
|
|
|
|
|
|
|
|
class PaletteContainer
|
|
|
|
{
|
2020-04-11 22:18:12 +00:00
|
|
|
public:
|
|
|
|
PalEntry BaseColors[256]; // non-gamma corrected palette
|
2020-04-12 06:21:08 +00:00
|
|
|
PalEntry RawColors[256]; // colors as read from the game data without the transparancy remap applied
|
2020-04-11 22:18:12 +00:00
|
|
|
uint8_t Remap[256]; // remap original palette indices to in-game indices
|
|
|
|
|
|
|
|
uint8_t WhiteIndex; // white in original palette index
|
|
|
|
uint8_t BlackIndex; // black in original palette index
|
2020-04-12 06:21:08 +00:00
|
|
|
|
|
|
|
bool HasGlobalBrightmap;
|
|
|
|
FRemapTable GlobalBrightmap;
|
2020-04-12 06:21:50 +00:00
|
|
|
FRemapTable GrayRamp;
|
|
|
|
FRemapTable GrayscaleMap;
|
|
|
|
FRemapTable IceMap; // This is used by the texture compositor so it must be globally accessible.
|
|
|
|
uint8_t GrayMap[256];
|
2020-04-12 06:21:08 +00:00
|
|
|
|
2020-05-24 21:26:47 +00:00
|
|
|
TArray<FRemapTable*> uniqueRemaps;
|
|
|
|
|
2020-04-11 22:18:12 +00:00
|
|
|
private:
|
2020-04-11 22:04:02 +00:00
|
|
|
FMemArena remapArena;
|
|
|
|
TArray<TAutoGrowArray<FRemapTablePtr, FRemapTable*>> TranslationTables;
|
|
|
|
public:
|
2021-08-14 08:26:04 +00:00
|
|
|
void Init(int numslots, const uint8_t *indexmap); // This cannot be a constructor!!!
|
2020-04-12 05:50:24 +00:00
|
|
|
void SetPalette(const uint8_t* colors, int transparent_index = -1);
|
2020-04-11 22:04:02 +00:00
|
|
|
void Clear();
|
2020-05-23 12:40:54 +00:00
|
|
|
int DetermineTranslucency(FileReader& file);
|
2020-04-11 22:04:02 +00:00
|
|
|
FRemapTable* AddRemap(FRemapTable* remap);
|
2023-11-09 18:22:32 +00:00
|
|
|
void UpdateTranslation(FTranslationID trans, FRemapTable* remap);
|
|
|
|
FTranslationID AddTranslation(int slot, FRemapTable* remap, int count = 1);
|
|
|
|
void CopyTranslation(FTranslationID dest, FTranslationID src);
|
|
|
|
FTranslationID StoreTranslation(int slot, FRemapTable* remap);
|
|
|
|
FRemapTable* TranslationToTable(int translation) const;
|
2023-11-09 22:00:36 +00:00
|
|
|
FRemapTable* TranslationToTable(FTranslationID translation) const
|
|
|
|
{
|
|
|
|
return TranslationToTable(translation.index());
|
|
|
|
}
|
|
|
|
|
2020-04-12 06:21:08 +00:00
|
|
|
void GenerateGlobalBrightmapFromColormap(const uint8_t* cmapdata, int numlevels);
|
2020-04-11 22:04:02 +00:00
|
|
|
|
|
|
|
void PushIdentityTable(int slot)
|
|
|
|
{
|
|
|
|
AddTranslation(slot, nullptr);
|
|
|
|
}
|
|
|
|
|
2023-11-09 18:22:32 +00:00
|
|
|
FRemapTable* GetTranslation(int slot, int index) const
|
2020-04-11 22:04:02 +00:00
|
|
|
{
|
2022-06-06 09:45:02 +00:00
|
|
|
if (TranslationTables.Size() <= (unsigned)slot) return nullptr;
|
2020-04-11 22:04:02 +00:00
|
|
|
return TranslationTables[slot].GetVal(index);
|
|
|
|
}
|
|
|
|
|
|
|
|
void ClearTranslationSlot(int slot)
|
|
|
|
{
|
2022-06-06 09:45:02 +00:00
|
|
|
if (TranslationTables.Size() <= (unsigned)slot) return;
|
2020-04-11 22:04:02 +00:00
|
|
|
TranslationTables[slot].Clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned NumTranslations(int slot) const
|
|
|
|
{
|
2022-06-06 09:45:02 +00:00
|
|
|
if (TranslationTables.Size() <= (unsigned)slot) return 0;
|
2020-04-11 22:04:02 +00:00
|
|
|
return TranslationTables[slot].Size();
|
|
|
|
}
|
|
|
|
|
2023-11-09 18:22:32 +00:00
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
struct LuminosityTranslationDesc
|
|
|
|
{
|
|
|
|
int colorrange;
|
|
|
|
int lum_min;
|
|
|
|
int lum_max;
|
|
|
|
|
|
|
|
static LuminosityTranslationDesc fromInt(int translation)
|
|
|
|
{
|
|
|
|
LuminosityTranslationDesc t;
|
|
|
|
t.colorrange = (translation >> 16) & 0x3fff;
|
|
|
|
t.lum_min = (translation >> 8) & 0xff;
|
|
|
|
t.lum_max = translation & 0xff;
|
|
|
|
return t;
|
|
|
|
}
|
|
|
|
|
|
|
|
static LuminosityTranslationDesc fromID(FTranslationID translation)
|
|
|
|
{
|
|
|
|
return fromInt(translation.index());
|
|
|
|
}
|
2020-04-11 22:04:02 +00:00
|
|
|
};
|
|
|
|
|
2020-04-11 22:18:12 +00:00
|
|
|
extern PaletteContainer GPalette;
|
2020-04-11 22:04:02 +00:00
|
|
|
|