#define VC_ENABLE_REFLECTION #include "ContentUnitLoader.hpp" #include "../ContentBuilder.hpp" #include "coders/json.hpp" #include "core_defs.hpp" #include "data/dv.hpp" #include "data/StructLayout.hpp" #include "debug/Logger.hpp" #include "io/io.hpp" #include "presets/ParticlesPreset.hpp" #include "util/stringutil.hpp" #include "voxels/Block.hpp" using namespace data; static debug::Logger logger("block-content-loader"); static void perform_user_block_fields( const std::string& blockName, StructLayout& layout ) { if (layout.size() > MAX_USER_BLOCK_FIELDS_SIZE) { throw std::runtime_error( util::quote(blockName) + " fields total size exceeds limit (" + std::to_string(layout.size()) + "/" + std::to_string(MAX_USER_BLOCK_FIELDS_SIZE) + ")"); } for (const auto& field : layout) { if (field.name.at(0) == '.') { throw std::runtime_error( util::quote(blockName) + " field " + field.name + ": user field may not start with '.'"); } } std::vector fields; fields.insert(fields.end(), layout.begin(), layout.end()); // add built-in fields here layout = StructLayout::create(fields); } template<> void ContentUnitLoader::loadUnit( Block& def, const std::string& name, const io::path& file ) { auto root = io::read_json(file); if (def.properties == nullptr) { def.properties = dv::object(); def.properties["name"] = name; } for (auto& [key, value] : root.asObject()) { auto pos = key.rfind('@'); if (pos == std::string::npos) { def.properties[key] = value; continue; } auto field = key.substr(0, pos); auto suffix = key.substr(pos + 1); process_method(def.properties, suffix, field, value); } if (root.has("parent")) { const auto& parentName = root["parent"].asString(); auto parentDef = builder.get(parentName); if (parentDef == nullptr) { throw std::runtime_error( "Failed to find parent(" + parentName + ") for " + name ); } parentDef->cloneTo(def); } root.at("caption").get(def.caption); // block texturing if (root.has("texture")) { const auto& texture = root["texture"].asString(); for (uint i = 0; i < 6; i++) { def.textureFaces[i] = texture; } } else if (root.has("texture-faces")) { const auto& texarr = root["texture-faces"]; for (uint i = 0; i < 6; i++) { def.textureFaces[i] = texarr[i].asString(); } } // block model std::string modelTypeName = BlockModelMeta.getNameString(def.model); root.at("model").get(modelTypeName); root.at("model-name").get(def.modelName); if (BlockModelMeta.getItem(modelTypeName, def.model)) { if (def.model == BlockModel::custom && def.customModelRaw == nullptr) { if (root.has("model-primitives")) { def.customModelRaw = root["model-primitives"]; } else if (def.modelName.empty()) { throw std::runtime_error(name + ": no 'model-primitives' or 'model-name' found"); } } } else if (!modelTypeName.empty()) { logger.error() << "unknown model: " << modelTypeName; def.model = BlockModel::none; } std::string cullingModeName = CullingModeMeta.getNameString(def.culling); root.at("culling").get(cullingModeName); if (!CullingModeMeta.getItem(cullingModeName, def.culling)) { logger.error() << "unknown culling mode: " << cullingModeName; } root.at("material").get(def.material); // rotation profile std::string profile = def.rotations.name; root.at("rotation").get(profile); def.rotatable = profile != "none"; if (profile == BlockRotProfile::PIPE_NAME) { def.rotations = BlockRotProfile::PIPE; } else if (profile == BlockRotProfile::PANE_NAME) { def.rotations = BlockRotProfile::PANE; } else if (profile != "none") { logger.error() << "unknown rotation profile " << profile; def.rotatable = false; } // block hitbox AABB [x, y, z, width, height, depth] if (auto found = root.at("hitboxes")) { const auto& boxarr = *found; def.hitboxes.resize(boxarr.size()); for (uint i = 0; i < boxarr.size(); i++) { const auto& box = boxarr[i]; auto& hitboxesIndex = def.hitboxes[i]; hitboxesIndex.a = glm::vec3( box[0].asNumber(), box[1].asNumber(), box[2].asNumber() ); hitboxesIndex.b = glm::vec3( box[3].asNumber(), box[4].asNumber(), box[5].asNumber() ); hitboxesIndex.b += hitboxesIndex.a; } } else if (auto found = root.at("hitbox")) { const auto& box = *found; AABB aabb; aabb.a = glm::vec3( box[0].asNumber(), box[1].asNumber(), box[2].asNumber() ); aabb.b = glm::vec3( box[3].asNumber(), box[4].asNumber(), box[5].asNumber() ); aabb.b += aabb.a; def.hitboxes = {aabb}; } // block light emission [r, g, b] where r,g,b in range [0..15] if (auto found = root.at("emission")) { const auto& emissionarr = *found; for (size_t i = 0; i < 3; i++) { def.emission[i] = std::clamp(emissionarr[i].asInteger(), static_cast(0), static_cast(15)); } } // block size if (auto found = root.at("size")) { const auto& sizearr = *found; def.size.x = sizearr[0].asInteger(); def.size.y = sizearr[1].asInteger(); def.size.z = sizearr[2].asInteger(); if (def.size.x < 1 || def.size.y < 1 || def.size.z < 1) { throw std::runtime_error( "block " + util::quote(def.name) + ": invalid block size" ); } if (def.model == BlockModel::block && (def.size.x != 1 || def.size.y != 1 || def.size.z != 1)) { def.model = BlockModel::aabb; def.hitboxes = {AABB(def.size)}; } } // primitive properties root.at("obstacle").get(def.obstacle); root.at("replaceable").get(def.replaceable); root.at("light-passing").get(def.lightPassing); root.at("sky-light-passing").get(def.skyLightPassing); root.at("shadeless").get(def.shadeless); root.at("ambient-occlusion").get(def.ambientOcclusion); root.at("breakable").get(def.breakable); root.at("selectable").get(def.selectable); root.at("grounded").get(def.grounded); root.at("hidden").get(def.hidden); root.at("draw-group").get(def.drawGroup); root.at("picking-item").get(def.pickingItem); root.at("surface-replacement").get(def.surfaceReplacement); root.at("script-name").get(def.scriptName); root.at("ui-layout").get(def.uiLayout); root.at("inventory-size").get(def.inventorySize); root.at("tick-interval").get(def.tickInterval); root.at("overlay-texture").get(def.overlayTexture); root.at("translucent").get(def.translucent); if (root.has("fields")) { def.dataStruct = std::make_unique(); def.dataStruct->deserialize(root["fields"]); perform_user_block_fields(def.name, *def.dataStruct); } if (root.has("particles")) { def.particles = std::make_unique(); def.particles->deserialize(root["particles"]); } if (def.tickInterval == 0) { def.tickInterval = 1; } if (def.hidden && def.pickingItem == def.name + BLOCK_ITEM_SUFFIX) { def.pickingItem = CORE_EMPTY; } def.scriptFile = pack.id + ":scripts/" + def.scriptName + ".lua"; }