1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357 |
- #pragma once
- #include <json/json.h>
- #include <functional>
- #include <stack>
- #include <string>
- #include <string_view>
- #include <variant>
- #include <vector>
- #include <array>
- #include <optional>
- #include <limits>
- #include <stdexcept>
- #include <unordered_map>
- #include <unordered_set>
- #include "util/base64.h"
- namespace tiniergltf {
- static inline void check(bool cond) {
- if (!cond)
- throw std::runtime_error("invalid glTF");
- }
- template <typename T>
- static inline void checkIndex(const std::optional<std::vector<T>> &vec,
- const std::optional<std::size_t> &i) {
- if (!i.has_value()) return;
- check(vec.has_value());
- check(i < vec->size());
- }
- template <typename T>
- static inline void checkIndex(const std::vector<T> &vec,
- const std::optional<std::size_t> &i) {
- if (!i.has_value()) return;
- check(i < vec.size());
- }
- template <typename T, typename F>
- static inline void checkForall(const std::optional<std::vector<T>> &vec, const F &cond) {
- if (!vec.has_value())
- return;
- for (const T &v : vec.value())
- cond(v);
- }
- template <typename T>
- static inline void checkDuplicateFree(const std::vector<T> &vec) {
- check(std::unordered_set<T>(vec.begin(), vec.end()).size() == vec.size());
- }
- template <typename T>
- static inline T as(const Json::Value &o);
- template<>
- bool as(const Json::Value &o) {
- check(o.isBool());
- return o.asBool();
- }
- template<>
- double as (const Json::Value &o) {
- check(o.isDouble());
- return o.asDouble();
- }
- template<>
- std::size_t as(const Json::Value &o) {
- check(o.isUInt64());
- auto u = o.asUInt64();
- check(u <= std::numeric_limits<std::size_t>::max());
- return u;
- }
- template<>
- std::string as(const Json::Value &o) {
- check(o.isString());
- return o.asString();
- }
- template<typename U>
- std::vector<U> asVec(const Json::Value &o) {
- check(o.isArray());
- std::vector<U> res;
- res.reserve(o.size());
- for (Json::ArrayIndex i = 0; i < o.size(); ++i) {
- res.push_back(as<U>(o[i]));
- }
- return res;
- }
- template<typename U, std::size_t n>
- std::array<U, n> asArr(const Json::Value &o) {
- check(o.isArray());
- check(o.size() == n);
- std::array<U, n> res;
- for (Json::ArrayIndex i = 0; i < n; ++i) {
- res[i] = as<U>(o[i]);
- }
- return res;
- }
- struct AccessorSparseIndices {
- std::size_t bufferView;
- std::size_t byteOffset;
- // as defined in the glTF specification
- enum class ComponentType {
- UNSIGNED_BYTE,
- UNSIGNED_SHORT,
- UNSIGNED_INT,
- };
- ComponentType componentType;
- std::size_t componentSize() const {
- switch (componentType) {
- case ComponentType::UNSIGNED_BYTE:
- return 1;
- case ComponentType::UNSIGNED_SHORT:
- return 2;
- case ComponentType::UNSIGNED_INT:
- return 4;
- }
- throw std::logic_error("invalid component type");
- }
- std::size_t elementSize() const {
- return componentSize();
- }
- AccessorSparseIndices(const Json::Value &o)
- : bufferView(as<std::size_t>(o["bufferView"]))
- , byteOffset(0)
- {
- check(o.isObject());
- if (o.isMember("byteOffset")) {
- byteOffset = as<std::size_t>(o["byteOffset"]);
- check(byteOffset >= 0);
- }
- {
- static std::unordered_map<Json::UInt64, ComponentType> map = {
- {5121, ComponentType::UNSIGNED_BYTE},
- {5123, ComponentType::UNSIGNED_SHORT},
- {5125, ComponentType::UNSIGNED_INT},
- };
- const auto &v = o["componentType"]; check(v.isUInt64());
- componentType = map.at(v.asUInt64());
- }
- }
- };
- template<> AccessorSparseIndices as(const Json::Value &o) { return o; }
- struct AccessorSparseValues {
- std::size_t bufferView;
- std::size_t byteOffset;
- AccessorSparseValues(const Json::Value &o)
- : bufferView(as<std::size_t>(o["bufferView"]))
- , byteOffset(0)
- {
- check(o.isObject());
- if (o.isMember("byteOffset")) {
- byteOffset = as<std::size_t>(o["byteOffset"]);
- check(byteOffset >= 0);
- }
- }
- };
- template<> AccessorSparseValues as(const Json::Value &o) { return o; }
- struct AccessorSparse {
- std::size_t count;
- AccessorSparseIndices indices;
- AccessorSparseValues values;
- AccessorSparse(const Json::Value &o)
- : count(as<std::size_t>(o["count"]))
- , indices(as<AccessorSparseIndices>(o["indices"]))
- , values(as<AccessorSparseValues>(o["values"]))
- {
- check(o.isObject());
- check(count >= 1);
- }
- };
- template<> AccessorSparse as(const Json::Value &o) { return o; }
- struct Accessor {
- std::optional<std::size_t> bufferView;
- std::size_t byteOffset;
- // as defined in the glTF specification
- enum class ComponentType {
- BYTE,
- UNSIGNED_BYTE,
- SHORT,
- UNSIGNED_SHORT,
- UNSIGNED_INT,
- FLOAT,
- };
- ComponentType componentType;
- std::size_t componentSize() const {
- switch (componentType) {
- case ComponentType::BYTE:
- case ComponentType::UNSIGNED_BYTE:
- return 1;
- case ComponentType::SHORT:
- case ComponentType::UNSIGNED_SHORT:
- return 2;
- case ComponentType::UNSIGNED_INT:
- case ComponentType::FLOAT:
- return 4;
- }
- throw std::logic_error("invalid component type");
- }
- std::size_t count;
- std::optional<std::vector<double>> max;
- std::optional<std::vector<double>> min;
- std::optional<std::string> name;
- bool normalized;
- std::optional<AccessorSparse> sparse;
- enum class Type {
- MAT2,
- MAT3,
- MAT4,
- SCALAR,
- VEC2,
- VEC3,
- VEC4,
- };
- std::size_t typeCount() const {
- switch (type) {
- case Type::SCALAR:
- return 1;
- case Type::VEC2:
- return 2;
- case Type::VEC3:
- return 3;
- case Type::MAT2:
- case Type::VEC4:
- return 4;
- case Type::MAT3:
- return 9;
- case Type::MAT4:
- return 16;
- }
- throw std::logic_error("invalid type");
- }
- Type type;
- std::size_t elementSize() const {
- return componentSize() * typeCount();
- }
- Accessor(const Json::Value &o)
- : byteOffset(0)
- , count(as<std::size_t>(o["count"]))
- , normalized(false)
- {
- check(o.isObject());
- if (o.isMember("bufferView")) {
- bufferView = as<std::size_t>(o["bufferView"]);
- }
- {
- static std::unordered_map<Json::UInt64, ComponentType> map = {
- {5120, ComponentType::BYTE},
- {5121, ComponentType::UNSIGNED_BYTE},
- {5122, ComponentType::SHORT},
- {5123, ComponentType::UNSIGNED_SHORT},
- {5125, ComponentType::UNSIGNED_INT},
- {5126, ComponentType::FLOAT},
- };
- const auto &v = o["componentType"]; check(v.isUInt64());
- componentType = map.at(v.asUInt64());
- }
- if (o.isMember("byteOffset")) {
- byteOffset = as<std::size_t>(o["byteOffset"]);
- check(byteOffset >= 0);
- check(byteOffset % componentSize() == 0);
- }
- check(count >= 1);
- if (o.isMember("name")) {
- name = as<std::string>(o["name"]);
- }
- if (o.isMember("normalized")) {
- normalized = as<bool>(o["normalized"]);
- }
- if (o.isMember("sparse")) {
- sparse = as<AccessorSparse>(o["sparse"]);
- }
- {
- static std::unordered_map<Json::String, Type> map = {
- {"MAT2", Type::MAT2},
- {"MAT3", Type::MAT3},
- {"MAT4", Type::MAT4},
- {"SCALAR", Type::SCALAR},
- {"VEC2", Type::VEC2},
- {"VEC3", Type::VEC3},
- {"VEC4", Type::VEC4},
- };
- const auto &v = o["type"]; check(v.isString());
- type = map.at(v.asString());
- }
- if (o.isMember("max")) {
- max = asVec<double>(o["max"]);
- check(max->size() == typeCount());
- }
- if (o.isMember("min")) {
- min = asVec<double>(o["min"]);
- check(min->size() == typeCount());
- }
- }
- };
- template<> Accessor as(const Json::Value &o) { return o; }
- struct AnimationChannelTarget {
- std::optional<std::size_t> node;
- enum class Path {
- ROTATION,
- SCALE,
- TRANSLATION,
- WEIGHTS,
- };
- Path path;
- AnimationChannelTarget(const Json::Value &o)
- {
- check(o.isObject());
- if (o.isMember("node")) {
- node = as<std::size_t>(o["node"]);
- }
- {
- static std::unordered_map<Json::String, Path> map = {
- {"rotation", Path::ROTATION},
- {"scale", Path::SCALE},
- {"translation", Path::TRANSLATION},
- {"weights", Path::WEIGHTS},
- };
- const auto &v = o["path"]; check(v.isString());
- path = map.at(v.asString());
- }
- }
- };
- template<> AnimationChannelTarget as(const Json::Value &o) { return o; }
- struct AnimationChannel {
- std::size_t sampler;
- AnimationChannelTarget target;
- AnimationChannel(const Json::Value &o)
- : sampler(as<std::size_t>(o["sampler"]))
- , target(as<AnimationChannelTarget>(o["target"]))
- {
- check(o.isObject());
- }
- };
- template<> AnimationChannel as(const Json::Value &o) { return o; }
- struct AnimationSampler {
- std::size_t input;
- enum class Interpolation {
- CUBICSPLINE,
- LINEAR,
- STEP,
- };
- Interpolation interpolation;
- std::size_t output;
- AnimationSampler(const Json::Value &o)
- : input(as<std::size_t>(o["input"]))
- , interpolation(Interpolation::LINEAR)
- , output(as<std::size_t>(o["output"]))
- {
- check(o.isObject());
- if (o.isMember("interpolation")) {
- static std::unordered_map<Json::String, Interpolation> map = {
- {"CUBICSPLINE", Interpolation::CUBICSPLINE},
- {"LINEAR", Interpolation::LINEAR},
- {"STEP", Interpolation::STEP},
- };
- const auto &v = o["interpolation"]; check(v.isString());
- interpolation = map.at(v.asString());
- }
- }
- };
- template<> AnimationSampler as(const Json::Value &o) { return o; }
- struct Animation {
- std::vector<AnimationChannel> channels;
- std::optional<std::string> name;
- std::vector<AnimationSampler> samplers;
- Animation(const Json::Value &o)
- : channels(asVec<AnimationChannel>(o["channels"]))
- , samplers(asVec<AnimationSampler>(o["samplers"]))
- {
- check(o.isObject());
- check(channels.size() >= 1);
- if (o.isMember("name")) {
- name = as<std::string>(o["name"]);
- }
- check(samplers.size() >= 1);
- }
- };
- template<> Animation as(const Json::Value &o) { return o; }
- struct Asset {
- std::optional<std::string> copyright;
- std::optional<std::string> generator;
- std::optional<std::string> minVersion;
- std::string version;
- Asset(const Json::Value &o)
- : version(as<std::string>(o["version"]))
- {
- check(o.isObject());
- if (o.isMember("copyright")) {
- copyright = as<std::string>(o["copyright"]);
- }
- if (o.isMember("generator")) {
- generator = as<std::string>(o["generator"]);
- }
- if (o.isMember("minVersion")) {
- minVersion = as<std::string>(o["minVersion"]);
- }
- }
- };
- template<> Asset as(const Json::Value &o) { return o; }
- struct BufferView {
- std::size_t buffer;
- std::size_t byteLength;
- std::size_t byteOffset;
- std::optional<std::size_t> byteStride;
- std::optional<std::string> name;
- enum class Target {
- ARRAY_BUFFER,
- ELEMENT_ARRAY_BUFFER,
- };
- std::optional<Target> target;
- BufferView(const Json::Value &o)
- : buffer(as<std::size_t>(o["buffer"]))
- , byteLength(as<std::size_t>(o["byteLength"]))
- , byteOffset(0)
- {
- check(o.isObject());
- check(byteLength >= 1);
- if (o.isMember("byteOffset")) {
- byteOffset = as<std::size_t>(o["byteOffset"]);
- check(byteOffset >= 0);
- }
- if (o.isMember("byteStride")) {
- byteStride = as<std::size_t>(o["byteStride"]);
- check(byteStride.value() >= 4);
- check(byteStride.value() <= 252);
- check(byteStride.value() % 4 == 0);
- }
- if (o.isMember("name")) {
- name = as<std::string>(o["name"]);
- }
- if (o.isMember("target")) {
- static std::unordered_map<Json::UInt64, Target> map = {
- {34962, Target::ARRAY_BUFFER},
- {34963, Target::ELEMENT_ARRAY_BUFFER},
- };
- const auto &v = o["target"]; check(v.isUInt64());
- target = map.at(v.asUInt64());
- }
- }
- };
- template<> BufferView as(const Json::Value &o) { return o; }
- struct Buffer {
- std::size_t byteLength;
- std::optional<std::string> name;
- std::string data;
- Buffer(const Json::Value &o,
- const std::function<std::string(const std::string &uri)> &resolveURI)
- : byteLength(as<std::size_t>(o["byteLength"]))
- {
- check(o.isObject());
- check(byteLength >= 1);
- if (o.isMember("name")) {
- name = as<std::string>(o["name"]);
- }
- check(o.isMember("uri"));
- bool dataURI = false;
- const std::string uri = as<std::string>(o["uri"]);
- for (auto &prefix : std::array<std::string, 2> {
- "data:application/octet-stream;base64,",
- "data:application/gltf-buffer;base64,"
- }) {
- if (std::string_view(uri).substr(0, prefix.length()) == prefix) {
- auto view = std::string_view(uri).substr(prefix.length());
- check(base64_is_valid(view));
- data = base64_decode(view);
- dataURI = true;
- break;
- }
- }
- if (!dataURI)
- data = resolveURI(uri);
- check(data.size() >= byteLength);
- data.resize(byteLength);
- }
- };
- struct CameraOrthographic {
- double xmag;
- double ymag;
- double zfar;
- double znear;
- CameraOrthographic(const Json::Value &o)
- : xmag(as<double>(o["xmag"]))
- , ymag(as<double>(o["ymag"]))
- , zfar(as<double>(o["zfar"]))
- , znear(as<double>(o["znear"]))
- {
- check(o.isObject());
- check(zfar > 0);
- check(znear >= 0);
- }
- };
- template<> CameraOrthographic as(const Json::Value &o) { return o; }
- struct CameraPerspective {
- std::optional<double> aspectRatio;
- double yfov;
- std::optional<double> zfar;
- double znear;
- CameraPerspective(const Json::Value &o)
- : yfov(as<double>(o["yfov"]))
- , znear(as<double>(o["znear"]))
- {
- check(o.isObject());
- if (o.isMember("aspectRatio")) {
- aspectRatio = as<double>(o["aspectRatio"]);
- check(aspectRatio.value() > 0);
- }
- check(yfov > 0);
- if (o.isMember("zfar")) {
- zfar = as<double>(o["zfar"]);
- check(zfar.value() > 0);
- }
- check(znear > 0);
- }
- };
- template<> CameraPerspective as(const Json::Value &o) { return o; }
- struct Camera {
- std::optional<std::string> name;
- std::optional<CameraOrthographic> orthographic;
- std::optional<CameraPerspective> perspective;
- enum class Type {
- ORTHOGRAPHIC,
- PERSPECTIVE,
- };
- Type type;
- Camera(const Json::Value &o)
- {
- check(o.isObject());
- if (o.isMember("name")) {
- name = as<std::string>(o["name"]);
- }
- if (o.isMember("orthographic")) {
- orthographic = as<CameraOrthographic>(o["orthographic"]);
- }
- if (o.isMember("perspective")) {
- perspective = as<CameraPerspective>(o["perspective"]);
- }
- {
- static std::unordered_map<Json::String, Type> map = {
- {"orthographic", Type::ORTHOGRAPHIC},
- {"perspective", Type::PERSPECTIVE},
- };
- const auto &v = o["type"]; check(v.isString());
- type = map.at(v.asString());
- }
- }
- };
- template<> Camera as(const Json::Value &o) { return o; }
- struct Image {
- std::optional<std::size_t> bufferView;
- enum class MimeType {
- IMAGE_JPEG,
- IMAGE_PNG,
- };
- std::optional<MimeType> mimeType;
- std::optional<std::string> name;
- std::optional<std::string> uri;
- Image(const Json::Value &o)
- {
- check(o.isObject());
- if (o.isMember("bufferView")) {
- bufferView = as<std::size_t>(o["bufferView"]);
- }
- if (o.isMember("mimeType")) {
- static std::unordered_map<Json::String, MimeType> map = {
- {"image/jpeg", MimeType::IMAGE_JPEG},
- {"image/png", MimeType::IMAGE_PNG},
- };
- const auto &v = o["mimeType"]; check(v.isString());
- mimeType = map.at(v.asString());
- }
- if (o.isMember("name")) {
- name = as<std::string>(o["name"]);
- }
- if (o.isMember("uri")) {
- uri = as<std::string>(o["uri"]);
- }
- }
- };
- template<> Image as(const Json::Value &o) { return o; }
- struct TextureInfo {
- std::size_t index;
- std::size_t texCoord;
- TextureInfo(const Json::Value &o)
- : index(as<std::size_t>(o["index"]))
- , texCoord(0)
- {
- check(o.isObject());
- if (o.isMember("texCoord")) {
- texCoord = as<std::size_t>(o["texCoord"]);
- check(texCoord >= 0);
- }
- }
- };
- template<> TextureInfo as(const Json::Value &o) { return o; }
- struct MaterialNormalTextureInfo {
- std::size_t index;
- double scale;
- std::size_t texCoord;
- MaterialNormalTextureInfo(const Json::Value &o)
- : index(as<std::size_t>(o["index"]))
- , scale(1)
- , texCoord(0)
- {
- check(o.isObject());
- if (o.isMember("scale")) {
- scale = as<double>(o["scale"]);
- }
- if (o.isMember("texCoord")) {
- texCoord = as<std::size_t>(o["texCoord"]);
- }
- }
- };
- template<> MaterialNormalTextureInfo as(const Json::Value &o) { return o; }
- struct MaterialOcclusionTextureInfo {
- std::size_t index;
- double strength;
- std::size_t texCoord;
- MaterialOcclusionTextureInfo(const Json::Value &o)
- : index(as<std::size_t>(o["index"]))
- , strength(1)
- , texCoord(0)
- {
- check(o.isObject());
- if (o.isMember("strength")) {
- strength = as<double>(o["strength"]);
- check(strength >= 0);
- check(strength <= 1);
- }
- if (o.isMember("texCoord")) {
- texCoord = as<std::size_t>(o["texCoord"]);
- }
- }
- };
- template<> MaterialOcclusionTextureInfo as(const Json::Value &o) { return o; }
- struct MaterialPbrMetallicRoughness {
- std::array<double, 4> baseColorFactor;
- std::optional<TextureInfo> baseColorTexture;
- double metallicFactor;
- std::optional<TextureInfo> metallicRoughnessTexture;
- double roughnessFactor;
- MaterialPbrMetallicRoughness(const Json::Value &o)
- : baseColorFactor{1, 1, 1, 1}
- , metallicFactor(1)
- , roughnessFactor(1)
- {
- check(o.isObject());
- if (o.isMember("baseColorFactor")) {
- baseColorFactor = asArr<double, 4>(o["baseColorFactor"]);
- for (auto v: baseColorFactor) {
- check(v >= 0);
- check(v <= 1);
- }
- }
- if (o.isMember("baseColorTexture")) {
- baseColorTexture = as<TextureInfo>(o["baseColorTexture"]);
- }
- if (o.isMember("metallicFactor")) {
- metallicFactor = as<double>(o["metallicFactor"]);
- check(metallicFactor >= 0);
- check(metallicFactor <= 1);
- }
- if (o.isMember("metallicRoughnessTexture")) {
- metallicRoughnessTexture = as<TextureInfo>(o["metallicRoughnessTexture"]);
- }
- if (o.isMember("roughnessFactor")) {
- roughnessFactor = as<double>(o["roughnessFactor"]);
- check(roughnessFactor >= 0);
- check(roughnessFactor <= 1);
- }
- }
- };
- template<> MaterialPbrMetallicRoughness as(const Json::Value &o) { return o; }
- struct Material {
- double alphaCutoff;
- enum class AlphaMode {
- BLEND,
- MASK,
- OPAQUE,
- };
- AlphaMode alphaMode;
- bool doubleSided;
- std::array<double, 3> emissiveFactor;
- std::optional<TextureInfo> emissiveTexture;
- std::optional<std::string> name;
- std::optional<MaterialNormalTextureInfo> normalTexture;
- std::optional<MaterialOcclusionTextureInfo> occlusionTexture;
- std::optional<MaterialPbrMetallicRoughness> pbrMetallicRoughness;
- Material(const Json::Value &o)
- : alphaCutoff(0.5)
- , alphaMode(AlphaMode::OPAQUE)
- , doubleSided(false)
- , emissiveFactor{0, 0, 0}
- {
- check(o.isObject());
- if (o.isMember("alphaCutoff")) {
- alphaCutoff = as<double>(o["alphaCutoff"]);
- check(alphaCutoff >= 0);
- }
- if (o.isMember("alphaMode")){
- static std::unordered_map<Json::String, AlphaMode> map = {
- {"BLEND", AlphaMode::BLEND},
- {"MASK", AlphaMode::MASK},
- {"OPAQUE", AlphaMode::OPAQUE},
- };
- const auto &v = o["alphaMode"]; check(v.isString());
- alphaMode = map.at(v.asString());
- }
- if (o.isMember("doubleSided")) {
- doubleSided = as<bool>(o["doubleSided"]);
- }
- if (o.isMember("emissiveFactor")) {
- emissiveFactor = asArr<double, 3>(o["emissiveFactor"]);
- for (const auto &v: emissiveFactor) {
- check(v >= 0);
- check(v <= 1);
- }
- }
- if (o.isMember("emissiveTexture")) {
- emissiveTexture = as<TextureInfo>(o["emissiveTexture"]);
- }
- if (o.isMember("name")) {
- name = as<std::string>(o["name"]);
- }
- if (o.isMember("normalTexture")) {
- normalTexture = as<MaterialNormalTextureInfo>(o["normalTexture"]);
- }
- if (o.isMember("occlusionTexture")) {
- occlusionTexture = as<MaterialOcclusionTextureInfo>(o["occlusionTexture"]);
- }
- if (o.isMember("pbrMetallicRoughness")) {
- pbrMetallicRoughness = as<MaterialPbrMetallicRoughness>(o["pbrMetallicRoughness"]);
- }
- }
- };
- template<> Material as(const Json::Value &o) { return o; }
- struct MeshPrimitive {
- static void enumeratedProps(const Json::Value &o, const std::string &name, std::optional<std::vector<std::size_t>> &attr) {
- for (std::size_t i = 0;; ++i) {
- const std::string s = name + "_" + std::to_string(i);
- if (!o.isMember(s)) break;
- if (i == 0) {
- attr = std::vector<std::size_t>();
- }
- attr->push_back(as<std::size_t>(o[s]));
- }
- }
- struct Attributes {
- std::optional<std::size_t> position, normal, tangent;
- std::optional<std::vector<std::size_t>> texcoord, color, joints, weights;
- Attributes(const Json::Value &o) {
- if (o.isMember("POSITION"))
- position = as<std::size_t>(o["POSITION"]);
- if (o.isMember("NORMAL"))
- normal = as<std::size_t>(o["NORMAL"]);
- if (o.isMember("TANGENT"))
- tangent = as<std::size_t>(o["TANGENT"]);
- enumeratedProps(o, "TEXCOORD", texcoord);
- enumeratedProps(o, "COLOR", color);
- enumeratedProps(o, "JOINTS", joints);
- enumeratedProps(o, "WEIGHTS", weights);
- check(joints.has_value() == weights.has_value());
- if (joints.has_value()) {
- check(joints->size() == weights->size());
- }
- check(position.has_value()
- || normal.has_value()
- || tangent.has_value()
- || texcoord.has_value()
- || color.has_value()
- || joints.has_value()
- || weights.has_value());
- }
- };
- Attributes attributes;
- std::optional<std::size_t> indices;
- std::optional<std::size_t> material;
- enum class Mode {
- POINTS,
- LINES,
- LINE_LOOP,
- LINE_STRIP,
- TRIANGLES,
- TRIANGLE_STRIP,
- TRIANGLE_FAN,
- };
- Mode mode;
- struct MorphTargets {
- std::optional<std::size_t> position, normal, tangent;
- std::optional<std::vector<std::size_t>> texcoord, color;
- MorphTargets(const Json::Value &o) {
- if (o.isMember("POSITION"))
- position = as<std::size_t>(o["POSITION"]);
- if (o.isMember("NORMAL"))
- normal = as<std::size_t>(o["NORMAL"]);
- if (o.isMember("TANGENT"))
- tangent = as<std::size_t>(o["TANGENT"]);
- enumeratedProps(o, "TEXCOORD", texcoord);
- enumeratedProps(o, "COLOR", color);
- check(position.has_value()
- || normal.has_value()
- || tangent.has_value()
- || texcoord.has_value()
- || color.has_value());
- }
- };
- std::optional<std::vector<MorphTargets>> targets;
- MeshPrimitive(const Json::Value &o)
- : attributes(Attributes(o["attributes"]))
- , mode(Mode::TRIANGLES)
- {
- check(o.isObject());
- if (o.isMember("indices")) {
- indices = as<std::size_t>(o["indices"]);
- }
- if (o.isMember("material")) {
- material = as<std::size_t>(o["material"]);
- }
- if (o.isMember("mode")) {
- static std::unordered_map<Json::UInt64, Mode> map = {
- {0, Mode::POINTS},
- {1, Mode::LINES},
- {2, Mode::LINE_LOOP},
- {3, Mode::LINE_STRIP},
- {4, Mode::TRIANGLES},
- {5, Mode::TRIANGLE_STRIP},
- {6, Mode::TRIANGLE_FAN},
- };
- const auto &v = o["mode"]; check(v.isUInt64());
- mode = map.at(v.asUInt64());
- }
- if (o.isMember("targets")) {
- targets = asVec<MorphTargets>(o["targets"]);
- check(targets->size() >= 1);
- }
- }
- };
- template<> MeshPrimitive::MorphTargets as(const Json::Value &o) { return o; }
- template<> MeshPrimitive as(const Json::Value &o) { return o; }
- struct Mesh {
- std::optional<std::string> name;
- std::vector<MeshPrimitive> primitives;
- std::optional<std::vector<double>> weights;
- Mesh(const Json::Value &o)
- : primitives(asVec<MeshPrimitive>(o["primitives"]))
- {
- check(o.isObject());
- if (o.isMember("name")) {
- name = as<std::string>(o["name"]);
- }
- check(primitives.size() >= 1);
- if (o.isMember("weights")) {
- weights = asVec<double>(o["weights"]);
- check(weights->size() >= 1);
- }
- }
- };
- template<> Mesh as(const Json::Value &o) { return o; }
- struct Node {
- std::optional<std::size_t> camera;
- std::optional<std::vector<std::size_t>> children;
- typedef std::array<double, 16> Matrix;
- struct TRS {
- std::array<double, 3> translation = {0, 0, 0};
- std::array<double, 4> rotation = {0, 0, 0, 1};
- std::array<double, 3> scale = {1, 1, 1};
- };
- std::variant<Matrix, TRS> transform;
- std::optional<std::size_t> mesh;
- std::optional<std::string> name;
- std::optional<std::size_t> skin;
- std::optional<std::vector<double>> weights;
- Node(const Json::Value &o)
- : transform(Matrix {
- 1, 0, 0, 0,
- 0, 1, 0, 0,
- 0, 0, 1, 0,
- 0, 0, 0, 1
- })
- {
- check(o.isObject());
- if (o.isMember("camera")) {
- camera = as<std::size_t>(o["camera"]);
- }
- if (o.isMember("children")) {
- children = asVec<std::size_t>(o["children"]);
- check(children->size() >= 1);
- checkDuplicateFree(*children);
- }
- bool hasTRS = o.isMember("translation") || o.isMember("rotation") || o.isMember("scale");
- if (o.isMember("matrix")) {
- check(!hasTRS);
- transform = asArr<double, 16>(o["matrix"]);
- } else if (hasTRS) {
- TRS trs;
- if (o.isMember("translation")) {
- trs.translation = asArr<double, 3>(o["translation"]);
- }
- if (o.isMember("rotation")) {
- trs.rotation = asArr<double, 4>(o["rotation"]);
- for (auto v: trs.rotation) {
- check(v >= -1);
- check(v <= 1);
- }
- }
- if (o.isMember("scale")) {
- trs.scale = asArr<double, 3>(o["scale"]);
- }
- transform = trs;
- }
- if (o.isMember("mesh")) {
- mesh = as<std::size_t>(o["mesh"]);
- }
- if (o.isMember("name")) {
- name = as<std::string>(o["name"]);
- }
- if (o.isMember("skin")) {
- check(mesh.has_value());
- skin = as<std::size_t>(o["skin"]);
- }
- if (o.isMember("weights")) {
- weights = asVec<double>(o["weights"]);
- check(weights->size() >= 1);
- }
- }
- };
- template<> Node as(const Json::Value &o) { return o; }
- struct Sampler {
- enum class MagFilter {
- NEAREST,
- LINEAR,
- };
- std::optional<MagFilter> magFilter;
- enum class MinFilter {
- NEAREST,
- LINEAR,
- NEAREST_MIPMAP_NEAREST,
- LINEAR_MIPMAP_NEAREST,
- NEAREST_MIPMAP_LINEAR,
- LINEAR_MIPMAP_LINEAR,
- };
- std::optional<MinFilter> minFilter;
- std::optional<std::string> name;
- enum class WrapS {
- REPEAT,
- CLAMP_TO_EDGE,
- MIRRORED_REPEAT,
- };
- WrapS wrapS;
- enum class WrapT {
- REPEAT,
- CLAMP_TO_EDGE,
- MIRRORED_REPEAT,
- };
- WrapT wrapT;
- Sampler(const Json::Value &o)
- : wrapS(WrapS::REPEAT)
- , wrapT(WrapT::REPEAT)
- {
- check(o.isObject());
- if (o.isMember("magFilter")) {
- static std::unordered_map<Json::UInt64, MagFilter> map = {
- {9728, MagFilter::NEAREST},
- {9729, MagFilter::LINEAR},
- };
- const auto &v = o["magFilter"]; check(v.isUInt64());
- magFilter = map.at(v.asUInt64());
- }
- if (o.isMember("minFilter")) {
- static std::unordered_map<Json::UInt64, MinFilter> map = {
- {9728, MinFilter::NEAREST},
- {9729, MinFilter::LINEAR},
- {9984, MinFilter::NEAREST_MIPMAP_NEAREST},
- {9985, MinFilter::LINEAR_MIPMAP_NEAREST},
- {9986, MinFilter::NEAREST_MIPMAP_LINEAR},
- {9987, MinFilter::LINEAR_MIPMAP_LINEAR},
- };
- const auto &v = o["minFilter"]; check(v.isUInt64());
- minFilter = map.at(v.asUInt64());
- }
- if (o.isMember("name")) {
- name = as<std::string>(o["name"]);
- }
- if (o.isMember("wrapS")) {
- static std::unordered_map<Json::UInt64, WrapS> map = {
- {10497, WrapS::REPEAT},
- {33071, WrapS::CLAMP_TO_EDGE},
- {33648, WrapS::MIRRORED_REPEAT},
- };
- const auto &v = o["wrapS"]; check(v.isUInt64());
- wrapS = map.at(v.asUInt64());
- }
- if (o.isMember("wrapT")) {
- static std::unordered_map<Json::UInt64, WrapT> map = {
- {10497, WrapT::REPEAT},
- {33071, WrapT::CLAMP_TO_EDGE},
- {33648, WrapT::MIRRORED_REPEAT},
- };
- const auto &v = o["wrapT"]; check(v.isUInt64());
- wrapT = map.at(v.asUInt64());
- }
- }
- };
- template<> Sampler as(const Json::Value &o) { return o; }
- struct Scene {
- std::optional<std::string> name;
- std::optional<std::vector<std::size_t>> nodes;
- Scene(const Json::Value &o)
- {
- check(o.isObject());
- if (o.isMember("name")) {
- name = as<std::string>(o["name"]);
- }
- if (o.isMember("nodes")) {
- nodes = asVec<std::size_t>(o["nodes"]);
- check(nodes->size() >= 1);
- checkDuplicateFree(*nodes);
- }
- }
- };
- template<> Scene as(const Json::Value &o) { return o; }
- struct Skin {
- std::optional<std::size_t> inverseBindMatrices;
- std::vector<std::size_t> joints;
- std::optional<std::string> name;
- std::optional<std::size_t> skeleton;
- Skin(const Json::Value &o)
- : joints(asVec<std::size_t>(o["joints"]))
- {
- check(o.isObject());
- if (o.isMember("inverseBindMatrices")) {
- inverseBindMatrices = as<std::size_t>(o["inverseBindMatrices"]);
- }
- check(joints.size() >= 1);
- checkDuplicateFree(joints);
- if (o.isMember("name")) {
- name = as<std::string>(o["name"]);
- }
- if (o.isMember("skeleton")) {
- skeleton = as<std::size_t>(o["skeleton"]);
- }
- }
- };
- template<> Skin as(const Json::Value &o) { return o; }
- struct Texture {
- std::optional<std::string> name;
- std::optional<std::size_t> sampler;
- std::optional<std::size_t> source;
- Texture(const Json::Value &o)
- {
- check(o.isObject());
- if (o.isMember("name")) {
- name = as<std::string>(o["name"]);
- }
- if (o.isMember("sampler")) {
- sampler = as<std::size_t>(o["sampler"]);
- }
- if (o.isMember("source")) {
- source = as<std::size_t>(o["source"]);
- }
- }
- };
- template<> Texture as(const Json::Value &o) { return o; }
- struct GlTF {
- std::optional<std::vector<Accessor>> accessors;
- std::optional<std::vector<Animation>> animations;
- Asset asset;
- std::optional<std::vector<BufferView>> bufferViews;
- std::optional<std::vector<Buffer>> buffers;
- std::optional<std::vector<Camera>> cameras;
- std::optional<std::vector<std::string>> extensionsRequired;
- std::optional<std::vector<std::string>> extensionsUsed;
- std::optional<std::vector<Image>> images;
- std::optional<std::vector<Material>> materials;
- std::optional<std::vector<Mesh>> meshes;
- std::optional<std::vector<Node>> nodes;
- std::optional<std::vector<Sampler>> samplers;
- std::optional<std::size_t> scene;
- std::optional<std::vector<Scene>> scenes;
- std::optional<std::vector<Skin>> skins;
- std::optional<std::vector<Texture>> textures;
- static std::string uriError(const std::string &uri) {
- // only base64 data URI support by default
- throw std::runtime_error("unsupported URI: " + uri);
- }
- GlTF(const Json::Value &o,
- const std::function<std::string(const std::string &uri)> &resolveURI = uriError)
- : asset(as<Asset>(o["asset"]))
- {
- check(o.isObject());
- if (o.isMember("accessors")) {
- accessors = asVec<Accessor>(o["accessors"]);
- check(accessors->size() >= 1);
- }
- if (o.isMember("animations")) {
- animations = asVec<Animation>(o["animations"]);
- check(animations->size() >= 1);
- }
- if (o.isMember("bufferViews")) {
- bufferViews = asVec<BufferView>(o["bufferViews"]);
- check(bufferViews->size() >= 1);
- }
- if (o.isMember("buffers")) {
- auto b = o["buffers"];
- check(b.isArray());
- std::vector<Buffer> bufs;
- bufs.reserve(b.size());
- for (Json::ArrayIndex i = 0; i < b.size(); ++i) {
- bufs.emplace_back(b[i], resolveURI);
- }
- check(bufs.size() >= 1);
- buffers = std::move(bufs);
- }
- if (o.isMember("cameras")) {
- cameras = asVec<Camera>(o["cameras"]);
- check(cameras->size() >= 1);
- }
- if (o.isMember("extensionsRequired")) {
- extensionsRequired = asVec<std::string>(o["extensionsRequired"]);
- check(extensionsRequired->size() >= 1);
- checkDuplicateFree(*extensionsRequired);
- }
- if (o.isMember("extensionsUsed")) {
- extensionsUsed = asVec<std::string>(o["extensionsUsed"]);
- check(extensionsUsed->size() >= 1);
- checkDuplicateFree(*extensionsUsed);
- }
- if (o.isMember("images")) {
- images = asVec<Image>(o["images"]);
- check(images->size() >= 1);
- }
- if (o.isMember("materials")) {
- materials = asVec<Material>(o["materials"]);
- check(materials->size() >= 1);
- }
- if (o.isMember("meshes")) {
- meshes = asVec<Mesh>(o["meshes"]);
- check(meshes->size() >= 1);
- }
- if (o.isMember("nodes")) {
- nodes = asVec<Node>(o["nodes"]);
- check(nodes->size() >= 1);
- // Nodes must be a forest:
- // 1. Each node should have indegree 0 or 1:
- std::vector<std::size_t> indeg(nodes->size());
- for (std::size_t i = 0; i < nodes->size(); ++i) {
- auto children = nodes->at(i).children;
- if (!children.has_value()) continue;
- for (auto child : children.value()) {
- ++indeg.at(child);
- }
- }
- for (const auto deg : indeg) {
- check(deg <= 1);
- }
- // 2. There should be no cycles:
- std::vector<bool> visited(nodes->size());
- std::stack<std::size_t, std::vector<std::size_t>> toVisit;
- for (std::size_t i = 0; i < nodes->size(); ++i) {
- // Only start DFS in roots.
- if (indeg[i] > 0)
- continue;
- toVisit.push(i);
- do {
- std::size_t j = toVisit.top();
- check(!visited.at(j));
- visited[j] = true;
- toVisit.pop();
- auto children = nodes->at(j).children;
- if (!children.has_value())
- continue;
- for (auto child : *children) {
- toVisit.push(child);
- }
- } while (!toVisit.empty());
- }
- }
- if (o.isMember("samplers")) {
- samplers = asVec<Sampler>(o["samplers"]);
- check(samplers->size() >= 1);
- }
- if (o.isMember("scene")) {
- scene = as<std::size_t>(o["scene"]);
- }
- if (o.isMember("scenes")) {
- scenes = asVec<Scene>(o["scenes"]);
- check(scenes->size() >= 1);
- }
- if (o.isMember("skins")) {
- skins = asVec<Skin>(o["skins"]);
- check(skins->size() >= 1);
- }
- if (o.isMember("textures")) {
- textures = asVec<Texture>(o["textures"]);
- check(textures->size() >= 1);
- }
- // Validation
- checkForall(bufferViews, [&](const BufferView &view) {
- check(buffers.has_value());
- const Buffer &buf = buffers->at(view.buffer);
- // Be careful because of possible integer overflows.
- check(view.byteOffset < buf.byteLength);
- check(view.byteLength <= buf.byteLength);
- check(view.byteOffset <= buf.byteLength - view.byteLength);
- });
- const auto checkAccessor = [&](const auto &accessor,
- std::size_t bufferView, std::size_t byteOffset, std::size_t count) {
- const BufferView &view = bufferViews->at(bufferView);
- if (view.byteStride.has_value())
- check(*view.byteStride % accessor.componentSize() == 0);
- check(byteOffset < view.byteLength);
- // Use division to avoid overflows.
- const auto effective_byte_stride = view.byteStride.value_or(accessor.elementSize());
- check(count <= (view.byteLength - byteOffset) / effective_byte_stride);
- };
- checkForall(accessors, [&](const Accessor &accessor) {
- if (accessor.bufferView.has_value())
- checkAccessor(accessor, *accessor.bufferView, accessor.byteOffset, accessor.count);
- if (accessor.sparse.has_value()) {
- const auto &indices = accessor.sparse->indices;
- checkAccessor(indices, indices.bufferView, indices.byteOffset, accessor.sparse->count);
- const auto &values = accessor.sparse->values;
- checkAccessor(accessor, values.bufferView, values.byteOffset, accessor.sparse->count);
- }
- });
- checkForall(images, [&](const Image &image) {
- checkIndex(bufferViews, image.bufferView);
- });
- checkForall(meshes, [&](const Mesh &mesh) {
- for (const auto &primitive : mesh.primitives) {
- checkIndex(accessors, primitive.indices);
- checkIndex(materials, primitive.material);
- checkIndex(accessors, primitive.attributes.normal);
- checkIndex(accessors, primitive.attributes.position);
- checkIndex(accessors, primitive.attributes.tangent);
- checkForall(primitive.attributes.texcoord, [&](const std::size_t &i) {
- checkIndex(accessors, i);
- });
- checkForall(primitive.attributes.color, [&](const std::size_t &i) {
- checkIndex(accessors, i);
- });
- checkForall(primitive.attributes.joints, [&](const std::size_t &i) {
- checkIndex(accessors, i);
- });
- checkForall(primitive.attributes.weights, [&](const std::size_t &i) {
- checkIndex(accessors, i);
- });
- if (primitive.material.has_value()) {
- const Material &material = materials->at(primitive.material.value());
- if (material.emissiveTexture.has_value()) {
- check(primitive.attributes.texcoord.has_value());
- check(material.emissiveTexture->texCoord < primitive.attributes.texcoord->size());
- }
- if (material.normalTexture.has_value()) {
- check(primitive.attributes.texcoord.has_value());
- check(material.normalTexture->texCoord < primitive.attributes.texcoord->size());
- }
- if (material.occlusionTexture.has_value()) {
- check(primitive.attributes.texcoord.has_value());
- check(material.occlusionTexture->texCoord < primitive.attributes.texcoord->size());
- }
- }
- checkForall(primitive.targets, [&](const MeshPrimitive::MorphTargets &target) {
- checkIndex(accessors, target.normal);
- checkIndex(accessors, target.position);
- checkIndex(accessors, target.tangent);
- checkForall(target.texcoord, [&](const std::size_t &i) {
- checkIndex(accessors, i);
- });
- checkForall(target.color, [&](const std::size_t &i) {
- checkIndex(accessors, i);
- });
- });
- }
- });
- checkForall(nodes, [&](const Node &node) {
- checkIndex(cameras, node.camera);
- checkIndex(meshes, node.mesh);
- checkIndex(skins, node.skin);
- });
- checkForall(scenes, [&](const Scene &scene) {
- checkForall(scene.nodes, [&](const size_t &i) {
- checkIndex(nodes, i);
- });
- });
- checkForall(skins, [&](const Skin &skin) {
- checkIndex(accessors, skin.inverseBindMatrices);
- for (const std::size_t &i : skin.joints)
- checkIndex(nodes, i);
- checkIndex(nodes, skin.skeleton);
- });
- checkForall(textures, [&](const Texture &texture) {
- checkIndex(samplers, texture.sampler);
- checkIndex(images, texture.source);
- });
- checkForall(animations, [&](const Animation &animation) {
- for (const auto &sampler : animation.samplers) {
- checkIndex(accessors, sampler.input);
- const auto &accessor = accessors->at(sampler.input);
- check(accessor.type == Accessor::Type::SCALAR);
- check(accessor.componentType == Accessor::ComponentType::FLOAT);
- checkIndex(accessors, sampler.output);
- }
- for (const auto &channel : animation.channels) {
- checkIndex(nodes, channel.target.node);
- checkIndex(animation.samplers, channel.sampler);
- }
- });
- checkIndex(scenes, scene);
- }
- };
- }
|