minetest/src/irrlichtwrapper.cpp

220 lines
5.4 KiB
C++
Raw Normal View History

2010-12-20 22:23:24 +02:00
#include "irrlichtwrapper.h"
2010-12-25 16:04:51 +02:00
#include "constants.h"
2010-12-20 22:23:24 +02:00
IrrlichtWrapper::IrrlichtWrapper(IrrlichtDevice *device)
{
m_main_thread = get_current_thread_id();
m_device_mutex.Init();
m_device = device;
}
void IrrlichtWrapper::Run()
{
/*
Fetch textures
*/
if(m_get_texture_queue.size() > 0)
{
GetRequest<TextureSpec, video::ITexture*, u8, u8>
request = m_get_texture_queue.pop();
2010-12-21 02:25:47 +02:00
dstream<<"got texture request with key.name="
<<request.key.name<<std::endl;
2010-12-20 22:23:24 +02:00
GetResult<TextureSpec, video::ITexture*, u8, u8>
result;
result.key = request.key;
result.callers = request.callers;
result.item = getTextureDirect(request.key);
request.dest->push_back(result);
}
}
video::ITexture* IrrlichtWrapper::getTexture(TextureSpec spec)
{
video::ITexture *t = m_texturecache.get(spec.name);
if(t != NULL)
return t;
if(get_current_thread_id() == m_main_thread)
{
2010-12-21 02:25:47 +02:00
dstream<<"Getting texture directly: name="
<<spec.name<<std::endl;
2010-12-20 22:23:24 +02:00
t = getTextureDirect(spec);
}
else
{
// We're gonna ask the result to be put into here
ResultQueue<TextureSpec, video::ITexture*, u8, u8> result_queue;
// Throw a request in
m_get_texture_queue.add(spec, 0, 0, &result_queue);
dstream<<"Waiting for texture from main thread: "
<<spec.name<<std::endl;
try
{
// Wait result for a second
GetResult<TextureSpec, video::ITexture*, u8, u8>
result = result_queue.pop_front(1000);
2010-12-20 22:23:24 +02:00
// Check that at least something worked OK
assert(result.key.name == spec.name);
2010-12-20 22:23:24 +02:00
t = result.item;
}
catch(ItemNotFoundException &e)
{
dstream<<"Waiting for texture timed out."<<std::endl;
t = NULL;
}
2010-12-20 22:23:24 +02:00
}
// Add to cache and return
m_texturecache.set(spec.name, t);
return t;
}
video::ITexture* IrrlichtWrapper::getTexture(const std::string &path)
{
return getTexture(TextureSpec(path, path, NULL));
}
/*
Non-thread-safe functions
*/
video::ITexture* IrrlichtWrapper::getTextureDirect(TextureSpec spec)
{
video::IVideoDriver* driver = m_device->getVideoDriver();
2010-12-21 02:25:47 +02:00
if(spec.mod == NULL)
2010-12-20 22:23:24 +02:00
{
2010-12-21 02:25:47 +02:00
dstream<<"IrrlichtWrapper::getTextureDirect: Loading texture "
<<spec.path<<std::endl;
return driver->getTexture(spec.path.c_str());
2010-12-20 22:23:24 +02:00
}
2010-12-21 02:25:47 +02:00
dstream<<"IrrlichtWrapper::getTextureDirect: Loading and modifying "
"texture "<<spec.path<<" to make "<<spec.name<<std::endl;
video::ITexture *base = driver->getTexture(spec.path.c_str());
video::ITexture *result = spec.mod->make(base, spec.name.c_str(), driver);
delete spec.mod;
return result;
2010-12-20 22:23:24 +02:00
}
video::ITexture * CrackTextureMod::make(video::ITexture *original,
2010-12-21 02:25:47 +02:00
const char *newname, video::IVideoDriver* driver)
2010-12-20 22:23:24 +02:00
{
// Size of the base image
2010-12-21 02:25:47 +02:00
core::dimension2d<u32> dim(16, 16);
// Size of the crack image
//core::dimension2d<u32> dim_crack(16, 16 * CRACK_ANIMATION_LENGTH);
// Position to copy the crack to in the base image
2010-12-21 02:25:47 +02:00
core::position2d<s32> pos_base(0, 0);
// Position to copy the crack from in the crack image
2010-12-21 02:25:47 +02:00
core::position2d<s32> pos_other(0, 16 * progression);
video::IImage *baseimage = driver->createImage(original, pos_base, dim);
assert(baseimage);
video::IImage *crackimage = driver->createImageFromFile("../data/crack.png");
assert(crackimage);
2010-12-21 02:25:47 +02:00
#if 0
2010-12-21 02:25:47 +02:00
video::ITexture *other = driver->getTexture("../data/crack.png");
2010-12-25 23:23:53 +02:00
dstream<<__FUNCTION_NAME<<": crack texture size is "
<<other->getSize().Width<<"x"
<<other->getSize().Height<<std::endl;
2010-12-25 16:04:51 +02:00
2010-12-21 02:25:47 +02:00
// We have to get the whole texture because getting a smaller area
// messes the whole thing. It is probably a bug in Irrlicht.
2010-12-25 16:04:51 +02:00
// NOTE: This doesn't work probably because some systems scale
// the image to fit a texture or something...
2010-12-21 02:25:47 +02:00
video::IImage *otherimage = driver->createImage(
2010-12-25 23:23:53 +02:00
other, core::position2d<s32>(0,0), other->getSize());
2010-12-21 02:25:47 +02:00
assert(otherimage);
// Now, the image might be 80 or 128 high depending on the computer
// Let's make an image of the right size and copy the possibly
// wrong sized one with scaling
// NOTE: This is an ugly hack.
video::IImage *crackimage = driver->createImage(
baseimage->getColorFormat(), dim_crack);
2010-12-21 02:25:47 +02:00
assert(crackimage);
2010-12-21 02:25:47 +02:00
otherimage->copyToScaling(crackimage);
otherimage->drop();
#endif
// Then copy the right part of crackimage to baseimage
crackimage->copyToWithAlpha(baseimage, v2s32(0,0),
2010-12-21 02:25:47 +02:00
core::rect<s32>(pos_other, dim),
video::SColor(255,255,255,255),
NULL);
crackimage->drop();
// Create texture from resulting image
2010-12-21 02:25:47 +02:00
video::ITexture *newtexture = driver->addTexture(newname, baseimage);
baseimage->drop();
return newtexture;
2010-12-20 22:23:24 +02:00
}
2010-12-24 11:44:26 +02:00
video::ITexture * ProgressBarTextureMod::make(video::ITexture *original,
const char *newname, video::IVideoDriver* driver)
2010-12-20 22:23:24 +02:00
{
2010-12-24 11:44:26 +02:00
core::position2d<s32> pos_base(0, 0);
core::dimension2d<u32> dim = original->getOriginalSize();
2010-12-20 22:23:24 +02:00
2010-12-24 11:44:26 +02:00
video::IImage *baseimage = driver->createImage(original, pos_base, dim);
2010-12-20 22:23:24 +02:00
assert(baseimage);
2010-12-24 11:44:26 +02:00
core::dimension2d<u32> size = baseimage->getDimension();
2010-12-20 22:23:24 +02:00
2010-12-24 11:44:26 +02:00
u32 barheight = 1;
u32 barpad_x = 1;
u32 barpad_y = 1;
u32 barwidth = size.Width - barpad_x*2;
v2u32 barpos(barpad_x, size.Height - barheight - barpad_y);
2010-12-25 02:33:41 +02:00
u32 barvalue_i = (u32)(((float)barwidth * value) + 0.5);
2010-12-24 11:44:26 +02:00
video::SColor active(255,255,0,0);
video::SColor inactive(255,0,0,0);
for(u32 x0=0; x0<barwidth; x0++)
{
video::SColor *c;
if(x0 < barvalue_i)
c = &active;
else
c = &inactive;
u32 x = x0 + barpos.X;
for(u32 y=barpos.Y; y<barpos.Y+barheight; y++)
{
baseimage->setPixel(x,y, *c);
}
}
video::ITexture *newtexture = driver->addTexture(newname, baseimage);
2010-12-20 22:23:24 +02:00
baseimage->drop();
return newtexture;
}