1
0
mirror of https://gitlab.com/OpenMW/openmw.git synced 2025-01-25 15:35:23 +00:00
OpenMW/components/vfs/manager.cpp

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

135 lines
3.6 KiB
C++
Raw Normal View History

#include "manager.hpp"
#include <algorithm>
#include <stdexcept>
#include <components/files/conversion.hpp>
#include <components/misc/strings/lower.hpp>
#include "archive.hpp"
namespace
{
char strict_normalize_char(char ch)
{
return ch == '\\' ? '/' : ch;
}
char nonstrict_normalize_char(char ch)
{
return ch == '\\' ? '/' : Misc::StringUtils::toLower(ch);
}
void normalize_path(std::string& path, bool strict)
{
char (*normalize_char)(char) = strict ? &strict_normalize_char : &nonstrict_normalize_char;
std::transform(path.begin(), path.end(), path.begin(), normalize_char);
}
}
namespace VFS
{
Manager::Manager(bool strict)
: mStrict(strict)
{
}
Manager::~Manager() {}
void Manager::reset()
{
mIndex.clear();
mArchives.clear();
}
void Manager::addArchive(std::unique_ptr<Archive>&& archive)
{
mArchives.push_back(std::move(archive));
}
void Manager::buildIndex()
{
mIndex.clear();
for (const auto& archive : mArchives)
archive->listResources(mIndex, mStrict ? &strict_normalize_char : &nonstrict_normalize_char);
}
2022-07-17 12:19:19 +02:00
Files::IStreamPtr Manager::get(std::string_view name) const
{
2022-07-17 12:19:19 +02:00
std::string normalized(name);
normalize_path(normalized, mStrict);
2015-03-26 18:02:51 +01:00
return getNormalized(normalized);
}
Files::IStreamPtr Manager::getNormalized(const std::string& normalizedName) const
{
std::map<std::string, File*>::const_iterator found = mIndex.find(normalizedName);
if (found == mIndex.end())
2015-03-26 18:02:51 +01:00
throw std::runtime_error("Resource '" + normalizedName + "' not found");
return found->second->open();
}
2022-07-17 12:19:19 +02:00
bool Manager::exists(std::string_view name) const
{
2022-07-17 12:19:19 +02:00
std::string normalized(name);
normalize_path(normalized, mStrict);
return mIndex.find(normalized) != mIndex.end();
}
2022-07-17 12:19:19 +02:00
std::string Manager::normalizeFilename(std::string_view name) const
2015-03-26 18:02:51 +01:00
{
2022-07-17 12:19:19 +02:00
std::string result(name);
2021-09-06 22:01:41 +02:00
normalize_path(result, mStrict);
return result;
2015-03-26 18:02:51 +01:00
}
2022-07-17 12:19:19 +02:00
std::string Manager::getArchive(std::string_view name) const
2020-12-29 21:45:59 +01:00
{
2022-07-17 12:19:19 +02:00
std::string normalized(name);
2020-12-29 21:45:59 +01:00
normalize_path(normalized, mStrict);
2020-12-30 10:35:51 +01:00
for (auto it = mArchives.rbegin(); it != mArchives.rend(); ++it)
2020-12-29 21:45:59 +01:00
{
2020-12-30 10:35:51 +01:00
if ((*it)->contains(normalized, mStrict ? &strict_normalize_char : &nonstrict_normalize_char))
return (*it)->getDescription();
2020-12-29 21:45:59 +01:00
}
return {};
}
std::filesystem::path Manager::getAbsoluteFileName(const std::filesystem::path& name) const
2022-05-13 18:58:00 -07:00
{
std::string normalized = Files::pathToUnicodeString(name);
normalize_path(normalized, mStrict);
2022-05-13 18:58:00 -07:00
const auto found = mIndex.find(normalized);
2022-05-13 18:58:00 -07:00
if (found == mIndex.end())
throw std::runtime_error("Resource '" + normalized + "' not found");
return found->second->getPath();
2022-05-13 18:58:00 -07:00
}
namespace
{
bool startsWith(std::string_view text, std::string_view start)
{
return text.rfind(start, 0) == 0;
}
}
2022-07-17 12:19:19 +02:00
Manager::RecursiveDirectoryRange Manager::getRecursiveDirectoryIterator(std::string_view path) const
{
if (path.empty())
return { mIndex.begin(), mIndex.end() };
auto normalized = normalizeFilename(path);
const auto it = mIndex.lower_bound(normalized);
if (it == mIndex.end() || !startsWith(it->first, normalized))
return { it, it };
++normalized.back();
return { it, mIndex.lower_bound(normalized) };
}
}