TDME2 1.9.121
GenerateImposterLOD.cpp
Go to the documentation of this file.
2
3#include <string>
4#include <vector>
5
6#include <tdme/tdme.h>
20#include <tdme/engine/Engine.h>
24#include <tdme/math/Math.h>
30
32
33using std::string;
34using std::vector;
35
59
60void GenerateImposterLOD::generate(
61 Model* model,
62 const string& pathName,
63 const string& fileName,
64 int billboardCount,
65 vector<string>& imposterModelFileNames,
66 vector<Model*>& imposterModels
67) {
68 auto osEngine = Engine::createOffScreenInstance(4096, 4096, true, true, false);
69 osEngine->setPartition(new SimplePartition());
70 osEngine->setSceneColor(Color4(0.0f, 0.0f, 0.0f, 0.0f));
71 //
72 auto light = osEngine->getLightAt(0);
73 light->setAmbient(Color4(1.0f, 1.0f, 1.0f, 1.0f));
74 light->setDiffuse(Color4(0.5f, 0.5f, 0.5f, 1.0f));
75 light->setSpecular(Color4(1.0f, 1.0f, 1.0f, 1.0f));
76 light->setPosition(Vector4(0.0f, 20000.0f, 0.0f, 0.0f));
77 light->setSpotDirection(Vector3(0.0f, 0.0f, 0.0f).sub(Vector3(light->getPosition().getX(), light->getPosition().getY(), light->getPosition().getZ())));
78 light->setConstantAttenuation(0.5f);
79 light->setLinearAttenuation(0.0f);
80 light->setQuadraticAttenuation(0.0f);
81 light->setSpotExponent(0.0f);
82 light->setSpotCutOff(180.0f);
83 light->setEnabled(true);
84 // do a feasible scale
85 auto boundingBox = model->getBoundingBox();
86 float maxAxisDimension = Tools::computeMaxAxisDimension(boundingBox);
87 if (maxAxisDimension < Math::EPSILON) maxAxisDimension = 1.0f;
88
89 //
90 auto camera = osEngine->getCamera();
91 camera->setLookAt(boundingBox->getCenter());
92 camera->setLookFrom(boundingBox->getCenter().clone().add(Vector3(0.0f, 0.0f, boundingBox->getCenter().getZ() + maxAxisDimension * 1.25f)));
93
94 //
95 Entity* entity = new Object3D("model", model);
96 entity->addRotation(Vector3(0.0f, 1.0f, 0.0f), 0.0f);
97 osEngine->addEntity(entity);
98
99 //
100 imposterModels.resize(billboardCount);
101 imposterModelFileNames.resize(billboardCount);
102
103 //
104 for (auto i = 0; i < billboardCount; i++) {
105 //
106 auto yRotation = static_cast<float>(i) * (360.0f / static_cast<float>(billboardCount));
107 entity->setRotationAngle(0, static_cast<float>(yRotation));
108 entity->update();
109
110 //
111 osEngine->display();
112 //
113 auto modelId = Tools::removeFileEnding(fileName) + "-y" + to_string(static_cast<int>(yRotation)) + "deg";
114 auto modelFileName = modelId + ".tm";
115 auto textureFileName = Tools::removeFileEnding(fileName) + "-y" + to_string(static_cast<int>(yRotation)) + "deg.png";
116 osEngine->makeScreenshot(pathName, textureFileName, false);
117
118 //
119 auto minX = 10000;
120 auto maxX = -1;
121 auto minY = 10000;
122 auto maxY = -1;
123 auto texture = TextureReader::read(pathName, textureFileName, false, false);
124 for (auto y = 0; y < texture->getTextureHeight(); y++) {
125 for (auto x = 0; x < texture->getTextureWidth(); x++) {
126 auto alpha = texture->getTextureData()->get(y * texture->getTextureWidth() * 4 + x * 4 + 3);
127 if (alpha < 5) continue;
128 minX = Math::min(minX, x);
129 maxX = Math::max(maxX, x);
130 minY = Math::min(minY, y);
131 maxY = Math::max(maxY, y);
132 }
133 }
134
135 // crop texture
136 auto croppedTextureWidth = maxX - minX;
137 auto croppedTextureHeight = maxY - minY;
138 auto croppedTextureByteBuffer = new ByteBuffer(croppedTextureWidth * croppedTextureHeight * 4);
139 auto croppedTexture = new Texture(
140 "tdme.engine.croppedtexture",
141 32,
142 croppedTextureWidth,
143 croppedTextureHeight,
144 croppedTextureWidth,
145 croppedTextureHeight,
146 croppedTextureByteBuffer
147 );
148 for (auto y = minY; y < maxY; y++) {
149 for (auto x = minX; x < maxX; x++) {
150 auto pixelOffset = y * texture->getTextureWidth() * 4 + x * 4;
151 auto red = texture->getTextureData()->get(pixelOffset + 0);
152 auto green = texture->getTextureData()->get(pixelOffset + 1);
153 auto blue = texture->getTextureData()->get(pixelOffset + 2);
154 auto alpha = texture->getTextureData()->get(pixelOffset + 3);
155 croppedTextureByteBuffer->put(red);
156 croppedTextureByteBuffer->put(green);
157 croppedTextureByteBuffer->put(blue);
158 croppedTextureByteBuffer->put(alpha);
159 }
160 }
161
162 //
163 texture->releaseReference();
164
165 //
166 croppedTexture->acquireReference();
167 auto scaledTexture = TextureReader::scale(croppedTexture, 1024, 1024);
168 croppedTexture->releaseReference();
169
170 // save
171 PNGTextureWriter::write(scaledTexture, pathName, textureFileName, false, false);
172 scaledTexture->releaseReference();
173
174 // create model
175 auto left = boundingBox->getMin().getX();
176 auto right = boundingBox->getMax().getX();
177 auto top = boundingBox->getMin().getY();
178 auto bottom = boundingBox->getMax().getY();
179 auto depth = boundingBox->getCenter().getZ();
180 auto billboard = new Model(modelId, modelId, UpVector::Y_UP, RotationOrder::ZYX, nullptr);
181 auto billboardMaterial = new Material("billboard");
182 billboardMaterial->setSpecularMaterialProperties(new SpecularMaterialProperties());
183 billboardMaterial->getSpecularMaterialProperties()->setSpecularColor(Color4(0.0f, 0.0f, 0.0f, 1.0f));
184 billboardMaterial->getSpecularMaterialProperties()->setDiffuseTexture(pathName, textureFileName);
185 billboardMaterial->getSpecularMaterialProperties()->setDiffuseTextureMaskedTransparency(true);
186 billboard->getMaterials()[billboardMaterial->getId()] = billboardMaterial;
187 auto billboardNode = new Node(billboard, nullptr, "billboard", "billboard");
188 vector<Vector3> billboardVertices;
189 billboardVertices.push_back(Vector3(left, top, depth));
190 billboardVertices.push_back(Vector3(left, bottom, depth));
191 billboardVertices.push_back(Vector3(right, bottom, depth));
192 billboardVertices.push_back(Vector3(right, top, depth));
193 vector<Vector3> billboardNormals;
194 billboardNormals.push_back(Vector3(0.0f, 1.0f, 0.0f));
195 vector<TextureCoordinate> billboardTextureCoordinates;
196 billboardTextureCoordinates.push_back(TextureCoordinate(0.0f, 0.0f));
197 billboardTextureCoordinates.push_back(TextureCoordinate(0.0f, 1.0f));
198 billboardTextureCoordinates.push_back(TextureCoordinate(1.0f, 1.0f));
199 billboardTextureCoordinates.push_back(TextureCoordinate(1.0f, 0.0f));
200 vector<Face> billboardFacesGround;
201 billboardFacesGround.push_back(Face(billboardNode, 0, 1, 2, 0, 0, 0, 0, 1, 2));
202 billboardFacesGround.push_back(Face(billboardNode, 2, 3, 0, 0, 0, 0, 2, 3, 0));
203 FacesEntity billboardNodeFacesEntity(billboardNode, "billboard.facesentity");
204 billboardNodeFacesEntity.setMaterial(billboardMaterial);
205 vector<FacesEntity> billboardNodeFacesEntities;
206 billboardNodeFacesEntity.setFaces(billboardFacesGround);
207 billboardNodeFacesEntities.push_back(billboardNodeFacesEntity);
208 billboardNode->setVertices(billboardVertices);
209 billboardNode->setNormals(billboardNormals);
210 billboardNode->setTextureCoordinates(billboardTextureCoordinates);
211 billboardNode->setFacesEntities(billboardNodeFacesEntities);
212 billboard->getNodes()["billboard"] = billboardNode;
213 billboard->getSubNodes()["billboard"] = billboardNode;
214 ModelTools::prepareForIndexedRendering(billboard);
215
216 //
217 TMWriter::write(
218 billboard,
219 pathName,
220 modelFileName
221 );
222
223 //
224 imposterModels[i] = billboard;
225 imposterModelFileNames[i] = (pathName.empty() == false?pathName + "/":"") + modelFileName;
226 }
227
228 //
229 osEngine->dispose();
230 delete osEngine;
231}
Engine main class.
Definition: Engine.h:122
TDME engine entity.
Definition: Entity.h:31
virtual void addRotation(const Vector3 &axis, const float angle)=0
Add rotation.
virtual void update()=0
Update transformations.
virtual void setRotationAngle(const int idx, const float angle)=0
LOD object 3D to be used with engine class.
Definition: LODObject3D.h:47
Object 3D to be used with engine class.
Definition: Object3D.h:60
Bogus/Simple partition implementation.
Color 4 definition.
Definition: Color4.h:20
Represents a model face, consisting of vertex, normal, tangent and bitangent vectors,...
Definition: Face.h:19
Node faces entity A node can have multiple entities containing faces and a applied material.
Definition: FacesEntity.h:28
void setMaterial(Material *material)
Set up the entity's material.
Definition: FacesEntity.h:72
void setFaces(const vector< Face > &faces)
Set up entity's faces.
Definition: FacesEntity.cpp:47
Represents a material.
Definition: Material.h:21
Representation of a 3d model.
Definition: Model.h:32
BoundingBox * getBoundingBox()
Definition: Model.cpp:142
Model node.
Definition: Node.h:31
Represents rotation orders of a model.
Definition: RotationOrder.h:23
Represents specular material properties.
Class representing texture UV coordinates data.
Model up vector.
Definition: UpVector.h:20
Standard math functions.
Definition: Math.h:21
3D vector 3 class
Definition: Vector3.h:22
3D vector 4 class
Definition: Vector4.h:19
Generate imposter LOD utility class.
static float computeMaxAxisDimension(BoundingBox *boundingBox)
Compute max axis dimension for given bounding box.
Definition: Tools.cpp:178
static const string removeFileEnding(const string &fileName)
Remove file ending, e.g.
Definition: Tools.cpp:552
Byte buffer class.
Definition: ByteBuffer.h:24
Console class.
Definition: Console.h:26
Exception base class.
Definition: ExceptionBase.h:20
Model tools functions class.
Definition: ModelTools.h:38
std::exception Exception
Exception base class.
Definition: Exception.h:19