Here's a sample code that reproduces the behaviour I'm seeing :
Code: Select all
#include <windows.h>
#include <irrlicht.h>
#include "main.h"
using namespace irr;
using namespace core;
using namespace scene;
using namespace video;
using namespace io;
using namespace gui;
#ifdef _IRR_WINDOWS_
#pragma comment(lib, "Irrlicht.lib")
#pragma comment(linker, "/subsystem:windows /ENTRY:mainCRTStartup")
#endif
#define NB_STRESS_TEST (10000)
#define SCREEN_WIDTH (1024)
#define SCREEN_HEIGHT (768)
int main(int argc, const char* argv[])
{
IrrlichtDevice *device = createDevice(video::EDT_DIRECT3D9, dimension2d<u32>(SCREEN_WIDTH, SCREEN_HEIGHT), 32, false, false, false, NULL);
if(device)
{
IVideoDriver* driver = device->getVideoDriver();
ISceneManager* smgr = device->getSceneManager();
// Declare a screensize object using the values from the driver
const core::dimension2d<u32> screenSize = driver->getScreenSize();
// Add a camera node
ICameraSceneNode *cam = device->getSceneManager()->addCameraSceneNode(0, irr::core::vector3df((f32)screenSize.Width / 2, -((f32)(screenSize.Height / 2)), -1), irr::core::vector3df((f32)screenSize.Width / 2, -((f32)(screenSize.Height / 2)), 100));
// Create a projection matrix to translate a 3D view to the exact screen size, and an orthographic field of view of 1000 units.
matrix4 projectionMatrix;
projectionMatrix.buildProjectionMatrixOrthoLH((f32)screenSize.Width, (f32)screenSize.Height, 0.0f, 1000.0f);
// Apply the projection matrix to the camera.
cam->setProjectionMatrix(projectionMatrix, true);
// The world and the view needs to be reset to the identity matrix, in order to remove any matrix transformation for field of view.
driver->setTransform(video::ETS_WORLD, core::matrix4());
driver->setTransform(video::ETS_VIEW, core::matrix4());
// Note that performance is not the issue here.
ISceneNode* testNodes[NB_STRESS_TEST];
for(int iNode = 0; iNode < NB_STRESS_TEST; iNode++)
{
testNodes[iNode] = smgr->addBillboardSceneNode(NULL, core::dimension2df((f32)8, (f32)8));
testNodes[iNode]->setMaterialType(video::EMT_TRANSPARENT_VERTEX_ALPHA);
testNodes[iNode]->setMaterialFlag(video::EMF_LIGHTING, false);
}
// Gap of 25 pixels on the sides as a validation that the billboards are really white.
dimension2du gap = dimension2du(25, 25);
// Size of a colored billboard
dimension2du cbSize = dimension2du(screenSize.Width/2 - gap.Width, screenSize.Height/2 - gap.Height);
// The four semi-transparent billboards.
ISceneNode* transparentBillboards[4];
// Colors of the billboards
SColor tbColors[4] =
{
SColor(150, 255, 0, 0),
SColor(150, 255, 255, 0),
SColor(150, 0, 255, 255),
SColor(150, 0, 0, 255),
};
transparentBillboards[0] = smgr->addBillboardSceneNode(NULL, core::dimension2df((f32)cbSize.Width, (f32)cbSize.Height), core::vector3df(gap.Width + (cbSize.Width / 2), -(f32)gap.Height - (cbSize.Height / 2), 1.0f), -1, tbColors[0], tbColors[0]);
transparentBillboards[1] = smgr->addBillboardSceneNode(NULL, core::dimension2df((f32)cbSize.Width, (f32)cbSize.Height), core::vector3df(screenSize.Width - gap.Width - (cbSize.Width / 2), -(f32)gap.Height - (cbSize.Height / 2), 1.0f), -1, tbColors[1], tbColors[1]);
transparentBillboards[2] = smgr->addBillboardSceneNode(NULL, core::dimension2df((f32)cbSize.Width, (f32)cbSize.Height), core::vector3df(gap.Width + (cbSize.Width / 2), -(f32)screenSize.Height + gap.Height + (cbSize.Height / 2), 1.0f), -1, tbColors[2], tbColors[2]);
transparentBillboards[3] = smgr->addBillboardSceneNode(NULL, core::dimension2df((f32)cbSize.Width, (f32)cbSize.Height), core::vector3df(screenSize.Width - gap.Width - (cbSize.Width / 2), -(f32)screenSize.Height + gap.Height + (cbSize.Height / 2), 1.0f), -1, tbColors[3], tbColors[3]);
for(int iNode = 0; iNode < 4; iNode++)
{
transparentBillboards[iNode]->setMaterialType(video::EMT_TRANSPARENT_VERTEX_ALPHA);
transparentBillboards [iNode]->setMaterialFlag(video::EMF_LIGHTING, false);
}
u32 currentTime = device->getTimer()->getTime();
while(device->run())
{
// Poor man's way of randomizing the white billboards position at each 200 ms
if(device->getTimer()->getTime() > currentTime + 200)
{
for(int i = 0; i < NB_STRESS_TEST; i++)
{
f32 X = (rand() % (screenSize.Width - 4));
f32 Y = 0.0f - (rand() % (screenSize.Height - 4));
f32 Z = 50.0f;
testNodes[i]->setPosition(core::vector3df(X, Y, Z));
}
currentTime = device->getTimer()->getTime();
}
// The rendering
driver->beginScene(true, true, SColor(255, 0, 0, 0));
smgr->drawAll();
driver->endScene();
}
}
device->drop();
return 0;
}
Please note that this code is not the prettiest I ever wrote, nor the fastest. It was written purely as a demonstration of my perceived problem.
All white billboards have their Z-position set to 50.0f and the four semi-transparent billboards have a Z-position of 1.0f. There's 10,000 white billboards because it's easier to see the problem (i.e. the round area where the billboards are rendered on top), but the problem's still present with fewer billboards.
Here's the result I'm getting :
What I'm expecting to see is the four semi-transparents billboards in front of all the small white billboards. Here's a top-down view of my comprehension on how I set up the camera/world :
Please disregard my "mad skills" with MSPaint.
I'm sure there's something I'm not understanding correctly. From what CuteAlien said, I think it might be the camera's position? Don't hesitate to set me straight, as I'm sure I made a wrong assumptions somewhere, and I'm really looking to learn the ropes of 3D development (since I come from 2D gaming development).