60void GenerateImposterLOD::generate(
62 const string& pathName,
63 const string& fileName,
65 vector<string>& imposterModelFileNames,
66 vector<Model*>& imposterModels
68 auto osEngine = Engine::createOffScreenInstance(4096, 4096,
true,
true,
false);
70 osEngine->setSceneColor(
Color4(0.0f, 0.0f, 0.0f, 0.0f));
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);
87 if (maxAxisDimension < Math::EPSILON) maxAxisDimension = 1.0f;
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)));
97 osEngine->addEntity(entity);
100 imposterModels.resize(billboardCount);
101 imposterModelFileNames.resize(billboardCount);
104 for (
auto i = 0; i < billboardCount; i++) {
106 auto yRotation =
static_cast<float>(i) * (360.0f /
static_cast<float>(billboardCount));
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);
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);
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",
143 croppedTextureHeight,
145 croppedTextureHeight,
146 croppedTextureByteBuffer
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);
163 texture->releaseReference();
166 croppedTexture->acquireReference();
167 auto scaledTexture = TextureReader::scale(croppedTexture, 1024, 1024);
168 croppedTexture->releaseReference();
171 PNGTextureWriter::write(scaledTexture, pathName, textureFileName,
false,
false);
172 scaledTexture->releaseReference();
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");
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;
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);
224 imposterModels[i] = billboard;
225 imposterModelFileNames[i] = (pathName.empty() ==
false?pathName +
"/":
"") + modelFileName;
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.
Object 3D to be used with engine class.
Bogus/Simple partition implementation.
PNG texture writer class.
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 setMaterial(Material *material)
Set up the entity's material.
void setFaces(const vector< Face > &faces)
Set up entity's faces.
Representation of a 3d model.
BoundingBox * getBoundingBox()
Represents rotation orders of a model.
Represents specular material properties.
Class representing texture UV coordinates data.
Generate imposter LOD utility class.
std::exception Exception
Exception base class.