blob: 08592d705e88dca6145aeb34e3f1741bf58ff5a0 [file] [log] [blame] [edit]
#include <pybind11/pybind11.h>
#include <pybind11/stl.h>
#include <pybind11/numpy.h>
#include <cstring>
// Use double precision for better python integration.
#define TINYOBJLOADER_USE_DOUBLE
// define some helper functions for pybind11
#define TINY_OBJ_LOADER_PYTHON_BINDING
#include "tiny_obj_loader.h"
namespace py = pybind11;
using namespace tinyobj;
PYBIND11_MODULE(tinyobjloader, tobj_module)
{
tobj_module.doc() = "Python bindings for TinyObjLoader.";
// register struct
py::class_<ObjReaderConfig>(tobj_module, "ObjReaderConfig")
.def(py::init<>())
.def_readwrite("triangulate", &ObjReaderConfig::triangulate);
// py::init<>() for default constructor
py::class_<ObjReader>(tobj_module, "ObjReader")
.def(py::init<>())
.def("ParseFromFile", &ObjReader::ParseFromFile, py::arg("filename"), py::arg("option") = ObjReaderConfig())
.def("ParseFromString", &ObjReader::ParseFromString, py::arg("obj_text"), py::arg("mtl_text"), py::arg("option") = ObjReaderConfig())
.def("Valid", &ObjReader::Valid)
.def("GetAttrib", &ObjReader::GetAttrib)
.def("GetShapes", &ObjReader::GetShapes)
.def("GetMaterials", &ObjReader::GetMaterials)
.def("Warning", &ObjReader::Warning)
.def("Error", &ObjReader::Error);
py::class_<attrib_t>(tobj_module, "attrib_t")
.def(py::init<>())
.def_readonly("vertices", &attrib_t::vertices)
.def("numpy_vertices", [] (attrib_t &instance) {
auto ret = py::array_t<real_t>(instance.vertices.size());
py::buffer_info buf = ret.request();
memcpy(buf.ptr, instance.vertices.data(), instance.vertices.size() * sizeof(real_t));
return ret;
})
.def_readonly("normals", &attrib_t::normals)
.def_readonly("texcoords", &attrib_t::texcoords)
.def_readonly("colors", &attrib_t::colors)
;
py::class_<shape_t>(tobj_module, "shape_t")
.def(py::init<>())
.def_readwrite("name", &shape_t::name)
.def_readwrite("mesh", &shape_t::mesh)
.def_readwrite("lines", &shape_t::lines)
.def_readwrite("points", &shape_t::points);
py::class_<index_t>(tobj_module, "index_t")
.def(py::init<>())
.def_readwrite("vertex_index", &index_t::vertex_index)
.def_readwrite("normal_index", &index_t::normal_index)
.def_readwrite("texcoord_index", &index_t::texcoord_index)
;
// NOTE(syoyo): It looks it is rather difficult to expose assignment by array index to
// python world for array variable.
// For example following python scripting does not work well.
//
// print(mat.diffuse)
// >>> [0.1, 0.2, 0.3]
// mat.diffuse[1] = 1.0
// print(mat.diffuse)
// >>> [0.1, 0.2, 0.3] # No modification
//
// https://github.com/pybind/pybind11/issues/1134
//
// so, we need to update array variable like this:
//
// diffuse = mat.diffuse
// diffuse[1] = 1.0
// mat.diffuse = diffuse
//
py::class_<material_t>(tobj_module, "material_t")
.def(py::init<>())
.def_readwrite("name", &material_t::name)
.def_property("ambient", &material_t::GetAmbient, &material_t::SetAmbient)
.def_property("diffuse", &material_t::GetDiffuse, &material_t::SetDiffuse)
.def_property("specular", &material_t::GetSpecular, &material_t::SetSpecular)
.def_property("transmittance", &material_t::GetTransmittance, &material_t::SetTransmittance)
.def_readwrite("shininess", &material_t::shininess)
.def_readwrite("ior", &material_t::ior)
.def_readwrite("dissolve", &material_t::dissolve)
.def_readwrite("illum", &material_t::illum)
.def_readwrite("ambient_texname", &material_t::ambient_texname)
.def_readwrite("diffuse_texname", &material_t::diffuse_texname)
.def_readwrite("specular_texname", &material_t::specular_texname)
.def_readwrite("specular_highlight_texname", &material_t::specular_highlight_texname)
.def_readwrite("bump_texname", &material_t::bump_texname)
.def_readwrite("displacement_texname", &material_t::displacement_texname)
.def_readwrite("alpha_texname", &material_t::alpha_texname)
.def_readwrite("reflection_texname", &material_t::reflection_texname)
// TODO(syoyo): Expose texture parameter
// PBR
.def_readwrite("roughness", &material_t::roughness)
.def_readwrite("metallic", &material_t::metallic)
.def_readwrite("sheen", &material_t::sheen)
.def_readwrite("clearcoat_thickness", &material_t::clearcoat_thickness)
.def_readwrite("clearcoat_roughness", &material_t::clearcoat_roughness)
.def_readwrite("anisotropy", &material_t::anisotropy)
.def_readwrite("anisotropy_rotation", &material_t::anisotropy_rotation)
.def_readwrite("roughness_texname", &material_t::roughness_texname)
.def_readwrite("metallic_texname", &material_t::metallic_texname)
.def_readwrite("sheen_texname", &material_t::sheen_texname)
.def_readwrite("emissive_texname", &material_t::emissive_texname)
.def_readwrite("normal_texname", &material_t::normal_texname)
.def("GetCustomParameter", &material_t::GetCustomParameter)
;
py::class_<mesh_t>(tobj_module, "mesh_t")
.def(py::init<>())
.def_readonly("num_face_vertices", &mesh_t::num_face_vertices)
.def("numpy_num_face_vertices", [] (mesh_t &instance) {
auto ret = py::array_t<unsigned char>(instance.num_face_vertices.size());
py::buffer_info buf = ret.request();
memcpy(buf.ptr, instance.num_face_vertices.data(), instance.num_face_vertices.size() * sizeof(unsigned char));
return ret;
})
.def_readonly("indices", &mesh_t::indices)
.def("numpy_indices", [] (mesh_t &instance) {
// Flatten indexes. index_t is composed of 3 ints(vertex_index, normal_index, texcoord_index).
// numpy_indices = [0, -1, -1, 1, -1, -1, ...]
// C++11 or later should pack POD struct tightly and does not reorder variables,
// so we can memcpy to copy data.
// Still, we check the size of struct and byte offsets of each variable just for sure.
static_assert(sizeof(index_t) == 12, "sizeof(index_t) must be 12");
static_assert(offsetof(index_t, vertex_index) == 0, "offsetof(index_t, vertex_index) must be 0");
static_assert(offsetof(index_t, normal_index) == 4, "offsetof(index_t, normal_index) must be 4");
static_assert(offsetof(index_t, texcoord_index) == 8, "offsetof(index_t, texcoord_index) must be 8");
auto ret = py::array_t<int>(instance.indices.size() * 3);
py::buffer_info buf = ret.request();
memcpy(buf.ptr, instance.indices.data(), instance.indices.size() * 3 * sizeof(int));
return ret;
})
.def_readonly("material_ids", &mesh_t::material_ids)
.def("numpy_material_ids", [] (mesh_t &instance) {
auto ret = py::array_t<int>(instance.material_ids.size());
py::buffer_info buf = ret.request();
memcpy(buf.ptr, instance.material_ids.data(), instance.material_ids.size() * sizeof(int));
return ret;
});
py::class_<lines_t>(tobj_module, "lines_t")
.def(py::init<>());
py::class_<points_t>(tobj_module, "points_t")
.def(py::init<>());
}