mirror of
https://github.com/MadeOfJelly/MushMachine.git
synced 2025-01-09 22:53:14 +01:00
149 lines
3.5 KiB
C++
149 lines
3.5 KiB
C++
#pragma once
|
|
|
|
#include <memory>
|
|
#include <utility>
|
|
#include <algorithm>
|
|
#include <cassert>
|
|
#include <functional>
|
|
#include <optional>
|
|
#include <unordered_map>
|
|
|
|
#include <entt/config/config.h>
|
|
#include <entt/core/hashed_string.hpp>
|
|
|
|
#include <mm/logger.hpp>
|
|
#define LOGRM(x) LOG("ResourceManager", x)
|
|
|
|
namespace MM {
|
|
|
|
// modeled heavily after entt::resource_cache
|
|
// https://github.com/skypjack/entt/blob/master/src/entt/resource/cache.hpp
|
|
template<typename Resource>
|
|
class ResourceManager {
|
|
public:
|
|
using res_id_type = typename entt::hashed_string::hash_type;
|
|
using handle_type = std::shared_ptr<Resource>;
|
|
using container_type = std::unordered_map<res_id_type, handle_type>;
|
|
using size_type = typename container_type::size_type;
|
|
|
|
private:
|
|
ResourceManager(void) = default;
|
|
|
|
container_type _storage;
|
|
|
|
public:
|
|
ResourceManager(ResourceManager const&) = delete;
|
|
void operator=(ResourceManager const&) = delete;
|
|
|
|
static ResourceManager& ref(void) {
|
|
static ResourceManager<Resource> instance{};
|
|
return instance;
|
|
}
|
|
|
|
// Loader s must implement a load method
|
|
template<typename Loader, typename... Args>
|
|
bool load(const res_id_type id, Args &&... args) {
|
|
//static_assert(std::is_base_of<entt::resource_loader<Loader, Resource>, Loader>::value);
|
|
|
|
if (!contains(id)) {
|
|
handle_type r = Loader{}.load(std::forward<Args>(args)...);
|
|
if (static_cast<bool>(r)) {
|
|
_storage[id] = std::move(r);
|
|
} else {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
// string overload
|
|
template<typename Loader, typename... Args>
|
|
bool load(const char* id, Args &&... args) {
|
|
return load<Loader>(entt::hashed_string{id}.value(), std::forward<Args>(args)...);
|
|
}
|
|
|
|
template<typename Loader, typename... Args>
|
|
bool reload(const res_id_type id, Args &&... args) {
|
|
discard(id);
|
|
return load<Loader>(id, std::forward<Args>(args)...);
|
|
}
|
|
|
|
// string overload
|
|
template<typename Loader, typename... Args>
|
|
bool reload(const char* id, Args &&... args) {
|
|
return reload<Loader>(entt::hashed_string{id}.value(), std::forward<Args>(args)...);
|
|
}
|
|
|
|
void discard(const res_id_type id) noexcept {
|
|
if (auto it = _storage.find(id); it != _storage.end()) {
|
|
_storage.erase(it);
|
|
}
|
|
}
|
|
|
|
void discard(const char* id) noexcept {
|
|
discard(entt::hashed_string{id}.value());
|
|
}
|
|
|
|
template<typename FN>
|
|
void each(FN&& fn) {
|
|
for (auto& it : _storage) {
|
|
fn(it.first, it.second);
|
|
}
|
|
}
|
|
|
|
size_type size(void) const {
|
|
return _storage.size();
|
|
}
|
|
|
|
bool empty(void) const {
|
|
return _storage.empty();
|
|
}
|
|
|
|
void clear(void) noexcept {
|
|
_storage.clear();
|
|
}
|
|
|
|
bool contains(const res_id_type id) const noexcept {
|
|
return (_storage.find(id) != _storage.cend());
|
|
}
|
|
|
|
handle_type get(const res_id_type id) const {
|
|
using namespace entt::literals;
|
|
|
|
auto it = _storage.find(id);
|
|
if (it == _storage.end()) {
|
|
if (id != "default"_hs) {
|
|
LOGRM("not in cache, loading default...");
|
|
return get("default"_hs);
|
|
} else {
|
|
return nullptr;
|
|
}
|
|
} else {
|
|
return it->second;
|
|
}
|
|
}
|
|
|
|
// TODO: fix
|
|
//handle_type get(const char* id) const {
|
|
//return get(entt::hashed_string{id}.value());
|
|
//}
|
|
|
|
// reverse lookup
|
|
std::optional<res_id_type> id_from_handle(const std::shared_ptr<Resource>& handle) {
|
|
if (static_cast<bool>(handle)) {
|
|
auto it = std::find_if(_storage.cbegin(), _storage.cend(), [&](auto& v) { return handle == v.second; });
|
|
if (it != _storage.cend()) {
|
|
return it->first;
|
|
}
|
|
}
|
|
|
|
return std::nullopt;
|
|
}
|
|
};
|
|
|
|
} // namespace MM
|
|
|
|
#undef LOGRM
|
|
|