6 #include "../core/log.h" 7 #include "../core/profiler.h" 18 MeshData &MeshData::colorize(
float r,
float g,
float b,
float a) {
19 return colorize(Eigen::Vector4f(r, g, b, a));
22 MeshData &MeshData::operator+=(
const MeshData &other) {
return append(other); }
24 MeshData &MeshData::translate(
float x,
float y,
float z) {
25 return translate(Eigen::Vector3f(x, y, z));
28 MeshData &MeshData::scale(
const Eigen::Vector3f &v) {
29 return transform(Eigen::Scaling(v));
32 MeshData &MeshData::scale(
float x,
float y,
float z) {
33 return scale(Eigen::Vector3f(x, y, z));
36 MeshData &MeshData::scale(
float s) {
return scale(s, s, s); }
38 MeshData &MeshData::rotate(
float angle,
const Eigen::Vector3f &axis) {
39 return transform(Eigen::AngleAxisf(angle, axis));
42 void MeshData::_transform(
const Eigen::Affine3f &transform) {
43 for (
auto &p : positions) {
46 Eigen::Matrix3f normal_matrix = transform.linear().inverse().transpose();
47 for (
auto *component : {&normals, &tangents, &bitangents}) {
48 for (
auto &n : *component) {
49 n = (normal_matrix * n).eval();
55 static void appendMeshComponent(
const MeshData &mesh, std::vector<T> &a,
56 const std::vector<T> &b,
const T &def) {
58 if (a.size() < mesh.positions.size()) {
59 a.resize(mesh.positions.size(), def);
61 a.insert(a.end(), b.begin(), b.end());
66 for (
auto i : other.indices) {
67 indices.push_back(i + positions.size());
69 appendMeshComponent(*
this, normals, other.normals, Eigen::Vector3f(0, 0, 0));
70 appendMeshComponent(*
this, texcoords, other.texcoords, Eigen::Vector2f(0, 0));
71 appendMeshComponent(*
this, tangents, other.tangents,
72 Eigen::Vector3f(0, 0, 0));
73 appendMeshComponent(*
this, bitangents, other.bitangents,
74 Eigen::Vector3f(0, 0, 0));
75 appendMeshComponent(*
this, colors, other.colors, Eigen::Vector4f(0, 0, 0, 1));
76 appendMeshComponent(*
this, colors8, other.colors8, uint32_t(0xff000000));
77 appendMeshComponent(*
this, extras, other.extras, Eigen::Vector4f(0, 0, 0, 1));
78 appendMeshComponent(*
this, positions, other.positions,
79 Eigen::Vector3f(0, 0, 0));
83 MeshData &MeshData::translate(
const Eigen::Vector3f &v) {
84 for (
auto &p : positions) {
90 MeshData &MeshData::colorize(
const Eigen::Vector4f &color) {
91 colors.resize(positions.size());
92 for (
auto &c : colors) {
98 MeshData &MeshData::computeNormals() {
99 if (positions.empty()) {
102 if (indices.empty() && !positions.empty()) {
103 while (indices.size() < positions.size()) {
104 indices.push_back(indices.size());
108 normals.resize(positions.size(), Eigen::Vector3f::Zero());
109 for (
size_t face_index = 0; face_index < indices.size() / 3; face_index++) {
110 auto &A = positions.at(indices.at(face_index * 3 + 0));
111 auto &B = positions.at(indices.at(face_index * 3 + 1));
112 auto &C = positions.at(indices.at(face_index * 3 + 2));
113 Eigen::Vector3f face_normal = (B - A).cross(C - A);
114 for (
size_t edge_index = 0; edge_index < 3; edge_index++) {
115 size_t index = face_index * 3 + edge_index;
116 normals.at(indices.at(index)) += face_normal;
119 for (
auto &n : normals) {
125 Mesh::Mesh(
const MeshData &data) : _data(data) { init(); }
127 Mesh::Mesh(
const std::function<
void(
MeshData &)> &loader) : _loader(loader) {}
129 Mesh::Mesh(
const std::function<
MeshData()> &loader)
130 : _loader([loader](
MeshData &d) { d = loader(); }) {}
134 PROFILER(
"check mesh transparency");
135 _transparent =
false;
136 if (!_data.colors.empty()) {
137 auto *alpha_begin = &(_data.colors[0].w());
138 auto *alpha_end = alpha_begin + _data.colors.size() * 4;
139 for (
auto *alpha_pointer = alpha_begin; alpha_pointer < alpha_end;
140 alpha_pointer += 4) {
141 if ((*alpha_pointer) < 1.0) {
150 void Mesh::createBuffer(GLenum type, GLuint index,
const void *data,
151 size_t size,
size_t stride,
size_t element_size,
152 GLenum datatype,
bool normalized) {
154 V_GL(glBindVertexArray(_vao));
155 V_GL(glGenBuffers(1, &vbo));
156 V_GL(glBindBuffer(type, vbo));
157 V_GL(glBufferData(type, size, data, GL_STATIC_DRAW));
158 if (type == GL_ARRAY_BUFFER) {
159 V_GL(glEnableVertexAttribArray(index));
160 V_GL(glVertexAttribPointer(index, stride / element_size, datatype,
161 normalized, stride, 0));
163 V_GL(glBindVertexArray(0));
164 V_GL(glDeleteBuffers(1, &vbo));
167 void Mesh::create() {
170 PROFILER(
"mesh callback");
176 PROFILER(
"create mesh buffers");
177 V_GL(glGenVertexArrays(1, &_vao));
178 if (!_data.positions.empty()) {
179 createBuffer(GL_ARRAY_BUFFER, (GLuint)VertexAttributes::position,
180 _data.positions.data(),
181 _data.positions.size() *
sizeof(_data.positions[0]),
182 sizeof(_data.positions[0]));
183 if (!_data.normals.empty()) {
184 createBuffer(GL_ARRAY_BUFFER, (GLuint)VertexAttributes::normal,
185 _data.normals.data(),
186 _data.normals.size() *
sizeof(_data.normals[0]),
187 sizeof(_data.normals[0]));
189 if (!_data.texcoords.empty()) {
190 createBuffer(GL_ARRAY_BUFFER, (GLuint)VertexAttributes::texcoord,
191 _data.texcoords.data(),
192 _data.texcoords.size() *
sizeof(_data.texcoords[0]),
193 sizeof(_data.texcoords[0]));
195 if (!_data.tangents.empty()) {
196 createBuffer(GL_ARRAY_BUFFER, (GLuint)VertexAttributes::tangent,
197 _data.tangents.data(),
198 _data.tangents.size() *
sizeof(_data.tangents[0]),
199 sizeof(_data.tangents[0]));
201 if (!_data.bitangents.empty()) {
202 createBuffer(GL_ARRAY_BUFFER, (GLuint)VertexAttributes::bitangent,
203 _data.bitangents.data(),
204 _data.bitangents.size() *
sizeof(_data.bitangents[0]),
205 sizeof(_data.bitangents[0]));
207 if (!_data.extras.empty()) {
208 createBuffer(GL_ARRAY_BUFFER, (GLuint)VertexAttributes::extra,
210 _data.extras.size() *
sizeof(_data.extras[0]),
211 sizeof(_data.extras[0]));
213 if (!_data.colors.empty()) {
214 createBuffer(GL_ARRAY_BUFFER, (GLuint)VertexAttributes::color,
216 _data.colors.size() *
sizeof(_data.colors[0]),
217 sizeof(_data.colors[0]));
218 }
else if (!_data.colors8.empty()) {
219 createBuffer(GL_ARRAY_BUFFER, (GLuint)VertexAttributes::color,
220 _data.colors8.data(),
221 _data.colors8.size() *
sizeof(_data.colors8[0]),
222 sizeof(_data.colors8[0]), 1, GL_UNSIGNED_BYTE,
true);
224 V_GL(glVertexAttrib4f((GLuint)VertexAttributes::color, 1, 1, 1, 1));
226 if (!_data.indices.empty()) {
227 createBuffer(GL_ELEMENT_ARRAY_BUFFER, 0, _data.indices.data(),
228 _data.indices.size() *
sizeof(_data.indices[0]),
229 sizeof(_data.indices[0]));
235 void Mesh::destroy() {
239 PROFILER(
"mesh cleanup");
240 V_GL(glDeleteVertexArrays(1, &vao));
248 throw std::runtime_error(
"mesh already destructed");
254 bool Mesh::transparent() {
261 GLuint Mesh::vertexArrayObject() {
272 V_GL(glBindVertexArray(_vao));