2021-07-31 18:48:47 +01:00
|
|
|
#ifndef VOX_DATA_H
|
|
|
|
#define VOX_DATA_H
|
|
|
|
|
|
|
|
#include "../util/fixed_array.h"
|
|
|
|
#include "../util/math/color8.h"
|
|
|
|
#include "../util/math/vector3i.h"
|
|
|
|
|
|
|
|
#include <core/math/basis.h>
|
2021-12-13 21:38:10 +00:00
|
|
|
#include <core/string/ustring.h>
|
|
|
|
#include <memory>
|
2021-07-31 18:48:47 +01:00
|
|
|
#include <unordered_map>
|
|
|
|
#include <vector>
|
|
|
|
|
|
|
|
namespace std {
|
2022-01-04 22:15:57 +00:00
|
|
|
template <>
|
|
|
|
struct hash<String> {
|
2021-07-31 18:48:47 +01:00
|
|
|
size_t operator()(const String &v) const {
|
|
|
|
return v.hash();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
} // namespace std
|
|
|
|
|
2022-01-03 02:35:40 +00:00
|
|
|
namespace zylann::voxel::magica {
|
2021-07-31 18:48:47 +01:00
|
|
|
|
|
|
|
struct Model {
|
|
|
|
Vector3i size;
|
|
|
|
// TODO Optimization: implement lazy loading/streaming to reduce intermediary memory allocations?
|
|
|
|
// Loading a full 256^3 model needs 16 megabytes, but a lot of areas might actually be uniform,
|
|
|
|
// and we might not need the actual model immediately
|
|
|
|
std::vector<uint8_t> color_indexes;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct Node {
|
2022-01-03 02:35:40 +00:00
|
|
|
enum Type { //
|
|
|
|
TYPE_TRANSFORM = 0,
|
|
|
|
TYPE_GROUP,
|
|
|
|
TYPE_SHAPE
|
|
|
|
};
|
2021-07-31 18:48:47 +01:00
|
|
|
|
|
|
|
int id;
|
|
|
|
// Depending on the type, a node pointer can be casted to different structs
|
|
|
|
const Type type;
|
|
|
|
std::unordered_map<String, String> attributes;
|
|
|
|
|
2021-12-13 21:38:10 +00:00
|
|
|
Node(Type p_type) : type(p_type) {}
|
2021-07-31 18:48:47 +01:00
|
|
|
|
|
|
|
virtual ~Node() {}
|
|
|
|
};
|
|
|
|
|
|
|
|
struct Rotation {
|
|
|
|
uint8_t data;
|
|
|
|
Basis basis;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct TransformNode : public Node {
|
|
|
|
int child_node_id;
|
|
|
|
int layer_id;
|
2021-08-14 16:49:46 +01:00
|
|
|
// Pivot position, which turns out to be at the center in MagicaVoxel
|
2021-07-31 18:48:47 +01:00
|
|
|
Vector3i position;
|
|
|
|
Rotation rotation;
|
|
|
|
String name;
|
|
|
|
bool hidden;
|
|
|
|
|
2021-12-13 21:38:10 +00:00
|
|
|
TransformNode() : Node(Node::TYPE_TRANSFORM) {}
|
2021-07-31 18:48:47 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
struct GroupNode : public Node {
|
|
|
|
std::vector<int> child_node_ids;
|
|
|
|
|
2021-12-13 21:38:10 +00:00
|
|
|
GroupNode() : Node(Node::TYPE_GROUP) {}
|
2021-07-31 18:48:47 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
struct ShapeNode : public Node {
|
|
|
|
int model_id; // corresponds to index in the array of models
|
|
|
|
std::unordered_map<String, String> model_attributes;
|
|
|
|
|
2021-12-13 21:38:10 +00:00
|
|
|
ShapeNode() : Node(Node::TYPE_SHAPE) {}
|
2021-07-31 18:48:47 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
struct Layer {
|
|
|
|
int id;
|
|
|
|
std::unordered_map<String, String> attributes;
|
|
|
|
String name;
|
|
|
|
bool hidden;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct Material {
|
2022-01-03 02:35:40 +00:00
|
|
|
enum Type { //
|
|
|
|
TYPE_DIFFUSE,
|
|
|
|
TYPE_METAL,
|
|
|
|
TYPE_GLASS,
|
|
|
|
TYPE_EMIT
|
|
|
|
};
|
2021-07-31 18:48:47 +01:00
|
|
|
int id;
|
|
|
|
Type type = TYPE_DIFFUSE;
|
|
|
|
float weight = 0.f;
|
|
|
|
float roughness = 1.f;
|
|
|
|
float specular = 0.f;
|
|
|
|
float ior = 1.f; // refraction
|
|
|
|
float flux = 0.f;
|
|
|
|
// TODO I don't know what `_att` means
|
|
|
|
float att = 0.f;
|
|
|
|
// TODO WTF is `_plastic`?
|
|
|
|
};
|
|
|
|
|
|
|
|
class Data {
|
|
|
|
public:
|
|
|
|
void clear();
|
|
|
|
Error load_from_file(String fpath);
|
|
|
|
|
|
|
|
unsigned int get_model_count() const;
|
|
|
|
const Model &get_model(unsigned int index) const;
|
|
|
|
|
|
|
|
// Can return -1 if there is no scene graph
|
|
|
|
int get_root_node_id() const;
|
|
|
|
const Node *get_node(int id) const;
|
|
|
|
|
|
|
|
unsigned int get_layer_count() const;
|
|
|
|
const Layer &get_layer_by_index(unsigned int index) const;
|
|
|
|
|
|
|
|
const int get_material_id_for_palette_index(unsigned int palette_index) const;
|
|
|
|
const Material &get_material_by_id(int id) const;
|
|
|
|
|
|
|
|
inline const FixedArray<Color8, 256> &get_palette() const {
|
|
|
|
return _palette;
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
Error _load_from_file(String fpath);
|
|
|
|
|
|
|
|
std::vector<std::unique_ptr<Model>> _models;
|
|
|
|
std::vector<std::unique_ptr<Layer>> _layers;
|
|
|
|
std::unordered_map<int, std::unique_ptr<Node>> _scene_graph;
|
|
|
|
// Material IDs are supposedly tied to palette indices
|
|
|
|
std::unordered_map<int, std::unique_ptr<Material>> _materials;
|
|
|
|
int _root_node_id = -1;
|
|
|
|
FixedArray<Color8, 256> _palette;
|
|
|
|
};
|
|
|
|
|
2022-01-03 02:35:40 +00:00
|
|
|
} // namespace zylann::voxel::magica
|
2021-07-31 18:48:47 +01:00
|
|
|
|
|
|
|
#endif // VOX_DATA_H
|