2013-11-06 01:41:49 +00:00
|
|
|
#include "FileData.h"
|
|
|
|
|
|
|
|
namespace fs = boost::filesystem;
|
|
|
|
|
|
|
|
std::string getCleanFileName(const fs::path& path)
|
|
|
|
{
|
2014-04-16 17:32:40 +00:00
|
|
|
// remove anything in parenthesis or brackets
|
|
|
|
// should be roughly equivalent to the regex replace "\((.*)\)|\[(.*)\]" with ""
|
|
|
|
// I would love to just use regex, but it's not worth pulling in another boost lib for one function that is used once
|
|
|
|
|
|
|
|
std::string ret = path.stem().generic_string();
|
|
|
|
size_t start, end;
|
|
|
|
|
|
|
|
static const int NUM_TO_REPLACE = 2;
|
|
|
|
static const char toReplace[NUM_TO_REPLACE*2] = { '(', ')', '[', ']' };
|
|
|
|
|
|
|
|
bool done = false;
|
|
|
|
while(!done)
|
|
|
|
{
|
|
|
|
done = true;
|
|
|
|
for(int i = 0; i < NUM_TO_REPLACE; i++)
|
|
|
|
{
|
|
|
|
start = ret.find(toReplace[i*2]);
|
|
|
|
end = ret.find(toReplace[i*2+1], start != std::string::npos ? start + 1 : 0);
|
|
|
|
if(start != std::string::npos && end != std::string::npos)
|
|
|
|
{
|
|
|
|
ret.replace(start, end, "");
|
|
|
|
done = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
2013-11-06 01:41:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-11-12 23:28:15 +00:00
|
|
|
FileData::FileData(FileType type, const fs::path& path, SystemData* system)
|
|
|
|
: mType(type), mPath(path), mSystem(system), mParent(NULL), metadata(type == GAME ? GAME_METADATA : FOLDER_METADATA) // metadata is REALLY set in the constructor!
|
2013-11-06 01:41:49 +00:00
|
|
|
{
|
|
|
|
// metadata needs at least a name field (since that's what getName() will return)
|
|
|
|
if(metadata.get("name").empty())
|
|
|
|
metadata.set("name", getCleanFileName(mPath));
|
|
|
|
}
|
|
|
|
|
|
|
|
FileData::~FileData()
|
|
|
|
{
|
|
|
|
if(mParent)
|
|
|
|
mParent->removeChild(this);
|
2013-11-12 23:28:15 +00:00
|
|
|
|
|
|
|
while(mChildren.size())
|
|
|
|
delete mChildren.back();
|
2013-11-06 01:41:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
const std::string& FileData::getThumbnailPath() const
|
|
|
|
{
|
|
|
|
if(!metadata.get("thumbnail").empty())
|
|
|
|
return metadata.get("thumbnail");
|
|
|
|
else
|
|
|
|
return metadata.get("image");
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
std::vector<FileData*> FileData::getFilesRecursive(unsigned int typeMask) const
|
|
|
|
{
|
|
|
|
std::vector<FileData*> out;
|
|
|
|
|
|
|
|
for(auto it = mChildren.begin(); it != mChildren.end(); it++)
|
|
|
|
{
|
|
|
|
if((*it)->getType() & typeMask)
|
|
|
|
out.push_back(*it);
|
|
|
|
|
|
|
|
if((*it)->getChildren().size() > 0)
|
|
|
|
{
|
|
|
|
std::vector<FileData*> subchildren = (*it)->getFilesRecursive(typeMask);
|
|
|
|
out.insert(out.end(), subchildren.cbegin(), subchildren.cend());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return out;
|
|
|
|
}
|
|
|
|
|
|
|
|
void FileData::addChild(FileData* file)
|
|
|
|
{
|
|
|
|
assert(mType == FOLDER);
|
|
|
|
assert(file->getParent() == NULL);
|
|
|
|
|
|
|
|
mChildren.push_back(file);
|
|
|
|
file->mParent = this;
|
|
|
|
}
|
|
|
|
|
|
|
|
void FileData::removeChild(FileData* file)
|
|
|
|
{
|
|
|
|
assert(mType == FOLDER);
|
|
|
|
assert(file->getParent() == this);
|
|
|
|
|
|
|
|
for(auto it = mChildren.begin(); it != mChildren.end(); it++)
|
|
|
|
{
|
|
|
|
if(*it == file)
|
|
|
|
{
|
|
|
|
mChildren.erase(it);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// File somehow wasn't in our children.
|
|
|
|
assert(false);
|
|
|
|
}
|
|
|
|
|
|
|
|
void FileData::sort(ComparisonFunction& comparator, bool ascending)
|
|
|
|
{
|
|
|
|
std::sort(mChildren.begin(), mChildren.end(), comparator);
|
|
|
|
|
|
|
|
for(auto it = mChildren.begin(); it != mChildren.end(); it++)
|
|
|
|
{
|
|
|
|
if((*it)->getChildren().size() > 0)
|
|
|
|
(*it)->sort(comparator, ascending);
|
|
|
|
}
|
|
|
|
|
|
|
|
if(!ascending)
|
|
|
|
std::reverse(mChildren.begin(), mChildren.end());
|
|
|
|
}
|
|
|
|
|
|
|
|
void FileData::sort(const SortType& type)
|
|
|
|
{
|
|
|
|
sort(*type.comparisonFunction, type.ascending);
|
|
|
|
}
|