2021-04-06 13:07:12 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
** parsefuncs.h
|
|
|
|
** handlers for .def parser
|
|
|
|
** only to be included by the actual parser
|
|
|
|
**
|
|
|
|
**---------------------------------------------------------------------------
|
|
|
|
** Copyright 2021 Christoph Oelckers
|
|
|
|
** 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.
|
|
|
|
**---------------------------------------------------------------------------
|
|
|
|
**
|
|
|
|
**
|
|
|
|
*/
|
|
|
|
|
2021-04-07 22:47:07 +00:00
|
|
|
int tileSetHightileReplacement(int picnum, int palnum, const char* filename, float alphacut, float xscale, float yscale, float specpower, float specfactor);
|
|
|
|
int tileSetSkybox(int picnum, int palnum, FString* facenames);
|
|
|
|
void tileRemoveReplacement(int num);
|
2021-04-12 22:28:51 +00:00
|
|
|
void AddUserMapHack(usermaphack_t&);
|
2021-04-07 22:47:07 +00:00
|
|
|
|
2021-04-07 19:59:47 +00:00
|
|
|
//===========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//===========================================================================
|
|
|
|
|
2021-04-08 18:03:50 +00:00
|
|
|
template<int cnt>
|
|
|
|
void parseSkip(FScanner& sc, FScriptPosition& pos)
|
|
|
|
{
|
2021-04-10 14:35:29 +00:00
|
|
|
for (int i = 0; i < cnt; i++) if (!sc.GetString()) return;
|
2021-04-08 18:03:50 +00:00
|
|
|
}
|
|
|
|
|
2021-04-09 17:57:56 +00:00
|
|
|
void parseDefine(FScanner& sc, FScriptPosition& pos)
|
|
|
|
{
|
|
|
|
FString name;
|
|
|
|
if (!sc.GetString(name)) return;
|
|
|
|
if (!sc.GetNumber()) return;
|
|
|
|
sc.AddSymbol(name, sc.Number);
|
|
|
|
}
|
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//===========================================================================
|
|
|
|
|
2021-04-07 22:47:07 +00:00
|
|
|
void parseDefineTexture(FScanner& sc, FScriptPosition& pos)
|
|
|
|
{
|
|
|
|
int tile, palette;
|
|
|
|
|
|
|
|
if (!sc.GetNumber(tile, true)) return;
|
|
|
|
if (!sc.GetNumber(palette, true)) return;
|
|
|
|
if (!sc.GetNumber(true)) return; //formerly x-center, unused
|
|
|
|
if (!sc.GetNumber(true)) return; //formerly y-center, unused
|
|
|
|
if (!sc.GetNumber(true)) return; //formerly x-size, unused
|
|
|
|
if (!sc.GetNumber(true)) return; //formerly y-size, unused
|
|
|
|
if (!sc.GetString()) return;
|
|
|
|
|
|
|
|
tileSetHightileReplacement(tile, palette, sc.String, -1.0, 1.0, 1.0, 1.0, 1.0);
|
|
|
|
}
|
|
|
|
|
2021-04-07 19:59:47 +00:00
|
|
|
//===========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//===========================================================================
|
|
|
|
|
2021-04-10 12:09:08 +00:00
|
|
|
static void parseTexturePaletteBlock(FScanner& sc, FScriptPosition& pos, int tile)
|
|
|
|
{
|
|
|
|
FScanner::SavedPos blockend;
|
|
|
|
|
|
|
|
int pal = -1, xsiz = 0, ysiz = 0;
|
|
|
|
FString fn;
|
|
|
|
double alphacut = -1.0, xscale = 1.0, yscale = 1.0, specpower = 1.0, specfactor = 1.0;
|
|
|
|
|
|
|
|
if (!sc.GetNumber(pal, true)) return;
|
|
|
|
|
|
|
|
if (sc.StartBraces(&blockend)) return;
|
|
|
|
while (!sc.FoundEndBrace(blockend))
|
|
|
|
{
|
|
|
|
sc.GetString();
|
|
|
|
if (sc.Compare("file")) sc.GetString(fn);
|
|
|
|
else if (sc.Compare("alphacut")) sc.GetFloat(alphacut, true);
|
|
|
|
else if (sc.Compare({ "xscale", "scale", "intensity", "detailscale" })) sc.GetFloat(xscale, true); // what's the point of all of these names?
|
|
|
|
else if (sc.Compare("yscale")) sc.GetFloat(yscale, true);
|
|
|
|
else if (sc.Compare({ "specpower", "specularpower", "parallaxscale" })) sc.GetFloat(specpower, true);
|
|
|
|
else if (sc.Compare({ "specfactor", "specularfactor", "parallaxbias" })) sc.GetFloat(specfactor, true);
|
|
|
|
else if (sc.Compare("orig_sizex")) sc.GetNumber(xsiz, true);
|
|
|
|
else if (sc.Compare("orig_sizey")) sc.GetNumber(ysiz, true);
|
|
|
|
};
|
|
|
|
|
|
|
|
if ((unsigned)tile < MAXUSERTILES)
|
|
|
|
{
|
|
|
|
if ((unsigned)pal >= MAXPALOOKUPS - RESERVEDPALS)
|
|
|
|
{
|
|
|
|
pos.Message(MSG_ERROR, "missing or invalid 'palette number' for texture definition");
|
|
|
|
}
|
|
|
|
else if (fn.IsEmpty())
|
|
|
|
{
|
|
|
|
pos.Message(MSG_ERROR, "missing 'file name' for texture definition");
|
|
|
|
}
|
|
|
|
else if (!fileSystem.FileExists(fn))
|
|
|
|
{
|
|
|
|
pos.Message(MSG_ERROR, "%s not found in replacement for tile %d", fn.GetChars(), tile);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (xsiz > 0 && ysiz > 0)
|
|
|
|
{
|
|
|
|
tileSetDummy(tile, xsiz, ysiz);
|
|
|
|
}
|
|
|
|
xscale = 1.0f / xscale;
|
|
|
|
yscale = 1.0f / yscale;
|
|
|
|
|
|
|
|
tileSetHightileReplacement(tile, pal, fn, alphacut, xscale, yscale, specpower, specfactor);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void parseTextureSpecialBlock(FScanner& sc, FScriptPosition& pos, int tile, int specialpal)
|
|
|
|
{
|
|
|
|
FScanner::SavedPos blockend;
|
|
|
|
|
|
|
|
int pal = -1;
|
|
|
|
FString fn;
|
|
|
|
double xscale = 1.0, yscale = 1.0, specpower = 1.0, specfactor = 1.0;
|
|
|
|
|
|
|
|
if (sc.StartBraces(&blockend)) return;
|
|
|
|
while (!sc.FoundEndBrace(blockend))
|
|
|
|
{
|
|
|
|
sc.GetString();
|
|
|
|
if (sc.Compare("file")) sc.GetString(fn);
|
|
|
|
else if (sc.Compare({ "xscale", "scale", "intensity", "detailscale" })) sc.GetFloat(xscale, true); // what's the point of all of these names?
|
|
|
|
else if (sc.Compare("yscale")) sc.GetFloat(yscale, true);
|
|
|
|
else if (sc.Compare({ "specpower", "specularpower", "parallaxscale" })) sc.GetFloat(specpower, true);
|
|
|
|
else if (sc.Compare({ "specfactor", "specularfactor", "parallaxbias" })) sc.GetFloat(specfactor, true);
|
|
|
|
};
|
|
|
|
|
|
|
|
if ((unsigned)tile < MAXUSERTILES)
|
|
|
|
{
|
|
|
|
if (fn.IsEmpty())
|
|
|
|
{
|
|
|
|
pos.Message(MSG_ERROR, "missing 'file name' for texture definition");
|
|
|
|
}
|
|
|
|
else if (!fileSystem.FileExists(fn))
|
|
|
|
{
|
|
|
|
pos.Message(MSG_ERROR, "%s not found in replacement for tile %d", fn.GetChars(), tile);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (pal == DETAILPAL)
|
|
|
|
{
|
|
|
|
xscale = 1.0f / xscale;
|
|
|
|
yscale = 1.0f / yscale;
|
|
|
|
}
|
|
|
|
|
|
|
|
tileSetHightileReplacement(tile, pal, fn, -1.f, xscale, yscale, specpower, specfactor);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void parseTexture(FScanner& sc, FScriptPosition& pos)
|
|
|
|
{
|
|
|
|
FScanner::SavedPos blockend;
|
|
|
|
int tile = -1;
|
|
|
|
|
|
|
|
if (!sc.GetNumber(tile, true)) return;
|
|
|
|
ValidateTilenum("texture", tile, pos); // do not abort, we still need to parse over the data.
|
|
|
|
|
|
|
|
if (sc.StartBraces(&blockend)) return;
|
|
|
|
while (!sc.FoundEndBrace(blockend))
|
|
|
|
{
|
|
|
|
sc.MustGetString();
|
|
|
|
if (sc.Compare("pal")) parseTexturePaletteBlock(sc, pos, tile);
|
|
|
|
else if (sc.Compare("detail")) parseTextureSpecialBlock(sc, pos, tile, DETAILPAL);
|
|
|
|
else if (sc.Compare("glow")) parseTextureSpecialBlock(sc, pos, tile, GLOWPAL);
|
|
|
|
else if (sc.Compare("specular")) parseTextureSpecialBlock(sc, pos, tile, SPECULARPAL);
|
|
|
|
else if (sc.Compare("normal")) parseTextureSpecialBlock(sc, pos, tile, NORMALPAL);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//===========================================================================
|
|
|
|
|
2021-04-13 16:17:28 +00:00
|
|
|
void parseTileFromTexture(FScanner& sc, FScriptPosition& pos)
|
|
|
|
{
|
|
|
|
FScanner::SavedPos blockend;
|
|
|
|
TileImport imp;
|
|
|
|
|
|
|
|
if (!sc.GetNumber(imp.tile, true)) return;
|
|
|
|
|
|
|
|
if (sc.StartBraces(&blockend)) return;
|
|
|
|
while (!sc.FoundEndBrace(blockend))
|
|
|
|
{
|
|
|
|
sc.MustGetString();
|
|
|
|
if (sc.Compare({ "file", "name" })) sc.GetString(imp.fn);
|
|
|
|
else if (sc.Compare("alphacut")) sc.GetNumber(imp.alphacut, true);
|
|
|
|
else if (sc.Compare({ "xoff", "xoffset" })) sc.GetNumber(imp.xoffset, true);
|
|
|
|
else if (sc.Compare({ "yoff", "yoffset" })) sc.GetNumber(imp.yoffset, true);
|
|
|
|
else if (sc.Compare("texhitscan")) imp.flags |= PICANM_TEXHITSCAN_BIT;
|
|
|
|
else if (sc.Compare("nofullbright")) imp.flags |= PICANM_NOFULLBRIGHT_BIT;
|
|
|
|
else if (sc.Compare("texture")) imp.istexture = 1;
|
|
|
|
else if (sc.Compare("ifcrc")) sc.GetNumber(imp.crc32, true);
|
|
|
|
else if (sc.Compare("extra")) sc.GetNumber(imp.extra, true);
|
|
|
|
else if (sc.Compare("surface")) sc.GetNumber(imp.surface, true);
|
|
|
|
else if (sc.Compare("voxel")) sc.GetNumber(imp.vox, true);
|
|
|
|
else if (sc.Compare("shade")) sc.GetNumber(imp.shade, true);
|
|
|
|
else if (sc.Compare("view")) { sc.GetNumber(imp.extra, true); imp.extra &= 7; }
|
|
|
|
else if (sc.Compare("ifmatch"))
|
|
|
|
{
|
|
|
|
FScanner::SavedPos blockend2;
|
|
|
|
if (sc.StartBraces(&blockend2)) return;
|
|
|
|
while (!sc.FoundEndBrace(blockend2))
|
|
|
|
{
|
|
|
|
sc.MustGetString();
|
|
|
|
if (sc.Compare("size"))
|
|
|
|
{
|
|
|
|
sc.GetNumber(imp.sizex, true);
|
|
|
|
sc.GetNumber(imp.sizey, true);
|
|
|
|
}
|
|
|
|
else if (sc.Compare("crc32")) sc.GetNumber(imp.crc32, true);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
processTileImport("tilefromtexture", pos, imp);
|
|
|
|
}
|
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//===========================================================================
|
|
|
|
|
|
|
|
void parseCopyTile(FScanner& sc, FScriptPosition& pos)
|
|
|
|
{
|
|
|
|
FScanner::SavedPos blockend;
|
|
|
|
int tile = -1, source = -1;
|
|
|
|
int havetile = 0, xoffset = -1024, yoffset = -1024;
|
|
|
|
int flags = 0, tsiz = 0, temppal = -1, tempsource = -1;
|
|
|
|
|
|
|
|
if (!sc.GetNumber(tile, true)) return;
|
|
|
|
|
|
|
|
if (sc.StartBraces(&blockend)) return;
|
|
|
|
while (!sc.FoundEndBrace(blockend))
|
|
|
|
{
|
|
|
|
sc.MustGetString();
|
|
|
|
if (sc.Compare("tile"))
|
|
|
|
{
|
|
|
|
if (sc.GetNumber(tempsource, true))
|
|
|
|
{
|
|
|
|
if (ValidateTilenum("copytile", tempsource, pos))
|
|
|
|
{
|
|
|
|
source = tempsource;
|
|
|
|
havetile = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (sc.Compare("pal"))
|
|
|
|
{
|
|
|
|
// This is a bit messy because it doesn't wait until everything is parsed. Sadly that quirk needs to be replicated...
|
|
|
|
if (sc.GetNumber(temppal, true))
|
|
|
|
{
|
|
|
|
// palettize self case
|
|
|
|
if (!havetile)
|
|
|
|
{
|
|
|
|
if (ValidateTilenum("copytile", tile, pos)) havetile = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((unsigned)temppal >= MAXPALOOKUPS - RESERVEDPALS)
|
|
|
|
{
|
|
|
|
pos.Message(MSG_ERROR, "copytile 'palette number' out of range (max=%d)\n", MAXPALOOKUPS - RESERVEDPALS - 1);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (sc.Compare({ "xoff", "xoffset" })) sc.GetNumber(xoffset, true);
|
|
|
|
else if (sc.Compare({ "yoff", "yoffset" })) sc.GetNumber(yoffset, true);
|
|
|
|
else if (sc.Compare("texhitscan")) flags |= PICANM_TEXHITSCAN_BIT;
|
|
|
|
else if (sc.Compare("nofullbright")) flags |= PICANM_NOFULLBRIGHT_BIT;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!ValidateTilenum("copytile", tile, pos)) return;
|
|
|
|
// if !havetile, we have never confirmed a valid source
|
|
|
|
if (!havetile && !ValidateTilenum("copytile", source, pos)) return;
|
|
|
|
|
|
|
|
tileCopy(tile, source, temppal, xoffset, yoffset, flags);
|
|
|
|
}
|
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//===========================================================================
|
|
|
|
|
2021-04-13 16:17:54 +00:00
|
|
|
void parseImportTile(FScanner& sc, FScriptPosition& pos)
|
|
|
|
{
|
|
|
|
int tile;
|
|
|
|
|
|
|
|
if (!sc.GetNumber(tile, true)) return;
|
|
|
|
if (!sc.GetString()) return;
|
|
|
|
if (!ValidateTilenum("importtile", tile, pos)) return;
|
|
|
|
|
|
|
|
int texstatus = tileImportFromTexture(sc.String, tile, 255, 0);
|
|
|
|
if (texstatus >= 0) TileFiles.tiledata[tile].picanm = {};
|
|
|
|
}
|
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//===========================================================================
|
|
|
|
|
|
|
|
void parseDummyTile(FScanner& sc, FScriptPosition& pos)
|
|
|
|
{
|
|
|
|
int tile, xsiz, ysiz;
|
|
|
|
|
|
|
|
if (!sc.GetNumber(tile, true)) return;
|
|
|
|
if (!sc.GetNumber(xsiz, true)) return;
|
|
|
|
if (!sc.GetNumber(ysiz, true)) return;
|
|
|
|
if (!ValidateTilenum("dummytile", tile, pos)) return;
|
|
|
|
tileSetDummy(tile, xsiz, ysiz);
|
|
|
|
}
|
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//===========================================================================
|
|
|
|
|
|
|
|
void parseDummyTileRange(FScanner& sc, FScriptPosition& pos)
|
|
|
|
{
|
|
|
|
int tile1, tile2, xsiz, ysiz;
|
|
|
|
|
|
|
|
if (!sc.GetNumber(tile1, true)) return;
|
|
|
|
if (!sc.GetNumber(tile2, true)) return;
|
|
|
|
if (!sc.GetNumber(xsiz, true)) return;
|
|
|
|
if (!sc.GetNumber(ysiz, true)) return;
|
|
|
|
if (!ValidateTileRange("dummytilerange", tile1, tile2, pos)) return;
|
|
|
|
if (xsiz < 0 || ysiz < 0) return;
|
|
|
|
|
|
|
|
for (int i = tile1; i <= tile2; i++) tileSetDummy(i, xsiz, ysiz);
|
|
|
|
}
|
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//===========================================================================
|
|
|
|
|
|
|
|
void parseUndefineTile(FScanner& sc, FScriptPosition& pos)
|
|
|
|
{
|
|
|
|
int tile;
|
|
|
|
|
|
|
|
if (!sc.GetNumber(tile, true)) return;
|
|
|
|
if (ValidateTilenum("undefinetile", tile, pos))
|
|
|
|
tileDelete(tile);
|
|
|
|
}
|
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//===========================================================================
|
|
|
|
|
|
|
|
void parseUndefineTileRange(FScanner& sc, FScriptPosition& pos)
|
|
|
|
{
|
|
|
|
int tile1, tile2;
|
|
|
|
|
|
|
|
if (!sc.GetNumber(tile1, true)) return;
|
|
|
|
if (!sc.GetNumber(tile2, true)) return;
|
|
|
|
if (!ValidateTileRange("undefinetilerange", tile1, tile2, pos)) return;
|
|
|
|
|
|
|
|
for (int i = tile1; i <= tile2; i++) tileDelete(i);
|
|
|
|
}
|
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//===========================================================================
|
|
|
|
|
2021-04-07 22:47:07 +00:00
|
|
|
void parseDefineSkybox(FScanner& sc, FScriptPosition& pos)
|
|
|
|
{
|
|
|
|
int tile, palette;
|
|
|
|
FString fn[6];
|
|
|
|
|
|
|
|
if (!sc.GetNumber(tile, true)) return;
|
|
|
|
if (!sc.GetNumber(palette, true)) return;
|
|
|
|
if (!sc.GetNumber(true)) return; //'future extension' (for what?)
|
|
|
|
for (int i = 0; i < 6; i++)
|
|
|
|
{
|
|
|
|
if (!sc.GetString()) return;
|
|
|
|
fn[i] = sc.String;
|
|
|
|
}
|
|
|
|
tileSetSkybox(tile, palette, fn);
|
|
|
|
}
|
|
|
|
|
2021-04-07 19:59:47 +00:00
|
|
|
//===========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//===========================================================================
|
|
|
|
|
|
|
|
void parseSkybox(FScanner& sc, FScriptPosition& pos)
|
|
|
|
{
|
2021-04-10 12:09:08 +00:00
|
|
|
FString faces[6];
|
|
|
|
FScanner::SavedPos blockend;
|
|
|
|
int tile = -1, pal = 0;
|
|
|
|
|
|
|
|
if (sc.StartBraces(&blockend)) return;
|
|
|
|
while (!sc.FoundEndBrace(blockend))
|
|
|
|
{
|
|
|
|
sc.MustGetString();
|
|
|
|
if (sc.Compare("tile")) sc.GetNumber(tile, true);
|
|
|
|
else if (sc.Compare("pal")) sc.GetNumber(pal, true);
|
|
|
|
else if (sc.Compare({ "ft", "front", "forward" })) sc.GetString(faces[0]);
|
|
|
|
else if (sc.Compare({ "rt", "right" })) sc.GetString(faces[1]);
|
|
|
|
else if (sc.Compare({ "bk", "back" })) sc.GetString(faces[2]);
|
|
|
|
else if (sc.Compare({ "lt", "lf", "left" })) sc.GetString(faces[3]);
|
|
|
|
else if (sc.Compare({ "up", "ceiling", "top", "ceil" })) sc.GetString(faces[4]);
|
|
|
|
else if (sc.Compare({ "dn", "floor", "bottom", "down" })) sc.GetString(faces[5]);
|
2021-04-07 19:59:47 +00:00
|
|
|
// skip over everything else.
|
2021-04-10 12:09:08 +00:00
|
|
|
}
|
2021-04-07 19:59:47 +00:00
|
|
|
if (tile < 0)
|
|
|
|
{
|
|
|
|
pos.Message(MSG_ERROR, "skybox: missing tile number");
|
|
|
|
return;
|
|
|
|
}
|
2021-04-10 12:09:08 +00:00
|
|
|
tileSetSkybox(tile, pal, faces);
|
2021-04-07 19:59:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//===========================================================================
|
|
|
|
|
2021-04-06 14:22:24 +00:00
|
|
|
void parseSetupTile(FScanner& sc, FScriptPosition& pos)
|
|
|
|
{
|
|
|
|
int tile;
|
|
|
|
if (!sc.GetNumber(tile, true)) return;
|
|
|
|
if (!ValidateTilenum("setuptile", tile, pos)) return;
|
|
|
|
auto& tiled = TileFiles.tiledata[tile];
|
|
|
|
if (!sc.GetNumber(tiled.hiofs.xsize, true)) return;
|
|
|
|
if (!sc.GetNumber(tiled.hiofs.ysize, true)) return;
|
|
|
|
if (!sc.GetNumber(tiled.hiofs.xoffs, true)) return;
|
|
|
|
if (!sc.GetNumber(tiled.hiofs.yoffs, true)) return;
|
|
|
|
}
|
|
|
|
|
2021-04-07 19:59:47 +00:00
|
|
|
//===========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//===========================================================================
|
|
|
|
|
2021-04-06 14:22:24 +00:00
|
|
|
void parseSetupTileRange(FScanner& sc, FScriptPosition& pos)
|
|
|
|
{
|
|
|
|
int tilestart, tileend;
|
|
|
|
if (!sc.GetNumber(tilestart, true)) return;
|
|
|
|
if (!sc.GetNumber(tileend, true)) return;
|
|
|
|
if (!ValidateTileRange("setuptilerange", tilestart, tileend, pos)) return;
|
|
|
|
|
|
|
|
TileOffs hiofs;
|
|
|
|
if (!sc.GetNumber(hiofs.xsize, true)) return;
|
|
|
|
if (!sc.GetNumber(hiofs.ysize, true)) return;
|
|
|
|
if (!sc.GetNumber(hiofs.xoffs, true)) return;
|
|
|
|
if (!sc.GetNumber(hiofs.yoffs, true)) return;
|
|
|
|
|
|
|
|
for (int i = tilestart; i <= tileend; i++) TileFiles.tiledata[i].hiofs = hiofs;
|
|
|
|
}
|
2021-04-06 13:07:12 +00:00
|
|
|
|
2021-04-07 19:59:47 +00:00
|
|
|
//===========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//===========================================================================
|
|
|
|
|
2021-04-06 13:07:12 +00:00
|
|
|
void parseAnimTileRange(FScanner& sc, FScriptPosition& pos)
|
|
|
|
{
|
|
|
|
SetAnim set;
|
|
|
|
if (!sc.GetNumber(set.tile1, true)) return;
|
|
|
|
if (!sc.GetNumber(set.tile2, true)) return;
|
|
|
|
if (!sc.GetNumber(set.speed, true)) return;
|
|
|
|
if (!sc.GetNumber(set.type, true)) return;
|
|
|
|
processSetAnim("animtilerange", pos, set);
|
|
|
|
}
|
|
|
|
|
2021-04-09 17:57:56 +00:00
|
|
|
//===========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//===========================================================================
|
|
|
|
|
|
|
|
void parseAlphahack(FScanner& sc, FScriptPosition& pos)
|
|
|
|
{
|
|
|
|
int tile;
|
|
|
|
|
|
|
|
if (!sc.GetNumber(tile, true)) return;
|
|
|
|
if (!sc.GetFloat(true)) return;
|
|
|
|
if ((unsigned)tile < MAXTILES) TileFiles.tiledata[tile].texture->alphaThreshold = (float)sc.Float;
|
|
|
|
}
|
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//===========================================================================
|
|
|
|
|
|
|
|
void parseAlphahackRange(FScanner& sc, FScriptPosition& pos)
|
|
|
|
{
|
|
|
|
int tilestart, tileend;
|
|
|
|
|
|
|
|
if (!sc.GetNumber(tilestart, true)) return;
|
|
|
|
if (!sc.GetNumber(tileend, true)) return;
|
|
|
|
if (!sc.GetFloat(true)) return;
|
|
|
|
if (!ValidateTileRange("alphahackrange", tilestart, tileend, pos)) return;
|
|
|
|
|
|
|
|
for (int i = tilestart; i <= tileend; i++)
|
|
|
|
TileFiles.tiledata[i].texture->alphaThreshold = (float)sc.Number;
|
|
|
|
}
|
|
|
|
|
2021-04-11 16:37:11 +00:00
|
|
|
//===========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//===========================================================================
|
|
|
|
static int lastvoxid = -1;
|
|
|
|
|
|
|
|
void parseDefineVoxel(FScanner& sc, FScriptPosition& pos)
|
|
|
|
{
|
|
|
|
sc.MustGetString();
|
|
|
|
while (nextvoxid < MAXVOXELS && voxreserve[nextvoxid]) nextvoxid++;
|
|
|
|
|
|
|
|
if (nextvoxid == MAXVOXELS)
|
|
|
|
{
|
|
|
|
pos.Message(MSG_ERROR, "Maximum number of voxels (%d) already defined.\n", MAXVOXELS);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (voxDefine(nextvoxid, sc.String))
|
|
|
|
{
|
|
|
|
pos.Message(MSG_ERROR, "Unable to load voxel file \"%s\"\n", sc.String);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
lastvoxid = nextvoxid++;
|
|
|
|
}
|
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//===========================================================================
|
|
|
|
|
|
|
|
void parseDefineVoxelTiles(FScanner& sc, FScriptPosition& pos)
|
|
|
|
{
|
|
|
|
int tilestart, tileend;
|
|
|
|
if (!sc.GetNumber(tilestart, true)) return;
|
|
|
|
if (!sc.GetNumber(tileend, true)) return;
|
|
|
|
if (!ValidateTileRange("definevoxeltiles", tilestart, tileend, pos)) return;
|
|
|
|
|
|
|
|
if (lastvoxid < 0)
|
|
|
|
{
|
|
|
|
pos.Message(MSG_WARNING, "Warning: Ignoring voxel tiles definition without valid voxel.\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
for (int i = tilestart; i <= tileend; i++) tiletovox[i] = lastvoxid;
|
|
|
|
}
|
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//===========================================================================
|
|
|
|
|
|
|
|
void parseVoxel(FScanner& sc, FScriptPosition& pos)
|
|
|
|
{
|
|
|
|
FScanner::SavedPos blockend;
|
|
|
|
int tile0 = MAXTILES, tile1 = -1;
|
|
|
|
FString fn;
|
|
|
|
|
|
|
|
if (!sc.GetString(fn)) return;
|
|
|
|
|
|
|
|
while (nextvoxid < MAXVOXELS && voxreserve[nextvoxid]) nextvoxid++;
|
|
|
|
|
|
|
|
if (nextvoxid == MAXVOXELS)
|
|
|
|
{
|
|
|
|
pos.Message(MSG_ERROR, "Maximum number of voxels (%d) already defined.\n", MAXVOXELS);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (voxDefine(nextvoxid, fn))
|
|
|
|
{
|
|
|
|
pos.Message(MSG_ERROR, "Unable to load voxel file \"%s\"\n", fn.GetChars());
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
int lastvoxid = nextvoxid++;
|
|
|
|
|
|
|
|
if (sc.StartBraces(&blockend)) return;
|
|
|
|
while (!sc.FoundEndBrace(blockend))
|
|
|
|
{
|
|
|
|
sc.MustGetString();
|
|
|
|
if (sc.Compare("tile"))
|
|
|
|
{
|
|
|
|
sc.GetNumber(true);
|
|
|
|
if (ValidateTilenum("voxel", sc.Number, pos)) tiletovox[sc.Number] = lastvoxid;
|
|
|
|
}
|
|
|
|
if (sc.Compare("tile0")) sc.GetNumber(tile0, true);
|
|
|
|
if (sc.Compare("tile1"))
|
|
|
|
{
|
|
|
|
sc.GetNumber(tile1, true);
|
|
|
|
if (ValidateTileRange("voxel", tile0, tile1, pos))
|
|
|
|
{
|
|
|
|
for (int i = tile0; i <= tile1; i++) tiletovox[i] = lastvoxid;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (sc.Compare("scale"))
|
|
|
|
{
|
|
|
|
sc.GetFloat(true);
|
|
|
|
voxscale[lastvoxid] = (float)sc.Float;
|
|
|
|
}
|
|
|
|
if (sc.Compare("rotate")) voxrotate.Set(lastvoxid);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-04-09 17:57:56 +00:00
|
|
|
//===========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//===========================================================================
|
|
|
|
|
|
|
|
void parseDefineTint(FScanner& sc, FScriptPosition& pos)
|
|
|
|
{
|
|
|
|
int pal, r, g, b, f;
|
|
|
|
|
|
|
|
if (!sc.GetNumber(pal, true)) return;
|
|
|
|
if (!sc.GetNumber(r)) return;
|
|
|
|
if (!sc.GetNumber(g)) return;
|
|
|
|
if (!sc.GetNumber(b)) return;
|
|
|
|
if (!sc.GetNumber(f)) return;
|
|
|
|
lookups.setPaletteTint(pal, r, g, b, 0, 0, 0, f);
|
|
|
|
}
|
2021-04-10 14:35:29 +00:00
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//===========================================================================
|
|
|
|
|
|
|
|
void parseFogpal(FScanner& sc, FScriptPosition& pos)
|
|
|
|
{
|
|
|
|
int pal, r, g, b;
|
|
|
|
|
|
|
|
if (!sc.GetNumber(pal, true)) return;
|
|
|
|
if (!sc.GetNumber(r)) return;
|
|
|
|
if (!sc.GetNumber(g)) return;
|
|
|
|
if (!sc.GetNumber(b)) return;
|
|
|
|
|
|
|
|
r = clamp(r, 0, 63);
|
|
|
|
g = clamp(g, 0, 63);
|
|
|
|
b = clamp(b, 0, 63);
|
|
|
|
|
|
|
|
lookups.makeTable(pal, nullptr, r << 2, g << 2, b << 2, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//===========================================================================
|
|
|
|
|
|
|
|
void parseNoFloorpalRange(FScanner& sc, FScriptPosition& pos)
|
|
|
|
{
|
|
|
|
int start, end;
|
|
|
|
if (!sc.GetNumber(start, true)) return;
|
|
|
|
if (!sc.GetNumber(end, true)) return;
|
|
|
|
if (start > 1) start = 1;
|
|
|
|
if (end > MAXPALOOKUPS - 1) end = MAXPALOOKUPS - 1;
|
|
|
|
for (int i = start; i <= end; i++)
|
|
|
|
lookups.tables[i].noFloorPal = true;
|
|
|
|
}
|
2021-04-11 13:23:42 +00:00
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//===========================================================================
|
|
|
|
|
|
|
|
void parseTint(FScanner& sc, FScriptPosition& pos)
|
|
|
|
{
|
|
|
|
int red = 255, green = 255, blue = 255, shadered = 0, shadegreen = 0, shadeblue = 0, pal = -1, flags = 0;
|
|
|
|
FScanner::SavedPos blockend;
|
|
|
|
|
|
|
|
if (sc.StartBraces(&blockend)) return;
|
|
|
|
while (!sc.FoundEndBrace(blockend))
|
|
|
|
{
|
|
|
|
sc.MustGetString();
|
|
|
|
if (sc.Compare("pal")) sc.GetNumber(pal, true);
|
|
|
|
else if (sc.Compare({"red", "r"})) sc.GetNumber(red);
|
|
|
|
else if (sc.Compare({ "green", "g" })) sc.GetNumber(green);
|
|
|
|
else if (sc.Compare({ "blue", "b" })) sc.GetNumber(blue);
|
|
|
|
else if (sc.Compare({ "shadered", "sr" })) sc.GetNumber(shadered);
|
|
|
|
else if (sc.Compare({ "shadegreen", "sg" })) sc.GetNumber(shadegreen);
|
|
|
|
else if (sc.Compare({ "shadeblue", "sb" })) sc.GetNumber(shadeblue);
|
|
|
|
else if (sc.Compare("flags")) sc.GetNumber(flags, true);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (pal < 0)
|
|
|
|
pos.Message(MSG_ERROR, "tint: missing palette number");
|
|
|
|
else
|
|
|
|
lookups.setPaletteTint(pal, clamp(red, 0, 255), clamp(green, 0, 255), clamp(blue, 0, 255),
|
|
|
|
clamp(shadered, 0, 255), clamp(shadegreen, 0, 255), clamp(shadeblue, 0, 255), flags);
|
|
|
|
}
|
2021-04-12 22:28:51 +00:00
|
|
|
|
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//===========================================================================
|
|
|
|
|
|
|
|
void parseMusic(FScanner& sc, FScriptPosition& pos)
|
|
|
|
{
|
|
|
|
FString id, file;
|
|
|
|
FScanner::SavedPos blockend;
|
|
|
|
|
|
|
|
if (sc.StartBraces(&blockend)) return;
|
|
|
|
while (!sc.FoundEndBrace(blockend))
|
|
|
|
{
|
|
|
|
sc.MustGetString();
|
|
|
|
if (sc.Compare("id")) sc.GetString(id);
|
|
|
|
else if (sc.Compare("file")) sc.GetString(file);
|
|
|
|
}
|
|
|
|
SetMusicForMap(id, file, true);
|
|
|
|
}
|
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//===========================================================================
|
|
|
|
|
|
|
|
void parseMapinfo(FScanner& sc, FScriptPosition& pos)
|
|
|
|
{
|
|
|
|
usermaphack_t mhk;
|
|
|
|
FScanner::SavedPos blockend;
|
|
|
|
|
|
|
|
if (sc.StartBraces(&blockend)) return;
|
|
|
|
while (!sc.FoundEndBrace(blockend))
|
|
|
|
{
|
|
|
|
sc.MustGetString();
|
|
|
|
if (sc.Compare("mapfile")) sc.GetString();
|
|
|
|
else if (sc.Compare("maptitle")) sc.GetString(mhk.title);
|
|
|
|
else if (sc.Compare("mhkfile")) sc.GetString(mhk.mhkfile);
|
|
|
|
else if (sc.Compare("mapmd4"))
|
|
|
|
{
|
|
|
|
sc.GetString();
|
|
|
|
for (int i = 0; i < 16; i++)
|
|
|
|
{
|
|
|
|
char smallbuf[3] = { sc.String[2 * i], sc.String[2 * i + 1], 0 };
|
|
|
|
mhk.md4[i] = strtol(smallbuf, nullptr, 16);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
AddUserMapHack(mhk);
|
|
|
|
}
|
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//===========================================================================
|
|
|
|
|
|
|
|
void parseEcho(FScanner& sc, FScriptPosition& pos)
|
|
|
|
{
|
|
|
|
sc.MustGetString();
|
|
|
|
Printf("%s\n", sc.String);
|
|
|
|
}
|
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//===========================================================================
|
|
|
|
|
|
|
|
void parseMultiPsky(FScanner& sc, FScriptPosition& pos)
|
|
|
|
{
|
|
|
|
usermaphack_t mhk;
|
|
|
|
FScanner::SavedPos blockend;
|
|
|
|
psky_t sky{};
|
|
|
|
|
|
|
|
sky.yscale = 65536;
|
|
|
|
if (!sc.GetNumber(sky.tilenum, true)) return;
|
|
|
|
|
|
|
|
if (sc.StartBraces(&blockend)) return;
|
|
|
|
while (!sc.FoundEndBrace(blockend))
|
|
|
|
{
|
|
|
|
sc.MustGetString();
|
|
|
|
if (sc.Compare("horizfrac")) sc.GetNumber(sky.horizfrac, true);
|
|
|
|
else if (sc.Compare("yoffset")) sc.GetNumber(sky.yoffs, true);
|
|
|
|
else if (sc.Compare("lognumtiles")) sc.GetNumber(sky.lognumtiles, true);
|
|
|
|
else if (sc.Compare("yscale")) sc.GetNumber(sky.yscale, true);
|
|
|
|
else if (sc.Compare({ "tile", "panel" }))
|
|
|
|
{
|
|
|
|
int panel, offset;
|
|
|
|
sc.GetNumber(panel, true);
|
|
|
|
sc.GetNumber(offset, true);
|
|
|
|
if ((unsigned)panel < MAXPSKYTILES && (unsigned)offset <= PSKYOFF_MAX) sky.tileofs[panel] = offset;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (sky.tilenum != DEFAULTPSKY && (unsigned)sky.tilenum >= MAXUSERTILES) return;
|
|
|
|
if ((1 << sky.lognumtiles) > MAXPSKYTILES) return;
|
|
|
|
auto psky = tileSetupSky(sky.tilenum);
|
|
|
|
*psky = sky;
|
|
|
|
}
|
2021-04-13 16:15:12 +00:00
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//===========================================================================
|
|
|
|
|
|
|
|
void parseRffDefineId(FScanner& sc, FScriptPosition& pos)
|
|
|
|
{
|
|
|
|
FString resName;
|
|
|
|
FString resType;
|
|
|
|
int resID;
|
|
|
|
|
|
|
|
if (!sc.GetString(resName)) return;
|
|
|
|
if (!sc.GetString(resType)) return;
|
|
|
|
if (!sc.GetNumber(resID)) return;
|
|
|
|
if (!sc.GetString()) return;
|
|
|
|
resName.AppendFormat(".%s", resType.GetChars());
|
|
|
|
fileSystem.CreatePathlessCopy(resName, resID, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
//
|
|
|
|
// empty stub
|
|
|
|
//
|
|
|
|
//===========================================================================
|
|
|
|
|
2021-04-13 16:16:34 +00:00
|
|
|
void parseEmptyBlock(FScanner& sc, FScriptPosition& pos)
|
2021-04-13 16:15:12 +00:00
|
|
|
{
|
|
|
|
FScanner::SavedPos blockend;
|
|
|
|
|
|
|
|
if (sc.StartBraces(&blockend)) return;
|
2021-04-13 16:17:04 +00:00
|
|
|
while (!sc.FoundEndBrace(blockend))
|
|
|
|
{
|
|
|
|
sc.MustGetString();
|
|
|
|
}
|
2021-04-13 16:15:12 +00:00
|
|
|
}
|
|
|
|
|
2021-04-13 16:15:56 +00:00
|
|
|
//===========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//===========================================================================
|
|
|
|
|
|
|
|
void parseTexHitscanRange(FScanner& sc, FScriptPosition& pos)
|
|
|
|
{
|
|
|
|
int start, end;
|
|
|
|
|
|
|
|
if (!sc.GetNumber(start, true)) return;
|
|
|
|
if (!sc.GetNumber(end, true)) return;
|
|
|
|
|
|
|
|
if (start < 0) start = 0;
|
|
|
|
if (end >= MAXUSERTILES) end = MAXUSERTILES - 1;
|
|
|
|
for (int i = start; i <= end; i++)
|
|
|
|
TileFiles.tiledata[i].picanm.sf |= PICANM_TEXHITSCAN_BIT;
|
|
|
|
}
|
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//===========================================================================
|
|
|
|
|
|
|
|
void parseNoFullbrightRange(FScanner& sc, FScriptPosition& pos)
|
|
|
|
{
|
|
|
|
int start, end;
|
|
|
|
|
|
|
|
if (!sc.GetNumber(start, true)) return;
|
|
|
|
if (!sc.GetNumber(end, true)) return;
|
|
|
|
|
|
|
|
if (start < 0) start = 0;
|
|
|
|
if (end >= MAXUSERTILES) end = MAXUSERTILES - 1;
|
|
|
|
for (int i = start; i <= end; i++)
|
|
|
|
{
|
|
|
|
auto tex = tileGetTexture(i);
|
|
|
|
if (tex->isValid()) tex->SetDisableBrightmap();
|
|
|
|
}
|
|
|
|
}
|
2021-04-13 16:17:04 +00:00
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//
|
|
|
|
//===========================================================================
|
|
|
|
|
|
|
|
void parseArtFile(FScanner& sc, FScriptPosition& pos)
|
|
|
|
{
|
|
|
|
FScanner::SavedPos blockend;
|
|
|
|
FString file;
|
|
|
|
int tile = -1;
|
|
|
|
|
|
|
|
if (sc.StartBraces(&blockend)) return;
|
|
|
|
while (!sc.FoundEndBrace(blockend))
|
|
|
|
{
|
|
|
|
sc.MustGetString();
|
|
|
|
if (sc.Compare("file")) sc.GetString(file);
|
|
|
|
else if (sc.Compare("tile")) sc.GetNumber(tile, true);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (file.IsEmpty())
|
|
|
|
{
|
|
|
|
pos.Message(MSG_ERROR, "missing 'file name' for artfile definition");
|
|
|
|
}
|
|
|
|
else if (tile >= 0 && ValidateTilenum("artfile", tile, pos))
|
|
|
|
TileFiles.LoadArtFile(file, nullptr, tile);
|
|
|
|
}
|