diff options
Diffstat (limited to 'loader/wall-atlas.cpp')
-rw-r--r-- | loader/wall-atlas.cpp | 150 |
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 |