#pragma once #include "textures.h" // picanm[].sf: // |bit(1<<7) // |animtype|animtype|texhitscan|nofullbright|speed|speed|speed|speed| enum AnimFlags { PICANM_ANIMTYPE_NONE = 0, PICANM_ANIMTYPE_OSC = (1 << 6), PICANM_ANIMTYPE_FWD = (2 << 6), PICANM_ANIMTYPE_BACK = (3 << 6), PICANM_ANIMTYPE_SHIFT = 6, PICANM_ANIMTYPE_MASK = (3 << 6), // must be 192 PICANM_MISC_MASK = (3 << 4), PICANM_TEXHITSCAN_BIT = (2 << 4), PICANM_NOFULLBRIGHT_BIT = (1 << 4), PICANM_ANIMSPEED_MASK = 15, // must be 15 }; enum { MAXTILES = 30720, MAXUSERTILES = (MAXTILES-16) // reserve 16 tiles at the end }; enum class ReplacementType : int { Art, Writable, Restorable, Canvas }; // NOTE: If the layout of this struct is changed, loadpics() must be modified // accordingly. struct picanm_t { uint8_t num; // animate number uint8_t sf; // anim. speed and flags uint8_t extra; void Clear() { extra = sf = num = 0; } }; picanm_t tileConvertAnimFormat(int32_t const picanmdisk, int* lo, int* to); struct rottile_t { int16_t newtile; int16_t owner; }; struct HightileReplacement { FTexture* faces[6]; // only one gets used by a texture, the other 5 are for skyboxes only vec2f_t scale; float alphacut, specpower, specfactor; uint16_t palnum, flags; }; class FTileTexture : public FTexture { public: FTileTexture() = default; void SetName(const char* name) { Name = name; } FBitmap GetBgraBitmap(const PalEntry* remap, int* ptrans) override; void Create8BitPixels(uint8_t* buffer) override; }; //========================================================================== // // A tile coming from an ART file. // //========================================================================== class FArtTile : public FTileTexture { const TArray& RawPixels; const uint32_t Offset; public: FArtTile(const TArray& backingstore, uint32_t offset, int width, int height, int picanm) : RawPixels(backingstore), Offset(offset) { SetSize(width, height); } const uint8_t* Get8BitPixels() override { return RawPixels.Data() ? RawPixels.Data() + Offset : nullptr; } }; //========================================================================== // // A tile with its own pixel buffer // //========================================================================== class FLooseTile : public FTileTexture { TArray RawPixels; public: FLooseTile(TArray &store, int width, int height) { RawPixels = std::move(store); SetSize(width, height); } const uint8_t* Get8BitPixels() override { return RawPixels.Data(); } }; //========================================================================== // // A non-existent tile // //========================================================================== class FDummyTile : public FTileTexture { public: FDummyTile(int width, int height) { SetSize(width, height); } const uint8_t* Get8BitPixels() override { return NULL; } }; //========================================================================== // // A tile with a writable surface // //========================================================================== class FWritableTile : public FTileTexture { protected: TArray buffer; public: const uint8_t* Get8BitPixels() override { return buffer.Data(); } uint8_t* GetWritableBuffer() override { // Todo: Invalidate all hardware textures depending on this. return buffer.Data(); } bool Resize(int w, int h) { if (w <= 0 || h <= 0) { buffer.Reset(); return false; } else { SetSize(w, h); buffer.Resize(w * h); return true; } } }; //========================================================================== // // A tile with a writable surface // //========================================================================== class FRestorableTile : public FWritableTile { FTexture* Base; public: FRestorableTile(FTexture* base) { Base = base; CopySize(base); Resize(GetWidth(), GetHeight()); Reload(); } void Reload() override { Base->Create8BitPixels(buffer.Data()); } }; //========================================================================== // // One ART file. // //========================================================================== struct BuildArtFile { FString filename; TArray RawData; BuildArtFile() = default; BuildArtFile(const BuildArtFile&) = delete; BuildArtFile& operator=(const BuildArtFile&) = delete; BuildArtFile(const BuildArtFile&& other) { filename = std::move(other.filename); RawData = std::move(other.RawData); } BuildArtFile& operator=(const BuildArtFile&& other) { filename = std::move(other.filename); RawData = std::move(other.RawData); return *this; } }; //========================================================================== // // THe tile container // //========================================================================== struct RawCacheNode { TArray data; uint64_t lastUseTime; }; struct TileDesc { FTexture* texture; // the currently active tile FTexture* backup; // original backup for map tiles RawCacheNode rawCache; // this is needed for hitscan testing to avoid reloading the texture each time. picanm_t picanm; // animation descriptor picanm_t picanmbackup; // animation descriptor backup when using map tiles rottile_t RotTile;// = { -1,-1 }; TArray Hightiles; ReplacementType replacement; }; struct BuildTiles { FTexture* Placeholder; TDeletingArray ArtFiles; TileDesc tiledata[MAXTILES]; TDeletingArray PerMapArtFiles; TDeletingArray AllTiles; // This is for deleting tiles when shutting down. TDeletingArray AllMapTiles; // Same for map tiles; TMap textures; TArray addedArt; TMap TextureToTile; BuildTiles() { Placeholder = new FDummyTile(0, 0); for (auto& tile : tiledata) { tile.backup = tile.texture = Placeholder; tile.RotTile = { -1,-1 }; tile.picanm = {}; tile.replacement = ReplacementType::Art; } } ~BuildTiles() { CloseAll(); } void CloseAll(); FTexture* GetTexture(const char* path); void AddTile(int tilenum, FTexture* tex, bool permap = false); void AddTiles(int firsttile, TArray& store, bool permap); void AddFile(BuildArtFile* bfd, bool permap) { if (!permap) ArtFiles.Push(bfd); else PerMapArtFiles.Push(bfd); } int FindFile(const FString& filename) { return ArtFiles.FindEx([filename](const BuildArtFile* element) { return filename.CompareNoCase(element->filename) == 0; }); } int LoadArtFile(const char* file, bool mapart = false, int firsttile = -1); void CloseAllMapArt(); void LoadArtSet(const char* filename); void AddArt(TArray& art) { addedArt = std::move(art); } int GetTileIndex(FTexture* tex) { auto p = TextureToTile.CheckKey(tex); return p ? *p : -1; } void SetupReverseTileMap() { TextureToTile.Clear(); for (int i = 0; i < MAXTILES; i++) { if (tiledata[i].texture != nullptr && tiledata[i].texture != Placeholder) TextureToTile.Insert(tiledata[i].texture, i); } } FTexture* ValidateCustomTile(int tilenum, ReplacementType type); int32_t artLoadFiles(const char* filename); uint8_t* tileMakeWritable(int num); uint8_t* tileCreate(int tilenum, int width, int height); void tileSetExternal(int tilenum, int width, int height, uint8_t* data); int findUnusedTile(void); int tileCreateRotated(int owner); void ClearTextureCache(bool artonly = false); void InvalidateTile(int num); void MakeCanvas(int tilenum, int width, int height); HightileReplacement* FindReplacement(int picnum, int palnum, bool skybox = false); void AddReplacement(int picnum, const HightileReplacement&); void DeleteReplacement(int picnum, int palnum); void DeleteReplacements(int picnum) { assert(picnum < MAXTILES); tiledata[picnum].Hightiles.Clear(); } }; int tileGetCRC32(int tileNum); int tileImportFromTexture(const char* fn, int tilenum, int alphacut, int istexture); void tileCopy(int tile, int tempsource, int temppal, int xoffset, int yoffset, int flags); void tileSetDummy(int tile, int width, int height); void tileDelete(int tile); void tileRemoveReplacement(int tile); bool tileLoad(int tileNum); void artClearMapArt(void); void artSetupMapArt(const char* filename); void tileSetAnim(int tile, const picanm_t& anm); int tileSetHightileReplacement(int picnum, int palnum, const char *filen, float alphacut, float xscale, float yscale, float specpower, float specfactor, uint8_t flags); int tileSetSkybox(int picnum, int palnum, const char **facenames, int flags ); int tileDeleteReplacement(int picnum, int palnum); void tileCopySection(int tilenum1, int sx1, int sy1, int xsiz, int ysiz, int tilenum2, int sx2, int sy2); extern BuildTiles TileFiles; inline bool tileCheck(int num) { auto tex = TileFiles.tiledata[num].texture; return tex->GetWidth() > 0 && tex->GetHeight() > 0; } inline const uint8_t* tilePtr(int num) { auto tex = TileFiles.tiledata[num].texture; auto p = tex->Get8BitPixels(); if (p) return p; return tex->CachedPixels.Data(); } inline uint8_t* tileData(int num) { auto tex = TileFiles.tiledata[num].texture; return tex->GetWritableBuffer(); } // Some hacks to allow accessing the no longer existing arrays as if they still were arrays to avoid changing hundreds of lines of code. struct TileSiz { const vec2_16_t operator[](size_t index) { assert(index < MAXTILES); vec2_16_t v = { (int16_t)TileFiles.tiledata[index].texture->GetDisplayWidth(), (int16_t)TileFiles.tiledata[index].texture->GetDisplayHeight() }; return v; } }; extern TileSiz tilesiz; struct PicAnm { picanm_t& operator[](size_t index) { assert(index < MAXTILES); return TileFiles.tiledata[index].picanm; } }; extern PicAnm picanm; // Helpers to read the refactored tilesiz array. inline int tileWidth(int num) { assert(num < MAXTILES); return TileFiles.tiledata[num].texture->GetDisplayWidth(); } inline int tileHeight(int num) { assert(num < MAXTILES); return TileFiles.tiledata[num].texture->GetDisplayHeight(); } inline int tileLeftOffset(int num) { assert(num < MAXTILES); return TileFiles.tiledata[num].texture->GetDisplayLeftOffset(); } inline int tileTopOffset(int num) { assert(num < MAXTILES); return TileFiles.tiledata[num].texture->GetDisplayTopOffset(); } inline int widthBits(int num) { int w = tileWidth(num); int j = 15; while ((j > 1) && ((1 << j) > w)) j--; return j; } inline int heightBits(int num) { int w = tileHeight(num); int j = 15; while ((j > 1) && ((1 << j) > w)) j--; return j; } inline rottile_t& RotTile(int tile) { assert(tile < MAXTILES); return TileFiles.tiledata[tile].RotTile; } inline void tileInvalidate(int tilenume, int32_t, int32_t) { TileFiles.InvalidateTile(tilenume); } inline FTexture* tileGetTexture(int tile) { assert(tile < MAXTILES); return TileFiles.tiledata[tile].texture; }