#pragma once #include "./texture_uploader.hpp" #include #include #include #include #include #include #include struct TextureEntry { uint32_t width {0}; uint32_t height {0}; std::vector textures; std::vector frame_duration; // ms size_t current_texture {0}; bool rendered_this_frame {false}; // or flipped for animations int64_t timestamp_last_rendered {0}; // ms TextureEntry(void) = default; TextureEntry(const TextureEntry& other) : width(other.width), height(other.height), textures(other.textures), frame_duration(other.frame_duration), current_texture(other.current_texture), rendered_this_frame(other.rendered_this_frame), timestamp_last_rendered(other.timestamp_last_rendered) {} TextureEntry& operator=(const TextureEntry& other) { width = other.width; height = other.height; textures = other.textures; frame_duration = other.frame_duration; current_texture = other.current_texture; rendered_this_frame = other.rendered_this_frame; timestamp_last_rendered = other.timestamp_last_rendered; return *this; } uint32_t getDuration(void) const { return frame_duration.at(current_texture); } void next(void) { current_texture = (current_texture + 1) % frame_duration.size(); } // returns ts for next frame int64_t doAnimation(const int64_t ts_now); template TextureType getID(void) { static_assert( sizeof(TextureType) == sizeof(uint64_t) || sizeof(TextureType) == sizeof(uint32_t) ); rendered_this_frame = true; assert(current_texture < textures.size()); if constexpr (sizeof(TextureType) == sizeof(uint64_t)) { return reinterpret_cast(textures.at(current_texture)); } else if constexpr (sizeof(TextureType) == sizeof(uint32_t)) { return reinterpret_cast(static_cast(textures.at(current_texture))); } } }; struct TextureLoaderResult { std::optional texture; bool keep_trying{false}; // if set, cant be cleared, as some async might be going on }; TextureEntry generateTestAnim(TextureUploaderI& tu); template struct TextureCache { static_assert( sizeof(TextureType) == sizeof(uint64_t) || sizeof(TextureType) == sizeof(uint32_t) ); Loader& _l; TextureUploaderI& _tu; TextureEntry _default_texture; entt::dense_map _cache; entt::dense_set _to_load; // to_reload // to_update? _marked_stale? const uint64_t ms_before_purge {60 * 1000ull}; const size_t min_count_before_purge {0}; // starts purging after that TextureCache(Loader& l, TextureUploaderI& tu) : _l(l), _tu(tu) { //_image_loaders.push_back(std::make_unique()); //_image_loaders.push_back(std::make_unique()); _default_texture = generateTestAnim(_tu); //_default_texture = loadTestWebPAnim(); } ~TextureCache(void) { for (const auto& it : _cache) { for (const auto& tex_id : it.second.textures) { _tu.destroy(tex_id); } } } struct GetInfo { TextureType id; uint32_t width; uint32_t height; }; GetInfo get(const KeyType& key) { auto it = _cache.find(key); if (it != _cache.end()) { return { it->second.template getID(), it->second.width, it->second.height }; } else { _to_load.insert(key); return { _default_texture.getID(), _default_texture.width, _default_texture.height }; } } // markes a texture as stale and will reload it // only if it already is loaded, does not update ts bool stale(const KeyType& key) { auto it = _cache.find(key); if (it == _cache.end()) { return false; } _to_load.insert(key); return true; } float update(void) { const uint64_t ts_now = getTimeMS(); uint64_t ts_min_next = ts_now + ms_before_purge; std::vector to_purge; for (auto&& [key, te] : _cache) { if (te.rendered_this_frame) { const uint64_t ts_next = te.doAnimation(ts_now); te.rendered_this_frame = false; ts_min_next = std::min(ts_min_next, ts_next); } else if ( _cache.size() > min_count_before_purge && ts_now - te.timestamp_last_rendered >= ms_before_purge ) { to_purge.push_back(key); } } invalidate(to_purge); // we ignore the default texture ts :) _default_texture.doAnimation(ts_now); return (ts_min_next - ts_now) / 1000.f; } void invalidate(const std::vector& to_purge) { for (const auto& key : to_purge) { if (_to_load.count(key)) { // TODO: only remove if not keep trying _to_load.erase(key); } if (_cache.count(key)) { for (const auto& tex_id : _cache.at(key).textures) { _tu.destroy(tex_id); } _cache.erase(key); } } } // returns true if there is still work queued up bool workLoadQueue(void) { auto it = _to_load.cbegin(); for (; it != _to_load.cend(); it++) { auto new_entry_opt = _l.load(_tu, *it); if (_cache.count(*it)) { if (new_entry_opt.texture.has_value()) { auto old_entry = _cache.at(*it); // copy assert(!old_entry.textures.empty()); for (const auto& tex_id : old_entry.textures) { _tu.destroy(tex_id); } _cache.erase(*it); auto& new_entry = _cache[*it] = new_entry_opt.texture.value(); // TODO: make update interface and let loader handle this //new_entry.current_texture = old_entry.current_texture; // ?? new_entry.rendered_this_frame = old_entry.rendered_this_frame; new_entry.timestamp_last_rendered = old_entry.timestamp_last_rendered; it = _to_load.erase(it); // TODO: not a good idea? break; // end load from queue/onlyload 1 per update } else if (!new_entry_opt.keep_trying) { // failed to load and the loader is done it = _to_load.erase(it); } } else { if (new_entry_opt.texture.has_value()) { _cache.emplace(*it, new_entry_opt.texture.value()); _cache.at(*it).rendered_this_frame = true; // ? it = _to_load.erase(it); // TODO: not a good idea? break; // end load from queue/onlyload 1 per update } else if (!new_entry_opt.keep_trying) { // failed to load and the loader is done it = _to_load.erase(it); } } } // peek return it != _to_load.cend(); } };