mirror of
https://github.com/AquariaOSE/Aquaria.git
synced 2025-01-26 02:07:26 +00:00
011b9f2a82
- correctly sort ElementTemplate by idx, not by pointer. oops. - selecting tiles works again (new: including wraparound) - don't display layer tile borders when switching to other edit modes - make sure that tile selection wraparound doesn't pick up font letter tiles
280 lines
6.2 KiB
C++
280 lines
6.2 KiB
C++
#include "Tileset.h"
|
|
#include "SimpleIStringStream.h"
|
|
#include "Base.h"
|
|
#include "ttvfs_stdio.h"
|
|
#include "TextureMgr.h"
|
|
#include "Core.h"
|
|
|
|
Tileset::Tileset()
|
|
{
|
|
}
|
|
|
|
Tileset::~Tileset()
|
|
{
|
|
clear();
|
|
}
|
|
|
|
static bool _etsort(const ElementTemplate *a, const ElementTemplate *b)
|
|
{
|
|
return a->idx < b->idx;
|
|
}
|
|
|
|
bool Tileset::loadFile(const char *fn, const unsigned char *usedIdx, size_t usedIdxLen)
|
|
{
|
|
clear();
|
|
|
|
InStream in(fn);
|
|
if(!in)
|
|
return false;
|
|
|
|
bool warn = false;
|
|
std::string line, gfx;
|
|
while (std::getline(in, line))
|
|
{
|
|
gfx.clear();
|
|
int idx=-1, w=0, h=0;
|
|
SimpleIStringStream is(line.c_str(), SimpleIStringStream::REUSE);
|
|
is >> idx >> gfx >> w >> h;
|
|
if(idx >= 0 && !gfx.empty())
|
|
{
|
|
if(idx < 1024)
|
|
{
|
|
ElementTemplate *et = new ElementTemplate;
|
|
et->idx = idx;
|
|
et->gfx = gfx;
|
|
et->w = w;
|
|
et->h = h;
|
|
elementTemplates.push_back(et);
|
|
}
|
|
else
|
|
warn = true;
|
|
}
|
|
}
|
|
in.close();
|
|
|
|
if(warn)
|
|
errorLog("Tileset indices of 1024 and above are reserved; ignored during load");
|
|
|
|
std::sort(elementTemplates.begin(), elementTemplates.end(), _etsort);
|
|
|
|
// begin preloading textures
|
|
|
|
std::vector<std::string> usedTex;
|
|
usedTex.reserve(elementTemplates.size()); // optimistically assume all textures in the tileset are used
|
|
|
|
for (size_t i = 0; i < elementTemplates.size(); i++)
|
|
{
|
|
size_t idx = elementTemplates[i]->idx;
|
|
if (!usedIdx || (idx < usedIdxLen && usedIdx[idx]))
|
|
usedTex.push_back(elementTemplates[i]->gfx);
|
|
}
|
|
std::sort(usedTex.begin(), usedTex.end());
|
|
// drop duplicates
|
|
usedTex.resize(std::distance(usedTex.begin(), std::unique(usedTex.begin(), usedTex.end())));
|
|
|
|
{
|
|
std::ostringstream os;
|
|
os << "Loading " << usedTex.size()
|
|
<< " used textures out of the " << elementTemplates.size() << " tileset entries";
|
|
debugLog(os.str());
|
|
}
|
|
|
|
// preload all used textures
|
|
size_t loaded = 0;
|
|
if(usedTex.size())
|
|
loaded = core->texmgr.loadBatch(NULL, &usedTex[0], usedTex.size());
|
|
|
|
{
|
|
std::ostringstream os;
|
|
os << "Loaded " << loaded << " textures successfully";
|
|
debugLog(os.str());
|
|
}
|
|
|
|
// finalize
|
|
size_t nfailed = 0;
|
|
std::ostringstream failed;
|
|
for (size_t i = 0; i < elementTemplates.size(); i++)
|
|
{
|
|
ElementTemplate *et = elementTemplates[i];
|
|
// only check those that are actualy loaded; otherwise this would load in textures
|
|
// that we didn't bother to batch-load above
|
|
if(!usedIdx || (et->idx < usedIdxLen && usedIdx[et->idx]))
|
|
{
|
|
et->finalize(); // assigns width/height and caches texture pointer
|
|
if(!et->tex)
|
|
{
|
|
++nfailed;
|
|
failed << et->gfx << " ";
|
|
}
|
|
}
|
|
}
|
|
|
|
if(nfailed)
|
|
{
|
|
std::ostringstream os;
|
|
os << "The following " << nfailed << " textures failed to load and would be used by tiles:";
|
|
debugLog(os.str());
|
|
debugLog(failed.str());
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
void Tileset::clear()
|
|
{
|
|
for(size_t i = 0; i < dummies.size(); ++i)
|
|
delete dummies[i];
|
|
dummies.clear();
|
|
|
|
for(size_t i = 0; i < elementTemplates.size(); ++i)
|
|
delete elementTemplates[i];
|
|
elementTemplates.clear();
|
|
}
|
|
|
|
const ElementTemplate * Tileset::getIfExists(size_t idx)
|
|
{
|
|
for (size_t i = 0; i < elementTemplates.size(); i++)
|
|
{
|
|
ElementTemplate *et = elementTemplates[i];
|
|
if (et->idx == idx)
|
|
{
|
|
et->finalize(); // HACK: make sure the texture is loaded before this gets used
|
|
return et;
|
|
}
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
const ElementTemplate *Tileset::getByIdx(size_t idx)
|
|
{
|
|
if(const ElementTemplate *existing = getIfExists(idx))
|
|
return existing;
|
|
|
|
// a tile that gets an ET attached must remember its tileset id even if the entry is not present
|
|
// in the tileset. since the tile does not store the idx as an integer, we need to return a dummy element.
|
|
for (size_t i = 0; i < dummies.size(); i++)
|
|
{
|
|
ElementTemplate *et = dummies[i];
|
|
if (et->idx == idx)
|
|
return et;
|
|
}
|
|
|
|
{
|
|
std::ostringstream os;
|
|
os << "Tileset idx " << idx << " not found, creating dummy";
|
|
debugLog(os.str());
|
|
}
|
|
|
|
ElementTemplate *dummy = new ElementTemplate;
|
|
dummy->idx = idx;
|
|
dummy->finalize();
|
|
dummies.push_back(dummy);
|
|
|
|
return dummy;
|
|
}
|
|
|
|
const ElementTemplate* Tileset::getAdjacent(size_t idx, int direction, bool wraparound, size_t maxidx)
|
|
{
|
|
ElementTemplate *et = _getAdjacent(idx, direction, wraparound, maxidx);
|
|
if(et)
|
|
et->finalize(); // load just in case
|
|
return et;
|
|
}
|
|
|
|
|
|
ElementTemplate::~ElementTemplate()
|
|
{
|
|
delete grid;
|
|
}
|
|
|
|
void ElementTemplate::finalize()
|
|
{
|
|
if(!gfx.empty())
|
|
tex = core->getTexture(gfx); // may end up NULL
|
|
if(tex)
|
|
{
|
|
if(!w)
|
|
w = tex->width;
|
|
if(!h)
|
|
h = tex->height;
|
|
}
|
|
else
|
|
{
|
|
if(!w)
|
|
w = 64;
|
|
if(!h)
|
|
h = 64;
|
|
}
|
|
|
|
if(tc.isStandard())
|
|
{
|
|
delete grid;
|
|
grid = NULL;
|
|
}
|
|
else
|
|
{
|
|
if(!grid)
|
|
grid = new RenderGrid;
|
|
grid->init(2, 2, tc);
|
|
}
|
|
}
|
|
|
|
ElementTemplate * Tileset::_getAdjacent(size_t idx, int direction, bool wraparound, size_t maxidx)
|
|
{
|
|
assert(direction == 1 || direction == -1);
|
|
|
|
const size_t maxn = elementTemplates.size();
|
|
if(!maxn)
|
|
return NULL;
|
|
|
|
if(idx >= maxidx)
|
|
return NULL;
|
|
|
|
size_t closest = 0;
|
|
int mindiff = 0;
|
|
for (size_t i = 0; i < maxn; i++)
|
|
{
|
|
if (elementTemplates[i]->idx == idx)
|
|
{
|
|
if(wraparound)
|
|
{
|
|
if(!i && direction < 0)
|
|
for(size_t k = maxn; k --> 0; )
|
|
if(elementTemplates[k]->idx < maxidx)
|
|
return elementTemplates[k];
|
|
if(i + direction >= maxn || elementTemplates[i+direction]->idx >= maxidx)
|
|
return elementTemplates[0];
|
|
}
|
|
|
|
|
|
i += direction; // may underflow
|
|
return i < maxn && elementTemplates[i]->idx < maxidx ? elementTemplates[i] : NULL;
|
|
}
|
|
int diff = labs((int)elementTemplates[i]->idx - (int)idx);
|
|
if(diff < mindiff || !mindiff)
|
|
{
|
|
mindiff = diff;
|
|
closest = i;
|
|
}
|
|
}
|
|
|
|
// not found? pick whatever was closest to the non-existing idx, and go back/forward from there
|
|
|
|
// avoid going "twice" in the given direction
|
|
if(closest < idx && direction < 0)
|
|
direction = 0; // this is already a step back, don't step again
|
|
else if(closest > idx && direction > 0)
|
|
direction = 0; // this is already a step forward, don't step again
|
|
else if(wraparound)
|
|
{
|
|
if(!closest && direction < 0)
|
|
for(size_t k = maxn; k --> 0; )
|
|
if(elementTemplates[k]->idx < maxidx)
|
|
return elementTemplates[k];
|
|
if(closest + direction >= maxn || elementTemplates[closest+direction]->idx >= maxidx)
|
|
return elementTemplates[0];
|
|
}
|
|
|
|
size_t i = closest + direction;
|
|
return i < maxn && elementTemplates[i]->idx < maxidx ? elementTemplates[i] : NULL;
|
|
}
|