#include "../libslic3r.h" #include "../Model.hpp" #include "../TriangleMesh.hpp" #include "OBJ.hpp" #include "objparser.hpp" #include #include #ifdef _WIN32 #define DIR_SEPARATOR '\\' #else #define DIR_SEPARATOR '/' #endif //Translation #include "I18N.hpp" #define _L(s) Slic3r::I18N::translate(s) namespace Slic3r { bool load_obj(const char *path, TriangleMesh *meshptr, ObjInfo& obj_info, std::string &message) { if (meshptr == nullptr) return false; // Parse the OBJ file. ObjParser::ObjData data; ObjParser::MtlData mtl_data; if (! ObjParser::objparse(path, data)) { BOOST_LOG_TRIVIAL(error) << "load_obj: failed to parse " << path; message = _L("load_obj: failed to parse"); return false; } bool exist_mtl = false; if (data.mtllibs.size() > 0) { // read mtl for (auto mtl_name : data.mtllibs) { if (mtl_name.size() == 0){ continue; } exist_mtl = true; bool mtl_name_is_path = false; boost::filesystem::path mtl_abs_path(mtl_name); if (boost::filesystem::exists(mtl_abs_path)) { mtl_name_is_path = true; } boost::filesystem::path mtl_path; if (!mtl_name_is_path) { boost::filesystem::path full_path(path); std::string dir = full_path.parent_path().string(); auto mtl_file = dir + "/" + mtl_name; boost::filesystem::path temp_mtl_path(mtl_file); mtl_path = temp_mtl_path; } auto _mtl_path = mtl_name_is_path ? mtl_abs_path.string().c_str() : mtl_path.string().c_str(); if (boost::filesystem::exists(mtl_name_is_path ? mtl_abs_path : mtl_path)) { if (!ObjParser::mtlparse(_mtl_path, mtl_data)) { BOOST_LOG_TRIVIAL(error) << "load_obj:load_mtl: failed to parse " << _mtl_path; message = _L("load mtl in obj: failed to parse"); return false; } } else { BOOST_LOG_TRIVIAL(error) << "load_obj: failed to load mtl_path:" << _mtl_path; } } } // Count the faces and verify, that all faces are triangular. size_t num_faces = 0; size_t num_quads = 0; for (size_t i = 0; i < data.vertices.size(); ++ i) { // Find the end of face. size_t j = i; for (; j < data.vertices.size() && data.vertices[j].coordIdx != -1; ++ j) ; if (size_t num_face_vertices = j - i; num_face_vertices > 0) { if (num_face_vertices > 4) { // Non-triangular and non-quad faces are not supported as of now. BOOST_LOG_TRIVIAL(error) << "load_obj: failed to parse " << path << ". The file contains polygons with more than 4 vertices."; message = _L("The file contains polygons with more than 4 vertices."); return false; } else if (num_face_vertices < 3) { // Non-triangular and non-quad faces are not supported as of now. BOOST_LOG_TRIVIAL(error) << "load_obj: failed to parse " << path << ". The file contains polygons with less than 2 vertices."; message = _L("The file contains polygons with less than 2 vertices."); return false; } if (num_face_vertices == 4) ++ num_quads; ++ num_faces; i = j; } } // Convert ObjData into indexed triangle set. indexed_triangle_set its; size_t num_vertices = data.coordinates.size() / OBJ_VERTEX_LENGTH; its.vertices.reserve(num_vertices); its.indices.reserve(num_faces + num_quads); if (exist_mtl) { obj_info.is_single_mtl = data.usemtls.size() == 1 && mtl_data.new_mtl_unmap.size() == 1; obj_info.face_colors.reserve(num_faces + num_quads); } bool has_color = data.has_vertex_color; for (size_t i = 0; i < num_vertices; ++ i) { size_t j = i * OBJ_VERTEX_LENGTH; its.vertices.emplace_back(data.coordinates[j], data.coordinates[j + 1], data.coordinates[j + 2]); if (data.has_vertex_color) { RGBA color{std::clamp(data.coordinates[j + 3], 0.f, 1.f), std::clamp(data.coordinates[j + 4], 0.f, 1.f), std::clamp(data.coordinates[j + 5], 0.f, 1.f), std::clamp(data.coordinates[j + 6], 0.f, 1.f)}; obj_info.vertex_colors.emplace_back(color); } } int indices[ONE_FACE_SIZE]; int uvs[ONE_FACE_SIZE]; for (size_t i = 0; i < data.vertices.size();) if (data.vertices[i].coordIdx == -1) ++ i; else { int cnt = 0; while (i < data.vertices.size()) if (const ObjParser::ObjVertex &vertex = data.vertices[i ++]; vertex.coordIdx == -1) { break; } else { assert(cnt < OBJ_VERTEX_LENGTH); if (vertex.coordIdx < 0 || vertex.coordIdx >= int(its.vertices.size())) { BOOST_LOG_TRIVIAL(error) << "load_obj: failed to parse " << path << ". The file contains invalid vertex index."; message = _L("The file contains invalid vertex index."); return false; } indices[cnt] = vertex.coordIdx; uvs[cnt] = vertex.textureCoordIdx; cnt++; } if (cnt) { assert(cnt == 3 || cnt == 4); // Insert one or two faces (triangulate a quad). its.indices.emplace_back(indices[0], indices[1], indices[2]); int face_index =its.indices.size() - 1; RGBA face_color; auto set_face_color = [&uvs, &data, &mtl_data, &obj_info, &face_color](int face_index, const std::string mtl_name) { if (mtl_data.new_mtl_unmap.find(mtl_name) != mtl_data.new_mtl_unmap.end()) { bool is_merge_ka_kd = true; for (size_t n = 0; n < 3; n++) { if (float(mtl_data.new_mtl_unmap[mtl_name]->Ka[n] + mtl_data.new_mtl_unmap[mtl_name]->Kd[n]) > 1.0) { is_merge_ka_kd=false; break; } } for (size_t n = 0; n < 3; n++) { if (is_merge_ka_kd) { face_color[n] = std::clamp(float(mtl_data.new_mtl_unmap[mtl_name]->Ka[n] + mtl_data.new_mtl_unmap[mtl_name]->Kd[n]), 0.f, 1.f); } else { face_color[n] = std::clamp(float(mtl_data.new_mtl_unmap[mtl_name]->Kd[n]), 0.f, 1.f); } } face_color[3] = mtl_data.new_mtl_unmap[mtl_name]->Tr; // alpha if (mtl_data.new_mtl_unmap[mtl_name]->map_Kd.size() > 0) { auto png_name = mtl_data.new_mtl_unmap[mtl_name]->map_Kd; obj_info.has_uv_png = true; if (obj_info.pngs.find(png_name) == obj_info.pngs.end()) { obj_info.pngs[png_name] = false; } obj_info.uv_map_pngs[face_index] = png_name; } if (data.textureCoordinates.size() > 0) { Vec2f uv0(data.textureCoordinates[uvs[0] * 2], data.textureCoordinates[uvs[0] * 2 + 1]); Vec2f uv1(data.textureCoordinates[uvs[1] * 2], data.textureCoordinates[uvs[1] * 2 + 1]); Vec2f uv2(data.textureCoordinates[uvs[2] * 2], data.textureCoordinates[uvs[2] * 2 + 1]); std::array uv_array{uv0, uv1, uv2}; obj_info.uvs.emplace_back(uv_array); } obj_info.face_colors.emplace_back(face_color); } }; auto set_face_color_by_mtl = [&data, &set_face_color](int face_index) { if (data.usemtls.size() == 1) { set_face_color(face_index, data.usemtls[0].name); } else { for (size_t k = 0; k < data.usemtls.size(); k++) { auto mtl = data.usemtls[k]; if (face_index >= mtl.face_start && face_index <= mtl.face_end) { set_face_color(face_index, data.usemtls[k].name); break; } } } }; if (exist_mtl) { set_face_color_by_mtl(face_index); } if (cnt == 4) { its.indices.emplace_back(indices[0], indices[2], indices[3]); int face_index = its.indices.size() - 1; if (exist_mtl) { set_face_color_by_mtl(face_index); } } } } *meshptr = TriangleMesh(std::move(its)); if (meshptr->empty()) { BOOST_LOG_TRIVIAL(error) << "load_obj: This OBJ file couldn't be read because it's empty. " << path; message = _L("This OBJ file couldn't be read because it's empty."); return false; } if (meshptr->volume() < 0) meshptr->flip_triangles(); return true; } bool load_obj(const char *path, Model *model, ObjInfo& obj_info, std::string &message, const char *object_name_in) { TriangleMesh mesh; bool ret = load_obj(path, &mesh, obj_info, message); if (ret) { std::string object_name; if (object_name_in == nullptr) { const char *last_slash = strrchr(path, DIR_SEPARATOR); object_name.assign((last_slash == nullptr) ? path : last_slash + 1); } else object_name.assign(object_name_in); model->add_object(object_name.c_str(), path, std::move(mesh)); } return ret; } bool store_obj(const char *path, TriangleMesh *mesh) { //FIXME returning false even if write failed. mesh->WriteOBJFile(path); return true; } bool store_obj(const char *path, ModelObject *model_object) { TriangleMesh mesh = model_object->mesh(); return store_obj(path, &mesh); } bool store_obj(const char *path, Model *model) { TriangleMesh mesh = model->mesh(); return store_obj(path, &mesh); } }; // namespace Slic3r