70Model* TMReader::read(
const string& pathName,
const string& fileName)
73 FileSystem::getInstance()->getContent(pathName, fileName, data);
74 return read(data, pathName, fileName);
80 if (fileId.length() == 0 || fileId !=
"TDME Model") {
82 "File is not a TDME model file, file id = '" +
87 array<uint8_t, 3> version;
91 if ((version[0] != 1 || version[1] != 0 || version[2] != 0) &&
92 (version[0] != 1 || version[1] != 9 || version[2] != 9) &&
93 (version[0] != 1 || version[1] != 9 || version[2] != 10) &&
94 (version[0] != 1 || version[1] != 9 || version[2] != 11) &&
95 (version[0] != 1 || version[1] != 9 || version[2] != 12) &&
96 (version[0] != 1 || version[1] != 9 || version[2] != 13) &&
97 (version[0] != 1 || version[1] != 9 || version[2] != 14) &&
98 (version[0] != 1 || version[1] != 9 || version[2] != 15) &&
99 (version[0] != 1 || version[1] != 9 || version[2] != 16) &&
100 (version[0] != 1 || version[1] != 9 || version[2] != 17)) {
102 "Version mismatch, should be 1.0.0, 1.9.9, 1.9.10, 1.9.11, 1.9.12, 1.9.13, 1.9.14, 1.9.15, 1.9.16, 1.9.17 but is " +
103 to_string(version[0]) +
105 to_string(version[1]) +
107 to_string(version[2])
111 auto upVector = UpVector::valueOf(is.
readString());
112 auto rotationOrder = RotationOrder::valueOf(is.
readString());
113 auto shaderModel = ShaderModel::SPECULAR;
114 if ((version[0] == 1 && version[1] == 9 && version[2] == 14) ||
115 (version[0] == 1 && version[1] == 9 && version[2] == 15) ||
116 (version[0] == 1 && version[1] == 9 && version[2] == 16) ||
117 (version[0] == 1 && version[1] == 9 && version[2] == 17)) {
118 shaderModel = ShaderModel::valueOf(is.
readString());
120 array<float, 3> boundingBoxMinXYZ;
122 array<float, 3> boundingBoxMaxXYZ;
125 auto model =
new Model(
127 fileName.empty() ==
true?name:fileName,
132 model->setShaderModel(shaderModel);
134 array<float, 16> importTransformationsMatrixArray;
136 model->setImportTransformationsMatrix(importTransformationsMatrixArray);
137 map<string, Texture*> embeddedTextures;
138 if (version[0] == 1 && version[1] == 9 && version[2] == 17) {
141 auto materialCount = is.
readInt();
142 for (
auto i = 0; i < materialCount; i++) {
143 auto material =
readMaterial(pathName, &is, embeddedTextures, version);
144 model->getMaterials()[material->getId()] = material;
146 readSubNodes(&is, model,
nullptr, model->getSubNodes());
147 auto animationSetupCount = is.
readInt();
148 for (
auto i = 0; i < animationSetupCount; i++) {
151 if (model->getAnimationSetup(Model::ANIMATIONSETUP_DEFAULT) ==
nullptr) {
152 model->addAnimationSetup(Model::ANIMATIONSETUP_DEFAULT, 0, 0,
true);
154 for (
auto it: embeddedTextures) {
155 it.second->releaseReference();
161 if (FileSystem::getInstance()->fileExists(texturePathName +
"/" + textureFileName) ==
true) {
162 return texturePathName;
164 if (FileSystem::getInstance()->fileExists(modelPathName +
"/" + textureFileName) ==
true) {
165 return modelPathName;
167 if (FileSystem::getInstance()->fileExists(FileSystem::getInstance()->getPathName(modelPathName) +
"/" + textureFileName) ==
true) {
168 return FileSystem::getInstance()->getPathName(modelPathName);
170 return texturePathName;
175 auto embeddedTextureCount = is->
readInt();
176 for (
auto i = 0; i < embeddedTextureCount; i++) {
178 auto embeddedTextureType = is->
readByte();
179 if (embeddedTextureType == 1) {
180 auto textureSize = is->
readInt();
181 vector<uint8_t> pngData;
182 pngData.resize(textureSize);
183 for (
auto j = 0; j < textureSize; j++) pngData[j] = is->
readByte();
184 auto embeddedTexture = TextureReader::readPNG(embeddedTextureId, pngData,
true);
185 if (embeddedTexture !=
nullptr) {
186 embeddedTexture->acquireReference();
187 embeddedTextures[embeddedTextureId] = embeddedTexture;
198 if (version[0] == 1 && version[1] == 9 && version[2] == 17) {
201 array<float, 4> colorRGBAArray;
203 smp->setAmbientColor(
Color4(colorRGBAArray));
205 smp->setDiffuseColor(
Color4(colorRGBAArray));
207 smp->setSpecularColor(
Color4(colorRGBAArray));
209 smp->setEmissionColor(
Color4(colorRGBAArray));
211 if ((version[0] == 1 && version[1] == 9 && version[2] == 15) ||
212 (version[0] == 1 && version[1] == 9 && version[2] == 16) ||
213 (version[0] == 1 && version[1] == 9 && version[2] == 17)) {
214 smp->setTextureAtlasSize(is->
readInt());
216 auto diffuseTexturePathName = is->
readString();
217 auto diffuseTextureFileName = is->
readString();
218 auto diffuseTransparencyTexturePathName = is->
readString();
219 auto diffuseTransparencyTextureFileName = is->
readString();
220 if (smp->hasEmbeddedTextures() ==
false && diffuseTextureFileName.empty() ==
false) {
221 smp->setDiffuseTexture(
222 getTexturePath(pathName, diffuseTexturePathName, diffuseTextureFileName),
223 diffuseTextureFileName,
224 getTexturePath(pathName, diffuseTransparencyTexturePathName, diffuseTransparencyTextureFileName),
225 diffuseTransparencyTextureFileName
228 auto specularTexturePathName = is->
readString();
229 auto specularTextureFileName = is->
readString();
230 if (smp->hasEmbeddedTextures() ==
false && specularTextureFileName.empty() ==
false) {
231 smp->setSpecularTexture(
232 getTexturePath(pathName, specularTexturePathName, specularTextureFileName),
233 specularTextureFileName
236 auto normalTexturePathName = is->
readString();
237 auto normalTextureFileName = is->
readString();
238 if (smp->hasEmbeddedTextures() ==
false && normalTextureFileName.empty() ==
false) {
239 smp->setNormalTexture(
240 getTexturePath(pathName, normalTexturePathName, normalTextureFileName),
241 normalTextureFileName
244 if ((version[0] == 1 && version[1] == 9 && version[2] == 9) ||
245 (version[0] == 1 && version[1] == 9 && version[2] == 10) ||
246 (version[0] == 1 && version[1] == 9 && version[2] == 11)) {
247 auto displacementTexturePathName = is->
readString();
248 auto displacementTextureFileName = is->
readString();
250 if ((version[0] == 1 && version[1] == 9 && version[2] == 15) ||
251 (version[0] == 1 && version[1] == 9 && version[2] == 16) ||
252 (version[0] == 1 && version[1] == 9 && version[2] == 17)) {
253 smp->setDiffuseTextureTransparency(is->
readBoolean());
255 smp->setDiffuseTextureMaskedTransparency(is->
readBoolean());
256 if ((version[0] == 1 && version[1] == 9 && version[2] == 9) ||
257 (version[0] == 1 && version[1] == 9 && version[2] == 10) ||
258 (version[0] == 1 && version[1] == 9 && version[2] == 11) ||
259 (version[0] == 1 && version[1] == 9 && version[2] == 12) ||
260 (version[0] == 1 && version[1] == 9 && version[2] == 13) ||
261 (version[0] == 1 && version[1] == 9 && version[2] == 14) ||
262 (version[0] == 1 && version[1] == 9 && version[2] == 15) ||
263 (version[0] == 1 && version[1] == 9 && version[2] == 16) ||
264 (version[0] == 1 && version[1] == 9 && version[2] == 17)) {
265 smp->setDiffuseTextureMaskedTransparencyThreshold(is->
readFloat());
267 if ((version[0] == 1 && version[1] == 9 && version[2] == 16) ||
268 (version[0] == 1 && version[1] == 9 && version[2] == 17)) {
271 if ((version[0] == 1 && version[1] == 9 && version[2] == 10) ||
272 (version[0] == 1 && version[1] == 9 && version[2] == 11) ||
273 (version[0] == 1 && version[1] == 9 && version[2] == 12) ||
274 (version[0] == 1 && version[1] == 9 && version[2] == 13) ||
275 (version[0] == 1 && version[1] == 9 && version[2] == 14) ||
276 (version[0] == 1 && version[1] == 9 && version[2] == 15) ||
277 (version[0] == 1 && version[1] == 9 && version[2] == 16) ||
278 (version[0] == 1 && version[1] == 9 && version[2] == 17)) {
279 array<float, 9> textureMatrix;
283 if (version[0] == 1 && version[1] == 9 && version[2] == 17) {
284 if (smp->hasEmbeddedTextures() ==
true) {
286 if (diffuseTextureFileName.empty() ==
false) {
287 auto diffuseTextureTransparency = smp->getDiffuseTextureTransparency();
288 auto diffuseTextureMaskedTransparency = smp->getDiffuseTextureMaskedTransparencyThreshold();
289 smp->setDiffuseTexture(embeddedTextures.find(diffuseTextureFileName)->second);
290 smp->setDiffuseTextureTransparency(diffuseTextureTransparency);
291 smp->setDiffuseTextureMaskedTransparency(diffuseTextureMaskedTransparency);
294 if (specularTextureFileName.empty() ==
false) {
295 smp->setSpecularTexture(embeddedTextures.find(specularTextureFileName)->second);
298 if (normalTextureFileName.empty() ==
false) {
299 smp->setNormalTexture(embeddedTextures.find(normalTextureFileName)->second);
303 m->setSpecularMaterialProperties(smp);
304 if ((version[0] == 1 && version[1] == 9 && version[2] == 13) ||
305 (version[0] == 1 && version[1] == 9 && version[2] == 14) ||
306 (version[0] == 1 && version[1] == 9 && version[2] == 15) ||
307 (version[0] == 1 && version[1] == 9 && version[2] == 16) ||
308 (version[0] == 1 && version[1] == 9 && version[2] == 17)) {
311 if (version[0] == 1 && version[1] == 9 && version[2] == 17) {
315 pmp->setBaseColorFactor(
Color4(colorRGBAArray));
316 auto baseColorTexturePathName = is->
readString();
317 auto baseColorTextureFileName = is->
readString();
318 if (pmp->hasEmbeddedTextures() ==
false && baseColorTextureFileName.empty() ==
false) {
319 pmp->setBaseColorTexture(baseColorTexturePathName, baseColorTextureFileName);
321 pmp->setBaseColorTextureMaskedTransparency(is->
readBoolean());
322 pmp->setBaseColorTextureMaskedTransparency(is->
readFloat());
324 pmp->setRoughnessFactor(is->
readFloat());
325 auto metallicRoughnessTexturePathName = is->
readString();
326 auto metallicRoughnessTextureFileName = is->
readString();
327 if (pmp->hasEmbeddedTextures() ==
false && metallicRoughnessTextureFileName.empty() ==
false) {
328 pmp->setMetallicRoughnessTexture(metallicRoughnessTexturePathName, metallicRoughnessTextureFileName);
331 auto pbrNormalTexturePathName = is->
readString();
332 auto pbrNormalTextureFileName = is->
readString();
333 if (pmp->hasEmbeddedTextures() ==
false && pbrNormalTextureFileName.empty() ==
false) {
334 pmp->setNormalTexture(pbrNormalTexturePathName, pbrNormalTextureFileName);
337 if (version[0] == 1 && version[1] == 9 && version[2] == 17) {
338 if (pmp->hasEmbeddedTextures() ==
true) {
340 if (baseColorTextureFileName.empty() ==
false) {
341 auto baseColorTextureTransparency = pmp->hasBaseColorTextureTransparency();
342 auto baseColorTextureMaskedTransparency = pmp->hasBaseColorTextureMaskedTransparency();
343 pmp->setBaseColorTexture(embeddedTextures.find(baseColorTextureFileName)->second);
344 pmp->setBaseColorTextureTransparency(baseColorTextureTransparency);
345 pmp->setBaseColorTextureMaskedTransparency(baseColorTextureMaskedTransparency);
348 if (metallicRoughnessTextureFileName.empty() ==
false) {
349 pmp->setMetallicRoughnessTexture(embeddedTextures.find(metallicRoughnessTextureFileName)->second);
352 if (pbrNormalTextureFileName.empty() ==
false) {
353 pmp->setNormalTexture(embeddedTextures.find(pbrNormalTextureFileName)->second);
357 m->setPBRMaterialProperties(pmp);
366 auto startFrame = is->
readInt();
370 if ((version[0] == 1 && version[1] == 9 && version[2] == 11) ||
371 (version[0] == 1 && version[1] == 9 && version[2] == 12) ||
372 (version[0] == 1 && version[1] == 9 && version[2] == 13) ||
373 (version[0] == 1 && version[1] == 9 && version[2] == 14) ||
374 (version[0] == 1 && version[1] == 9 && version[2] == 15) ||
375 (version[0] == 1 && version[1] == 9 && version[2] == 16) ||
376 (version[0] == 1 && version[1] == 9 && version[2] == 17)) {
379 if (overlayFromNodeId.length() == 0) {
389 array<float, 3> vXYZ;
392 for (
auto i = 0; i < v.size(); i++) {
402 array<float, 2> tcUV;
403 vector<TextureCoordinate> tc;
406 for (
auto i = 0; i < tc.size(); i++) {
420 if (length != indices->size()) {
423 for (
auto i = 0; i < indices->size(); i++) {
435 array<float, 16> matrixArray;
438 vector<Matrix4x4> transformationsMatrices;
439 transformationsMatrices.resize(frames);
440 for (
auto i = 0; i < transformationsMatrices.size(); i++) {
442 transformationsMatrices[i].set(matrixArray);
444 animation->setTransformationsMatrices(transformationsMatrices);
452 vector<FacesEntity> facesEntities;
453 facesEntities.resize(is->
readInt());
454 for (
auto i = 0; i < facesEntities.size(); i++) {
460 material = materialIt->second;
462 facesEntities[i].setMaterial(material);
466 array<int32_t,3> vertexIndices;
467 array<int32_t,3> normalIndices;
468 array<int32_t,3> textureCoordinateIndices;
469 array<int32_t,3> tangentIndices;
470 array<int32_t,3> bitangentIndices;
471 bool haveTextureCoordinateIndices;
472 bool haveTangentIndices;
473 bool haveBitangentIndices;
474 for (
auto j = 0; j < faces.size(); j++) {
477 haveTextureCoordinateIndices =
readIndices(is, &textureCoordinateIndices);
478 haveTangentIndices =
readIndices(is, &tangentIndices);
479 haveBitangentIndices =
readIndices(is, &bitangentIndices);
481 vertexIndices[0], vertexIndices[1], vertexIndices[2],
482 normalIndices[0], normalIndices[1], normalIndices[2]
484 if (haveTextureCoordinateIndices ==
true) {
485 faces[j].setTextureCoordinateIndices(
486 textureCoordinateIndices[0], textureCoordinateIndices[1], textureCoordinateIndices[2]
489 if (haveTangentIndices ==
true && haveBitangentIndices ==
true) {
490 faces[j].setTangentIndices(tangentIndices[0], tangentIndices[1], tangentIndices[2]);
491 faces[j].setBitangentIndices(bitangentIndices[0], bitangentIndices[1], bitangentIndices[2]);
494 facesEntities[i].setFaces(faces);
501 array<float, 16> matrixArray;
511 int32_t jointIndex = is->
readInt();
512 int32_t weightIndex = is->
readInt();
521 vector<Joint> joints;
523 for (
auto i = 0; i < joints.size(); i++) {
526 skinning->setJoints(joints);
527 vector<vector<JointWeight>> verticesJointsWeight;
528 verticesJointsWeight.resize(is->
readInt());
529 for (
auto i = 0; i < verticesJointsWeight.size(); i++) {
530 verticesJointsWeight[i].resize(is->
readInt());
531 for (
auto j = 0; j < verticesJointsWeight[i].size(); j++) {
535 skinning->setVerticesJointsWeights(verticesJointsWeight);
542 auto subNodeCount = is->
readInt();
543 for (
auto i = 0; i < subNodeCount; i++) {
544 auto subNode =
readNode(is, model, parentNode);
545 subNodes[subNode->getId()] = subNode;
546 model->
getNodes()[subNode->getId()] = subNode;
555 auto node =
new Node(model, parentNode, nodeId, nodeName);
557 array<float, 16> matrixArray;
559 node->setTransformationsMatrix(
Matrix4x4(matrixArray));
561 node->setVertices(vertices);
563 node->setNormals(normals);
565 node->setTextureCoordinates(textureCoordinates);
567 node->setTangents(tangents);
569 node->setBitangents(bitangents);
static void readEmbeddedTextures(TMReaderInputStream *is, map< string, Texture * > &embeddedTextures)
Read material.
static Material * readMaterial(const string &pathName, TMReaderInputStream *is, const map< string, Texture * > &embeddedTextures, const array< uint8_t, 3 > &version)
Read material.
static void readSubNodes(TMReaderInputStream *is, Model *model, Node *parentNode, map< string, Node * > &subNodes)
Read sub nodes.
static Joint readSkinningJoint(TMReaderInputStream *is)
Read skinning joint.
static const string getTexturePath(const string &modelPathName, const string &texturePathName, const string &textureFileName)
Get texture path.
static const vector< Vector3 > readVertices(TMReaderInputStream *is)
Read vertices from input stream.
static Model * read(const string &pathName, const string &fileName)
TDME model format reader.
static Animation * readAnimation(TMReaderInputStream *is, Node *g)
Read animation from input stream into node.
static void readFacesEntities(TMReaderInputStream *is, Node *g)
Read faces entities from input stream.
static void readAnimationSetup(TMReaderInputStream *is, Model *model, const array< uint8_t, 3 > &version)
Read animation setup.
static void readSkinning(TMReaderInputStream *is, Node *g)
Read skinning from input stream.
static bool readIndices(TMReaderInputStream *is, array< int32_t, 3 > *indices)
Read indices from input stream.
static JointWeight readSkinningJointWeight(TMReaderInputStream *is)
Read skinning joint weight.
static const vector< TextureCoordinate > readTextureCoordinates(TMReaderInputStream *is)
Read texture coordinates from input stream.
static Node * readNode(TMReaderInputStream *is, Model *model, Node *parentNode)
Write node to output stream.
Represents a model face, consisting of vertex, normal, tangent and bitangent vectors,...
Node faces entity A node can have multiple entities containing faces and a applied material.
void setBindMatrix(const Matrix4x4 &bindMatrix)
Bind matrix.
Representation of a 3d model.
map< string, Node * > & getNodes()
Returns all object's nodes.
AnimationSetup * addAnimationSetup(const string &id, int32_t startFrame, int32_t endFrame, bool loop, float speed=1.0f)
Adds an base animation setup.
AnimationSetup * addOverlayAnimationSetup(const string &id, const string &overlayFromNodeId, int32_t startFrame, int32_t endFrame, bool loop, float speed=1.0f)
Adds an overlay animation setup.
map< string, Material * > & getMaterials()
Returns all object materials.
void setAnimation(Animation *animation)
Sets animation object.
void setFacesEntities(const vector< FacesEntity > &facesEntities)
Set up faces entities.
void setSkinning(Skinning *skinning)
Sets skinning object.
Animation * getAnimation()
map< string, Node * > & getSubNodes()
Represents specular material properties.
Represents rotation orders of a model.
Skinning definition for nodes.
Represents specular material properties.
Class representing texture UV coordinates data.
Axis aligned bounding box used for frustum, this is not directly connectable with physics engine.
File system singleton class.