tibia-client/src/framework/graphics/texturemanager.cpp

156 lines
4.9 KiB
C++
Raw Normal View History

2011-08-28 15:17:58 +02:00
/*
2017-01-13 11:47:07 +01:00
* Copyright (c) 2010-2017 OTClient <https://github.com/edubart/otclient>
2011-08-28 15:17:58 +02:00
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal
* in the Software without restriction, including without limitation the rights
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in
* all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
* THE SOFTWARE.
*/
2011-08-14 04:09:11 +02:00
#include "texturemanager.h"
#include "animatedtexture.h"
2012-03-30 09:39:46 +02:00
#include "graphics.h"
2012-06-08 18:58:08 +02:00
#include "image.h"
2011-08-15 16:06:15 +02:00
#include <framework/core/resourcemanager.h>
#include <framework/core/clock.h>
2013-03-01 09:46:55 +01:00
#include <framework/core/eventdispatcher.h>
#include <framework/graphics/apngloader.h>
2011-08-14 04:09:11 +02:00
TextureManager g_textures;
void TextureManager::init()
{
m_emptyTexture = TexturePtr(new Texture);
}
void TextureManager::terminate()
{
2013-03-01 09:46:55 +01:00
if(m_liveReloadEvent) {
m_liveReloadEvent->cancel();
m_liveReloadEvent = nullptr;
}
m_textures.clear();
m_animatedTextures.clear();
m_emptyTexture = nullptr;
}
void TextureManager::poll()
{
// update only every 16msec, this allows upto 60 fps for animated textures
static ticks_t lastUpdate = 0;
ticks_t now = g_clock.millis();
if(now - lastUpdate < 16)
return;
lastUpdate = now;
for(const AnimatedTexturePtr& animatedTexture : m_animatedTextures)
animatedTexture->updateAnimation();
}
2013-03-01 09:46:55 +01:00
void TextureManager::clearCache()
2012-07-14 23:30:00 +02:00
{
m_animatedTextures.clear();
2012-07-14 23:30:00 +02:00
m_textures.clear();
}
2013-03-01 09:46:55 +01:00
void TextureManager::liveReload()
{
if(m_liveReloadEvent)
return;
m_liveReloadEvent = g_dispatcher.cycleEvent([this] {
for(auto& it : m_textures) {
const std::string& path = g_resources.guessFilePath(it.first, "png");
const TexturePtr& tex = it.second;
if(tex->getTime() >= g_resources.getFileTime(path))
continue;
ImagePtr image = Image::load(path);
if(!image)
continue;
tex->uploadPixels(image, tex->hasMipmaps());
tex->setTime(stdext::time());
}
}, 1000);
}
2012-04-24 18:18:45 +02:00
TexturePtr TextureManager::getTexture(const std::string& fileName)
2011-08-14 04:09:11 +02:00
{
TexturePtr texture;
2012-04-24 18:18:45 +02:00
// before must resolve filename to full path
std::string filePath = g_resources.resolvePath(fileName);
2011-08-14 04:09:11 +02:00
// check if the texture is already loaded
2012-04-24 18:18:45 +02:00
auto it = m_textures.find(filePath);
2011-08-14 04:09:11 +02:00
if(it != m_textures.end()) {
texture = it->second;
2011-08-14 04:09:11 +02:00
}
// texture not found, load it
if(!texture) {
try {
2013-01-28 02:23:53 +01:00
std::string filePathEx = g_resources.guessFilePath(filePath, "png");
2011-08-14 04:09:11 +02:00
// load texture file data
std::stringstream fin;
g_resources.readFileStream(filePathEx, fin);
2013-03-01 09:46:55 +01:00
texture = loadTexture(fin);
} catch(stdext::exception& e) {
g_logger.error(stdext::format("Unable to load texture '%s': %s", fileName, e.what()));
texture = g_textures.getEmptyTexture();
2011-08-14 04:09:11 +02:00
}
2013-01-19 17:44:07 +01:00
if(texture) {
2013-03-01 09:46:55 +01:00
texture->setTime(stdext::time());
2013-01-19 17:44:07 +01:00
texture->setSmooth(true);
m_textures[filePath] = texture;
2013-01-19 17:44:07 +01:00
}
2011-08-14 04:09:11 +02:00
}
return texture;
}
2013-03-01 09:46:55 +01:00
TexturePtr TextureManager::loadTexture(std::stringstream& file)
2011-08-14 04:09:11 +02:00
{
TexturePtr texture;
apng_data apng;
if(load_apng(file, &apng) == 0) {
Size imageSize(apng.width, apng.height);
2011-08-14 04:09:11 +02:00
if(apng.num_frames > 1) { // animated texture
std::vector<ImagePtr> frames;
std::vector<int> framesDelay;
for(uint i=0;i<apng.num_frames;++i) {
uchar *frameData = apng.pdata + ((apng.first_frame+i) * imageSize.area() * apng.bpp);
int frameDelay = apng.frames_delay[i];
framesDelay.push_back(frameDelay);
frames.push_back(ImagePtr(new Image(imageSize, apng.bpp, frameData)));
}
AnimatedTexturePtr animatedTexture = new AnimatedTexture(imageSize, frames, framesDelay);
m_animatedTextures.push_back(animatedTexture);
texture = animatedTexture;
2012-06-08 18:58:08 +02:00
} else {
ImagePtr image = ImagePtr(new Image(imageSize, apng.bpp, apng.pdata));
2012-06-08 18:58:08 +02:00
texture = TexturePtr(new Texture(image));
}
2011-08-14 04:09:11 +02:00
free_apng(&apng);
}
return texture;
}