2023-08-02 19:24:51 +02:00
|
|
|
#include "./media_meta_info_loader.hpp"
|
|
|
|
|
|
|
|
#include "./image_loader_webp.hpp"
|
|
|
|
#include "./image_loader_sdl_bmp.hpp"
|
2024-03-04 13:38:55 +01:00
|
|
|
#include "./image_loader_qoi.hpp"
|
2024-04-16 10:23:17 +02:00
|
|
|
#include "./image_loader_sdl_image.hpp"
|
2023-08-02 19:24:51 +02:00
|
|
|
|
|
|
|
#include <solanaceae/message3/components.hpp>
|
|
|
|
|
2024-07-31 18:10:52 +02:00
|
|
|
#include "./os_comps.hpp"
|
|
|
|
|
|
|
|
#include <solanaceae/object_store/object_store.hpp>
|
|
|
|
|
|
|
|
#include <solanaceae/file/file2.hpp>
|
|
|
|
|
|
|
|
#include <limits>
|
2023-08-02 19:24:51 +02:00
|
|
|
#include <iostream>
|
|
|
|
|
2023-08-02 20:58:16 +02:00
|
|
|
void MediaMetaInfoLoader::handleMessage(const Message3Handle& m) {
|
2024-07-31 18:10:52 +02:00
|
|
|
if (!static_cast<bool>(m)) {
|
2023-08-02 20:58:16 +02:00
|
|
|
return;
|
2023-08-02 19:24:51 +02:00
|
|
|
}
|
|
|
|
|
2024-07-31 18:10:52 +02:00
|
|
|
// move to obj
|
|
|
|
if (m.any_of<Message::Components::TagNotImage>()) {
|
2023-08-02 20:58:16 +02:00
|
|
|
return;
|
2023-08-02 19:24:51 +02:00
|
|
|
}
|
|
|
|
|
2024-07-31 18:10:52 +02:00
|
|
|
if (!m.all_of<Message::Components::MessageFileObject>()) {
|
|
|
|
// not a file message
|
2023-08-02 20:58:16 +02:00
|
|
|
return;
|
2023-08-02 19:24:51 +02:00
|
|
|
}
|
2024-07-31 18:10:52 +02:00
|
|
|
const auto& o = m.get<Message::Components::MessageFileObject>().o;
|
2023-08-02 19:24:51 +02:00
|
|
|
|
2024-07-31 18:10:52 +02:00
|
|
|
if (!static_cast<bool>(o)) {
|
|
|
|
std::cerr << "MMIL error: invalid object in file message\n";
|
|
|
|
return;
|
|
|
|
}
|
2023-08-02 19:24:51 +02:00
|
|
|
|
2024-07-31 18:10:52 +02:00
|
|
|
if (o.any_of<ObjComp::F::FrameDims>()) {
|
|
|
|
return;
|
|
|
|
}
|
2023-08-02 19:24:51 +02:00
|
|
|
|
2024-07-31 18:10:52 +02:00
|
|
|
if (!o.all_of<ObjComp::F::TagLocalHaveAll>()) {
|
|
|
|
return; // we dont have all data
|
|
|
|
}
|
2023-08-02 19:24:51 +02:00
|
|
|
|
2024-07-31 18:10:52 +02:00
|
|
|
if (!o.all_of<ObjComp::Ephemeral::Backend, ObjComp::F::SingleInfo>()) {
|
|
|
|
std::cerr << "MMIL error: object missing backend/file info (?)\n";
|
|
|
|
return;
|
|
|
|
}
|
2023-08-02 19:24:51 +02:00
|
|
|
|
2024-07-31 18:10:52 +02:00
|
|
|
// TODO: handle collections
|
|
|
|
const auto file_size = o.get<ObjComp::F::SingleInfo>().file_size;
|
2023-08-02 19:24:51 +02:00
|
|
|
|
2024-07-31 18:10:52 +02:00
|
|
|
if (file_size > 50*1024*1024) {
|
|
|
|
std::cerr << "MMIL error: image file too large\n";
|
|
|
|
return;
|
|
|
|
}
|
2023-08-02 19:24:51 +02:00
|
|
|
|
2024-07-31 18:10:52 +02:00
|
|
|
if (file_size == 0) {
|
|
|
|
std::cerr << "MMIL warning: empty file\n";
|
|
|
|
return;
|
|
|
|
}
|
2023-08-02 19:24:51 +02:00
|
|
|
|
2024-07-31 18:10:52 +02:00
|
|
|
if (!o.all_of<ObjComp::F::TagLocalHaveAll>()) {
|
|
|
|
// not ready yet
|
|
|
|
return;
|
|
|
|
}
|
2023-08-02 19:24:51 +02:00
|
|
|
|
2024-07-31 18:10:52 +02:00
|
|
|
auto* file_backend = o.get<ObjComp::Ephemeral::Backend>().ptr;
|
|
|
|
if (file_backend == nullptr) {
|
|
|
|
std::cerr << "MMIL error: object backend nullptr\n";
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
auto file2 = file_backend->file2(o, StorageBackendI::FILE2_READ);
|
|
|
|
if (!file2 || !file2->isGood() || !file2->can_read) {
|
|
|
|
std::cerr << "MMIL error: creating file2 from object via backendI\n";
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto read_data = file2->read(file_size, 0);
|
|
|
|
if (read_data.ptr == nullptr) {
|
|
|
|
std::cerr << "MMIL error: reading from file2 returned nullptr\n";
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (read_data.size != file_size) {
|
|
|
|
std::cerr << "MMIL error: reading from file2 size missmatch, should be " << file_size << ", is " << read_data.size << "\n";
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// try all loaders after another
|
|
|
|
for (auto& il : _image_loaders) {
|
|
|
|
// TODO: impl callback based load
|
|
|
|
auto res = il->loadInfoFromMemory(read_data.ptr, read_data.size);
|
|
|
|
if (res.height == 0 || res.width == 0) {
|
|
|
|
continue;
|
2023-08-02 19:24:51 +02:00
|
|
|
}
|
2024-07-31 18:10:52 +02:00
|
|
|
|
|
|
|
o.emplace<ObjComp::F::FrameDims>(
|
|
|
|
static_cast<uint16_t>(std::min<uint32_t>(res.width, std::numeric_limits<uint16_t>::max())),
|
|
|
|
static_cast<uint16_t>(std::min<uint32_t>(res.height, std::numeric_limits<uint16_t>::max()))
|
|
|
|
);
|
|
|
|
|
|
|
|
std::cout << "MMIL: loaded image file o:" << /*file_path*/ entt::to_integral(o.entity()) << "\n";
|
|
|
|
|
|
|
|
_rmm.throwEventUpdate(m);
|
|
|
|
return;
|
2023-08-02 19:24:51 +02:00
|
|
|
}
|
2024-07-31 18:10:52 +02:00
|
|
|
|
|
|
|
m.emplace<Message::Components::TagNotImage>();
|
|
|
|
|
|
|
|
std::cout << "MMIL: loading failed image info o:" << /*file_path*/ entt::to_integral(o.entity()) << "\n";
|
|
|
|
|
|
|
|
// TODO: update object too
|
|
|
|
// recursion
|
|
|
|
_rmm.throwEventUpdate(m);
|
2023-08-02 20:58:16 +02:00
|
|
|
}
|
2023-08-02 19:24:51 +02:00
|
|
|
|
2023-08-02 20:58:16 +02:00
|
|
|
MediaMetaInfoLoader::MediaMetaInfoLoader(RegistryMessageModel& rmm) : _rmm(rmm) {
|
|
|
|
// HACK: make them be added externally?
|
|
|
|
_image_loaders.push_back(std::make_unique<ImageLoaderWebP>());
|
|
|
|
_image_loaders.push_back(std::make_unique<ImageLoaderSDLBMP>());
|
2024-03-04 13:38:55 +01:00
|
|
|
_image_loaders.push_back(std::make_unique<ImageLoaderQOI>());
|
2024-04-16 10:23:17 +02:00
|
|
|
_image_loaders.push_back(std::make_unique<ImageLoaderSDLImage>());
|
2023-08-02 20:58:16 +02:00
|
|
|
|
|
|
|
_rmm.subscribe(this, RegistryMessageModel_Event::message_construct);
|
|
|
|
_rmm.subscribe(this, RegistryMessageModel_Event::message_updated);
|
|
|
|
}
|
|
|
|
|
|
|
|
MediaMetaInfoLoader::~MediaMetaInfoLoader(void) {
|
|
|
|
}
|
|
|
|
|
|
|
|
bool MediaMetaInfoLoader::onEvent(const Message::Events::MessageConstruct& e) {
|
|
|
|
handleMessage(e.e);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool MediaMetaInfoLoader::onEvent(const Message::Events::MessageUpdated& e) {
|
|
|
|
handleMessage(e.e);
|
2023-08-02 19:24:51 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|