irrlicht/source/Irrlicht/CCubeSceneNode.cpp

181 lines
4.8 KiB
C++
Raw Normal View History

// Copyright (C) 2002-2007 Nikolaus Gebhardt
// This file is part of the "Irrlicht Engine".
// For conditions of distribution and use, see copyright notice in irrlicht.h
#include "CCubeSceneNode.h"
#include "IVideoDriver.h"
#include "ISceneManager.h"
#include "S3DVertex.h"
#include "os.h"
namespace irr
{
namespace scene
{
//! constructor
CCubeSceneNode::CCubeSceneNode(f32 size, ISceneNode* parent, ISceneManager* mgr, s32 id,
const core::vector3df& position, const core::vector3df& rotation, const core::vector3df& scale)
: ISceneNode(parent, mgr, id, position, rotation, scale), Size(size)
{
#ifdef _DEBUG
setDebugName("CCubeSceneNode");
#endif
u16 u[36] = { 0,2,1, 0,3,2, 1,5,4, 1,2,5, 4,6,7, 4,5,6,
7,3,0, 7,6,3, 9,5,2, 9,8,5, 0,11,10, 0,10,7};
Buffer.Indices.set_used(36);
for (s32 i=0; i<36; ++i)
Buffer.Indices[i] = u[i];
setSize();
}
//! destructor
CCubeSceneNode::~CCubeSceneNode()
{
}
void CCubeSceneNode::setSize()
{
// we are creating the cube mesh here.
// nicer texture mapping sent in by Dr Andros C Bragianos
// .. and then improved by jox.
video::SColor clr(255,255,255,255);
Buffer.Vertices.set_used(12);
Buffer.Vertices[0] = video::S3DVertex(0,0,0, -1,-1,-1, clr, 0, 1);
Buffer.Vertices[1] = video::S3DVertex(1,0,0, 1,-1,-1, clr, 1, 1);
Buffer.Vertices[2] = video::S3DVertex(1,1,0, 1, 1,-1, clr, 1, 0);
Buffer.Vertices[3] = video::S3DVertex(0,1,0, -1, 1,-1, clr, 0, 0);
Buffer.Vertices[4] = video::S3DVertex(1,0,1, 1,-1, 1, clr, 0, 1);
Buffer.Vertices[5] = video::S3DVertex(1,1,1, 1, 1, 1, clr, 0, 0);
Buffer.Vertices[6] = video::S3DVertex(0,1,1, -1, 1, 1, clr, 1, 0);
Buffer.Vertices[7] = video::S3DVertex(0,0,1, -1,-1, 1, clr, 1, 1);
Buffer.Vertices[8] = video::S3DVertex(0,1,1, -1, 1, 1, clr, 0, 1);
Buffer.Vertices[9] = video::S3DVertex(0,1,0, -1, 1,-1, clr, 1, 1);
Buffer.Vertices[10] = video::S3DVertex(1,0,1, 1,-1, 1, clr, 1, 0);
Buffer.Vertices[11] = video::S3DVertex(1,0,0, 1,-1,-1, clr, 0, 0);
Buffer.BoundingBox.reset(0,0,0);
for (int i=0; i<12; ++i)
{
Buffer.Vertices[i].Pos -= core::vector3df(0.5f, 0.5f, 0.5f);
Buffer.Vertices[i].Pos *= Size;
Buffer.BoundingBox.addInternalPoint(Buffer.Vertices[i].Pos);
}
}
//! renders the node.
void CCubeSceneNode::render()
{
/*
011 111
/6--------/5 y
/ | / | ^ z
/ | / | | /
010 3---------2 | |/
| 7- - -| -4 101 *---->x
| / | /
|/ | /
0---------1/
000 100
*/
video::IVideoDriver* driver = SceneManager->getVideoDriver();
driver->setMaterial(Buffer.Material);
driver->setTransform(video::ETS_WORLD, AbsoluteTransformation);
driver->drawMeshBuffer(&Buffer);
}
//! returns the axis aligned bounding box of this node
const core::aabbox3d<f32>& CCubeSceneNode::getBoundingBox() const
{
return Buffer.BoundingBox;
}
void CCubeSceneNode::OnRegisterSceneNode()
{
if (IsVisible)
{
SceneManager->registerNodeForRendering(this);
ISceneNode::OnRegisterSceneNode();
}
}
//! returns the material based on the zero based index i. To get the amount
//! of materials used by this scene node, use getMaterialCount().
//! This function is needed for inserting the node into the scene hirachy on a
//! optimal position for minimizing renderstate changes, but can also be used
//! to directly modify the material of a scene node.
video::SMaterial& CCubeSceneNode::getMaterial(u32 i)
{
return Buffer.Material;
}
//! returns amount of materials used by this scene node.
u32 CCubeSceneNode::getMaterialCount()
{
return 1;
}
//! Writes attributes of the scene node.
void CCubeSceneNode::serializeAttributes(io::IAttributes* out, io::SAttributeReadWriteOptions* options)
{
ISceneNode::serializeAttributes(out, options);
out->addFloat("Size", Size);
}
//! Reads attributes of the scene node.
void CCubeSceneNode::deserializeAttributes(io::IAttributes* in, io::SAttributeReadWriteOptions* options)
{
Size = in->getAttributeAsFloat("Size");
Size = irr::core::max_(Size, 0.0001f);
setSize();
ISceneNode::deserializeAttributes(in, options);
}
//! Creates a clone of this scene node and its children.
ISceneNode* CCubeSceneNode::clone(ISceneNode* newParent, ISceneManager* newManager)
{
if (!newParent) newParent = Parent;
if (!newManager) newManager = SceneManager;
CCubeSceneNode* nb = new CCubeSceneNode(Size, newParent,
newManager, ID, RelativeTranslation);
nb->cloneMembers(this, newManager);
nb->Buffer.Material = Buffer.Material;
nb->drop();
return nb;
}
} // end namespace scene
} // end namespace irr