3 Copyright (C) 2010-2013 celeron55, Perttu Ahola <celeron55@gmail.com>
5 This program is free software; you can redistribute it and/or modify
6 it under the terms of the GNU Lesser General Public License as published by
7 the Free Software Foundation; either version 2.1 of the License, or
8 (at your option) any later version.
10 This program is distributed in the hope that it will be useful,
11 but WITHOUT ANY WARRANTY; without even the implied warranty of
12 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 GNU Lesser General Public License for more details.
15 You should have received a copy of the GNU Lesser General Public License along
16 with this program; if not, write to the Free Software Foundation, Inc.,
17 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
24 #include <IAnimatedMesh.h>
25 #include <SAnimatedMesh.h>
26 #include <ICameraSceneNode.h>
28 // In Irrlicht 1.8 the signature of ITexture::lock was changed from
29 // (bool, u32) to (E_TEXTURE_LOCK_MODE, u32).
30 #if IRRLICHT_VERSION_MAJOR == 1 && IRRLICHT_VERSION_MINOR <= 7
31 #define MY_ETLM_READ_ONLY true
33 #define MY_ETLM_READ_ONLY video::ETLM_READ_ONLY
36 scene::IAnimatedMesh* createCubeMesh(v3f scale)
38 video::SColor c(255,255,255,255);
39 video::S3DVertex vertices[24] =
42 video::S3DVertex(-0.5,+0.5,-0.5, 0,1,0, c, 0,1),
43 video::S3DVertex(-0.5,+0.5,+0.5, 0,1,0, c, 0,0),
44 video::S3DVertex(+0.5,+0.5,+0.5, 0,1,0, c, 1,0),
45 video::S3DVertex(+0.5,+0.5,-0.5, 0,1,0, c, 1,1),
47 video::S3DVertex(-0.5,-0.5,-0.5, 0,-1,0, c, 0,0),
48 video::S3DVertex(+0.5,-0.5,-0.5, 0,-1,0, c, 1,0),
49 video::S3DVertex(+0.5,-0.5,+0.5, 0,-1,0, c, 1,1),
50 video::S3DVertex(-0.5,-0.5,+0.5, 0,-1,0, c, 0,1),
52 video::S3DVertex(+0.5,-0.5,-0.5, 1,0,0, c, 0,1),
53 video::S3DVertex(+0.5,+0.5,-0.5, 1,0,0, c, 0,0),
54 video::S3DVertex(+0.5,+0.5,+0.5, 1,0,0, c, 1,0),
55 video::S3DVertex(+0.5,-0.5,+0.5, 1,0,0, c, 1,1),
57 video::S3DVertex(-0.5,-0.5,-0.5, -1,0,0, c, 1,1),
58 video::S3DVertex(-0.5,-0.5,+0.5, -1,0,0, c, 0,1),
59 video::S3DVertex(-0.5,+0.5,+0.5, -1,0,0, c, 0,0),
60 video::S3DVertex(-0.5,+0.5,-0.5, -1,0,0, c, 1,0),
62 video::S3DVertex(-0.5,-0.5,+0.5, 0,0,1, c, 1,1),
63 video::S3DVertex(+0.5,-0.5,+0.5, 0,0,1, c, 0,1),
64 video::S3DVertex(+0.5,+0.5,+0.5, 0,0,1, c, 0,0),
65 video::S3DVertex(-0.5,+0.5,+0.5, 0,0,1, c, 1,0),
67 video::S3DVertex(-0.5,-0.5,-0.5, 0,0,-1, c, 0,1),
68 video::S3DVertex(-0.5,+0.5,-0.5, 0,0,-1, c, 0,0),
69 video::S3DVertex(+0.5,+0.5,-0.5, 0,0,-1, c, 1,0),
70 video::S3DVertex(+0.5,-0.5,-0.5, 0,0,-1, c, 1,1),
73 u16 indices[6] = {0,1,2,2,3,0};
75 scene::SMesh *mesh = new scene::SMesh();
76 for (u32 i=0; i<6; ++i)
78 scene::IMeshBuffer *buf = new scene::SMeshBuffer();
79 buf->append(vertices + 4 * i, 4, indices, 6);
80 // Set default material
81 buf->getMaterial().setFlag(video::EMF_LIGHTING, false);
82 buf->getMaterial().setFlag(video::EMF_BILINEAR_FILTER, false);
83 buf->getMaterial().MaterialType = video::EMT_TRANSPARENT_ALPHA_CHANNEL_REF;
84 // Add mesh buffer to mesh
85 mesh->addMeshBuffer(buf);
89 scene::SAnimatedMesh *anim_mesh = new scene::SAnimatedMesh(mesh);
91 scaleMesh(anim_mesh, scale); // also recalculates bounding box
95 static scene::IAnimatedMesh* extrudeARGB(u32 twidth, u32 theight, u8 *data)
97 const s32 argb_wstep = 4 * twidth;
98 const s32 alpha_threshold = 1;
100 scene::IMeshBuffer *buf = new scene::SMeshBuffer();
101 video::SColor c(255,255,255,255);
105 video::S3DVertex vertices[8] =
107 video::S3DVertex(-0.5,-0.5,-0.5, 0,0,-1, c, 0,1),
108 video::S3DVertex(-0.5,+0.5,-0.5, 0,0,-1, c, 0,0),
109 video::S3DVertex(+0.5,+0.5,-0.5, 0,0,-1, c, 1,0),
110 video::S3DVertex(+0.5,-0.5,-0.5, 0,0,-1, c, 1,1),
111 video::S3DVertex(+0.5,-0.5,+0.5, 0,0,+1, c, 1,1),
112 video::S3DVertex(+0.5,+0.5,+0.5, 0,0,+1, c, 1,0),
113 video::S3DVertex(-0.5,+0.5,+0.5, 0,0,+1, c, 0,0),
114 video::S3DVertex(-0.5,-0.5,+0.5, 0,0,+1, c, 0,1),
116 u16 indices[12] = {0,1,2,2,3,0,4,5,6,6,7,4};
117 buf->append(vertices, 8, indices, 12);
121 // (add faces where a solid pixel is next to a transparent one)
122 u8 *solidity = new u8[(twidth+2) * (theight+2)];
123 u32 wstep = twidth + 2;
124 for (u32 y = 0; y < theight + 2; ++y)
126 u8 *scanline = solidity + y * wstep;
127 if (y == 0 || y == theight + 1)
129 for (u32 x = 0; x < twidth + 2; ++x)
135 u8 *argb_scanline = data + (y - 1) * argb_wstep;
136 for (u32 x = 0; x < twidth; ++x)
137 scanline[x+1] = (argb_scanline[x*4+3] >= alpha_threshold);
138 scanline[twidth + 1] = 0;
142 // without this, there would be occasional "holes" in the mesh
145 for (u32 y = 0; y <= theight; ++y)
147 u8 *scanline = solidity + y * wstep + 1;
148 for (u32 x = 0; x <= twidth; ++x)
150 if (scanline[x] && !scanline[x + wstep])
153 while (scanline[xx] && !scanline[xx + wstep])
155 f32 vx1 = (x - eps) / (f32) twidth - 0.5;
156 f32 vx2 = (xx + eps) / (f32) twidth - 0.5;
157 f32 vy = 0.5 - (y - eps) / (f32) theight;
158 f32 tx1 = x / (f32) twidth;
159 f32 tx2 = xx / (f32) twidth;
160 f32 ty = (y - 0.5) / (f32) theight;
161 video::S3DVertex vertices[8] =
163 video::S3DVertex(vx1,vy,-0.5, 0,-1,0, c, tx1,ty),
164 video::S3DVertex(vx2,vy,-0.5, 0,-1,0, c, tx2,ty),
165 video::S3DVertex(vx2,vy,+0.5, 0,-1,0, c, tx2,ty),
166 video::S3DVertex(vx1,vy,+0.5, 0,-1,0, c, tx1,ty),
168 u16 indices[6] = {0,1,2,2,3,0};
169 buf->append(vertices, 4, indices, 6);
172 if (!scanline[x] && scanline[x + wstep])
175 while (!scanline[xx] && scanline[xx + wstep])
177 f32 vx1 = (x - eps) / (f32) twidth - 0.5;
178 f32 vx2 = (xx + eps) / (f32) twidth - 0.5;
179 f32 vy = 0.5 - (y + eps) / (f32) theight;
180 f32 tx1 = x / (f32) twidth;
181 f32 tx2 = xx / (f32) twidth;
182 f32 ty = (y + 0.5) / (f32) theight;
183 video::S3DVertex vertices[8] =
185 video::S3DVertex(vx1,vy,-0.5, 0,1,0, c, tx1,ty),
186 video::S3DVertex(vx1,vy,+0.5, 0,1,0, c, tx1,ty),
187 video::S3DVertex(vx2,vy,+0.5, 0,1,0, c, tx2,ty),
188 video::S3DVertex(vx2,vy,-0.5, 0,1,0, c, tx2,ty),
190 u16 indices[6] = {0,1,2,2,3,0};
191 buf->append(vertices, 4, indices, 6);
197 for (u32 x = 0; x <= twidth; ++x)
199 u8 *scancol = solidity + x + wstep;
200 for (u32 y = 0; y <= theight; ++y)
202 if (scancol[y * wstep] && !scancol[y * wstep + 1])
205 while (scancol[yy * wstep] && !scancol[yy * wstep + 1])
207 f32 vx = (x - eps) / (f32) twidth - 0.5;
208 f32 vy1 = 0.5 - (y - eps) / (f32) theight;
209 f32 vy2 = 0.5 - (yy + eps) / (f32) theight;
210 f32 tx = (x - 0.5) / (f32) twidth;
211 f32 ty1 = y / (f32) theight;
212 f32 ty2 = yy / (f32) theight;
213 video::S3DVertex vertices[8] =
215 video::S3DVertex(vx,vy1,-0.5, 1,0,0, c, tx,ty1),
216 video::S3DVertex(vx,vy1,+0.5, 1,0,0, c, tx,ty1),
217 video::S3DVertex(vx,vy2,+0.5, 1,0,0, c, tx,ty2),
218 video::S3DVertex(vx,vy2,-0.5, 1,0,0, c, tx,ty2),
220 u16 indices[6] = {0,1,2,2,3,0};
221 buf->append(vertices, 4, indices, 6);
224 if (!scancol[y * wstep] && scancol[y * wstep + 1])
227 while (!scancol[yy * wstep] && scancol[yy * wstep + 1])
229 f32 vx = (x + eps) / (f32) twidth - 0.5;
230 f32 vy1 = 0.5 - (y - eps) / (f32) theight;
231 f32 vy2 = 0.5 - (yy + eps) / (f32) theight;
232 f32 tx = (x + 0.5) / (f32) twidth;
233 f32 ty1 = y / (f32) theight;
234 f32 ty2 = yy / (f32) theight;
235 video::S3DVertex vertices[8] =
237 video::S3DVertex(vx,vy1,-0.5, -1,0,0, c, tx,ty1),
238 video::S3DVertex(vx,vy2,-0.5, -1,0,0, c, tx,ty2),
239 video::S3DVertex(vx,vy2,+0.5, -1,0,0, c, tx,ty2),
240 video::S3DVertex(vx,vy1,+0.5, -1,0,0, c, tx,ty1),
242 u16 indices[6] = {0,1,2,2,3,0};
243 buf->append(vertices, 4, indices, 6);
250 scene::SMesh *mesh = new scene::SMesh();
251 mesh->addMeshBuffer(buf);
253 scene::SAnimatedMesh *anim_mesh = new scene::SAnimatedMesh(mesh);
258 scene::IAnimatedMesh* createExtrudedMesh(video::ITexture *texture,
259 video::IVideoDriver *driver, v3f scale)
261 scene::IAnimatedMesh *mesh = NULL;
262 core::dimension2d<u32> size = texture->getSize();
263 video::ECOLOR_FORMAT format = texture->getColorFormat();
264 if (format == video::ECF_A8R8G8B8)
266 // Texture is in the correct color format, we can pass it
267 // to extrudeARGB right away.
268 void *data = texture->lock(MY_ETLM_READ_ONLY);
271 mesh = extrudeARGB(size.Width, size.Height, (u8*) data);
276 video::IImage *img1 = driver->createImageFromData(format, size, texture->lock(MY_ETLM_READ_ONLY));
280 // img1 is in the texture's color format, convert to 8-bit ARGB
281 video::IImage *img2 = driver->createImage(video::ECF_A8R8G8B8, size);
287 mesh = extrudeARGB(size.Width, size.Height, (u8*) img2->lock());
294 // Set default material
295 mesh->getMeshBuffer(0)->getMaterial().setTexture(0, texture);
296 mesh->getMeshBuffer(0)->getMaterial().setFlag(video::EMF_LIGHTING, false);
297 mesh->getMeshBuffer(0)->getMaterial().setFlag(video::EMF_BILINEAR_FILTER, false);
298 mesh->getMeshBuffer(0)->getMaterial().MaterialType = video::EMT_TRANSPARENT_ALPHA_CHANNEL_REF;
300 scaleMesh(mesh, scale); // also recalculates bounding box
304 void scaleMesh(scene::IMesh *mesh, v3f scale)
309 core::aabbox3d<f32> bbox;
312 u16 mc = mesh->getMeshBufferCount();
313 for(u16 j=0; j<mc; j++)
315 scene::IMeshBuffer *buf = mesh->getMeshBuffer(j);
316 video::S3DVertex *vertices = (video::S3DVertex*)buf->getVertices();
317 u16 vc = buf->getVertexCount();
318 for(u16 i=0; i<vc; i++)
320 vertices[i].Pos *= scale;
322 buf->recalculateBoundingBox();
324 // calculate total bounding box
326 bbox = buf->getBoundingBox();
328 bbox.addInternalBox(buf->getBoundingBox());
330 mesh->setBoundingBox(bbox);
333 void translateMesh(scene::IMesh *mesh, v3f vec)
338 core::aabbox3d<f32> bbox;
341 u16 mc = mesh->getMeshBufferCount();
342 for(u16 j=0; j<mc; j++)
344 scene::IMeshBuffer *buf = mesh->getMeshBuffer(j);
345 video::S3DVertex *vertices = (video::S3DVertex*)buf->getVertices();
346 u16 vc = buf->getVertexCount();
347 for(u16 i=0; i<vc; i++)
349 vertices[i].Pos += vec;
351 buf->recalculateBoundingBox();
353 // calculate total bounding box
355 bbox = buf->getBoundingBox();
357 bbox.addInternalBox(buf->getBoundingBox());
359 mesh->setBoundingBox(bbox);
362 void setMeshColor(scene::IMesh *mesh, const video::SColor &color)
367 u16 mc = mesh->getMeshBufferCount();
368 for(u16 j=0; j<mc; j++)
370 scene::IMeshBuffer *buf = mesh->getMeshBuffer(j);
371 video::S3DVertex *vertices = (video::S3DVertex*)buf->getVertices();
372 u16 vc = buf->getVertexCount();
373 for(u16 i=0; i<vc; i++)
375 vertices[i].Color = color;
380 void setMeshColorByNormalXYZ(scene::IMesh *mesh,
381 const video::SColor &colorX,
382 const video::SColor &colorY,
383 const video::SColor &colorZ)
388 u16 mc = mesh->getMeshBufferCount();
389 for(u16 j=0; j<mc; j++)
391 scene::IMeshBuffer *buf = mesh->getMeshBuffer(j);
392 video::S3DVertex *vertices = (video::S3DVertex*)buf->getVertices();
393 u16 vc = buf->getVertexCount();
394 for(u16 i=0; i<vc; i++)
396 f32 x = fabs(vertices[i].Normal.X);
397 f32 y = fabs(vertices[i].Normal.Y);
398 f32 z = fabs(vertices[i].Normal.Z);
400 vertices[i].Color = colorX;
402 vertices[i].Color = colorY;
404 vertices[i].Color = colorZ;
410 video::ITexture *generateTextureFromMesh(scene::IMesh *mesh,
411 IrrlichtDevice *device,
412 core::dimension2d<u32> dim,
413 std::string texture_name,
416 core::CMatrix4<f32> camera_projection_matrix,
417 video::SColorf ambient_light,
419 video::SColorf light_color,
422 video::IVideoDriver *driver = device->getVideoDriver();
423 if(driver->queryFeature(video::EVDF_RENDER_TO_TARGET) == false)
425 static bool warned = false;
428 errorstream<<"generateTextureFromMesh(): EVDF_RENDER_TO_TARGET"
429 " not supported."<<std::endl;
435 // Create render target texture
436 video::ITexture *rtt = driver->addRenderTargetTexture(
437 dim, texture_name.c_str(), video::ECF_A8R8G8B8);
440 errorstream<<"generateTextureFromMesh(): addRenderTargetTexture"
441 " returned NULL."<<std::endl;
446 driver->setRenderTarget(rtt, false, true, video::SColor(0,0,0,0));
448 // Get a scene manager
449 scene::ISceneManager *smgr_main = device->getSceneManager();
451 scene::ISceneManager *smgr = smgr_main->createNewSceneManager();
454 scene::IMeshSceneNode* meshnode = smgr->addMeshSceneNode(mesh, NULL, -1, v3f(0,0,0), v3f(0,0,0), v3f(1,1,1), true);
455 meshnode->setMaterialFlag(video::EMF_LIGHTING, true);
456 meshnode->setMaterialFlag(video::EMF_ANTI_ALIASING, true);
457 meshnode->setMaterialFlag(video::EMF_BILINEAR_FILTER, true);
459 scene::ICameraSceneNode* camera = smgr->addCameraSceneNode(0,
460 camera_position, camera_lookat);
461 // second parameter of setProjectionMatrix (isOrthogonal) is ignored
462 camera->setProjectionMatrix(camera_projection_matrix, false);
464 smgr->setAmbientLight(ambient_light);
465 smgr->addLightSceneNode(0, light_position, light_color, light_radius);
468 driver->beginScene(true, true, video::SColor(0,0,0,0));
472 // NOTE: The scene nodes should not be dropped, otherwise
473 // smgr->drop() segfaults
477 // Drop scene manager
480 // Unset render target
481 driver->setRenderTarget(0, false, true, 0);