summaryrefslogtreecommitdiffhomepage
path: root/loader/wall-atlas.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'loader/wall-atlas.cpp')
-rw-r--r--loader/wall-atlas.cpp150
1 files changed, 18 insertions, 132 deletions
diff --git a/loader/wall-atlas.cpp b/loader/wall-atlas.cpp
index 3c4e3896..b07c73f2 100644
--- a/loader/wall-atlas.cpp
+++ b/loader/wall-atlas.cpp
@@ -1,157 +1,43 @@
#include "loader/impl.hpp"
-#include "src/tile-constants.hpp"
-#include "loader/wall-cell.hpp"
-#include "compat/assert.hpp"
-#include "compat/exception.hpp"
#include "compat/vector-wrapper.hpp"
#include "src/wall-atlas.hpp"
-//#include "serialize/json-helper.hpp"
-//#include "serialize/corrade-string.hpp"
+#include "loader/wall-cell.hpp"
+#include "loader/wall-traits.hpp"
+#include "loader/atlas-loader.inl"
#include <Corrade/Containers/Array.h>
-#include <Corrade/Containers/ArrayViewStl.h>
#include <Corrade/Containers/StringIterable.h>
-#include <Corrade/Utility/Path.h>
#include <Magnum/Trade/ImageData.h>
#include <Magnum/ImageView.h>
-namespace floormat {
+namespace floormat::loader_detail {
+
+template class atlas_loader<class wall_atlas>;
-std::shared_ptr<wall_atlas> loader_::get_wall_atlas(StringView name) noexcept(false)
+std::shared_ptr<class wall_atlas>
+loader_impl::get_wall_atlas(StringView name) noexcept(false)
{
- fm_assert(name != "<invalid>"_s);
- char buf[fm_FILENAME_MAX];
- auto filename = make_atlas_path(buf, loader.WALL_TILESET_PATH, name);
- auto def = wall_atlas_def::deserialize(""_s.join({filename, ".json"_s}));
- auto tex = texture(""_s, filename);
-
- fm_soft_assert(name == def.header.name);
- fm_soft_assert(!def.frames.isEmpty());
- auto atlas = std::make_shared<class wall_atlas>(std::move(def), loader.WALL_TILESET_PATH, tex);
- return atlas;
+ return _wall_loader->make_atlas(name, {});
}
-} // namespace floormat
-
-namespace floormat::loader_detail {
-
-const wall_cell& loader_impl::make_invalid_wall_atlas()
+atlas_loader<class wall_atlas>* loader_impl::make_wall_atlas_loader()
{
- if (invalid_wall_atlas) [[likely]]
- return *invalid_wall_atlas;
-
- constexpr auto name = "<invalid>"_s;
- constexpr auto frame_size = Vector2ui{iTILE_SIZE.x(), iTILE_SIZE.z()};
-
- auto a = std::make_shared<class wall_atlas>(
- wall_atlas_def {
- Wall::Info{.name = name, .depth = 8},
- array<Wall::Frame>({{ {}, frame_size}, }),
- array<Wall::Direction>({{ {.index = 0, .count = 1, .pixel_size = frame_size, .is_defined = true, } } }),
- {{ {.val = 0}, {}, }},
- {1u},
- }, name, make_error_texture(frame_size));
- invalid_wall_atlas = Pointer<wall_cell>{InPlaceInit, wall_cell{ .name = name, .atlas = std::move(a) } };
- return *invalid_wall_atlas;
+ return new atlas_loader<class wall_atlas>;
}
-std::shared_ptr<class wall_atlas> loader_impl::wall_atlas(StringView name, loader_policy policy) noexcept(false)
+ArrayView<const wall_cell> loader_impl::wall_atlas_list() noexcept(false)
{
- (void)wall_atlas_list();
-
- switch (policy)
- {
- case loader_policy::error:
- fm_assert(name != INVALID);
- break;
- case loader_policy::ignore:
- case loader_policy::warn:
- break;
- default:
- fm_abort("invalid loader_policy");
- }
-
- fm_soft_assert(check_atlas_name(name));
- auto it = wall_atlas_map.find(name);
-
- if (it != wall_atlas_map.end()) [[likely]]
- {
- if (it->second == (wall_cell*)-1) [[unlikely]]
- {
- switch (policy)
- {
- case loader_policy::error:
- goto error;
- case loader_policy::warn:
- case loader_policy::ignore:
- goto missing_ok;
- }
- std::unreachable();
- }
- else if (!it->second->atlas)
- return it->second->atlas = get_wall_atlas(name);
- else
- return it->second->atlas;
- }
- else
- {
- switch (policy)
- {
- case loader_policy::error:
- goto error;
- case loader_policy::warn:
- goto missing_warn;
- case loader_policy::ignore:
- goto missing_ok;
- }
- std::unreachable();
- }
-
-missing_warn:
- {
- missing_wall_atlases.push_back(String { AllocatedInit, name });
- auto string_view = StringView{missing_wall_atlases.back()};
- wall_atlas_map[string_view] = (wall_cell*)-1;
- }
-
- if (name != "<invalid>")
- DBG_nospace << "wall_atlas '" << name << "' doesn't exist";
-
-missing_ok:
- return make_invalid_wall_atlas().atlas;
-
-error:
- fm_throw("no such wall atlas '{}'"_cf, name);
+ return _wall_loader->ensure_atlas_list();
}
-void loader_impl::get_wall_atlas_list()
+const wall_cell& loader_impl::make_invalid_wall_atlas()
{
- fm_assert(wall_atlas_map.empty());
-
- wall_atlas_array = wall_cell::load_atlases_from_json().vec;
- wall_atlas_array.shrink_to_fit();
- wall_atlas_map.clear();
- wall_atlas_map.reserve(wall_atlas_array.size()*2);
-
- for (auto& x : wall_atlas_array)
- {
- fm_soft_assert(x.name != "<invalid>"_s);
- fm_soft_assert(check_atlas_name(x.name));
- StringView name = x.name;
- wall_atlas_map[name] = &x;
- fm_debug_assert(name.data() == wall_atlas_map[name]->name.data());
- }
-
- fm_assert(!wall_atlas_map.empty());
+ return _wall_loader->get_invalid_atlas();
}
-ArrayView<const wall_cell> loader_impl::wall_atlas_list()
+const std::shared_ptr<class wall_atlas>&
+loader_impl::wall_atlas(StringView filename, loader_policy policy) noexcept(false)
{
- if (wall_atlas_map.empty()) [[unlikely]]
- {
- get_wall_atlas_list();
- fm_assert(!wall_atlas_map.empty());
- }
- return wall_atlas_array;
+ return _wall_loader->get_atlas(filename, policy);
}
} // namespace floormat::loader_detail