2010-08-02 10:58:00 +08:00
|
|
|
/****************************************************************************
|
|
|
|
Copyright (c) 2010 cocos2d-x.org
|
|
|
|
|
|
|
|
http://www.cocos2d-x.org
|
|
|
|
|
|
|
|
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.
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
#include <stack>
|
|
|
|
#include <string>
|
2010-07-26 17:32:23 +08:00
|
|
|
#include <cctype>
|
2010-07-16 16:28:11 +08:00
|
|
|
#include "CCTextureCache.h"
|
|
|
|
#include "CCTexture2D.h"
|
|
|
|
#include "ccMacros.h"
|
2010-08-04 16:45:00 +08:00
|
|
|
#include "NSData.h"
|
2010-07-30 16:35:07 +08:00
|
|
|
#include "CCDirector.h"
|
2010-08-25 14:14:31 +08:00
|
|
|
#include "platform/platform.h"
|
2010-08-25 17:21:38 +08:00
|
|
|
#include "CCXFileUtils.h"
|
|
|
|
#include "CCXUIImage.h"
|
2010-07-16 14:15:06 +08:00
|
|
|
|
|
|
|
/// @todo EAGLContext static EAGLContext *auxEAGLcontext = NULL;
|
2010-08-04 15:46:12 +08:00
|
|
|
namespace cocos2d {
|
2010-07-16 14:15:06 +08:00
|
|
|
|
2010-07-16 16:28:11 +08:00
|
|
|
class CCAsyncObject : NSObject
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
fpAsyncCallback m_pfnCallback;
|
|
|
|
NSObject* m_pTarget;
|
2010-07-20 13:49:13 +08:00
|
|
|
std::string * m_pData;
|
2010-07-16 16:28:11 +08:00
|
|
|
public:
|
|
|
|
CCAsyncObject();
|
|
|
|
~CCAsyncObject()
|
|
|
|
{
|
|
|
|
CCLOGINFO("cocos2d: deallocing CCAsyncObject.");
|
|
|
|
CCX_SAFE_DELETE(m_pTarget);
|
|
|
|
CCX_SAFE_DELETE(m_pData);
|
|
|
|
}
|
|
|
|
};
|
2010-07-16 14:15:06 +08:00
|
|
|
|
2010-07-16 16:28:11 +08:00
|
|
|
// implementation CCTextureCache
|
2010-07-16 14:15:06 +08:00
|
|
|
|
2010-07-16 16:28:11 +08:00
|
|
|
// TextureCache - Alloc, Init & Dealloc
|
|
|
|
static CCTextureCache *g_sharedTextureCache;
|
2010-07-16 14:15:06 +08:00
|
|
|
|
2010-07-16 16:28:11 +08:00
|
|
|
CCTextureCache * CCTextureCache::sharedTextureCache()
|
2010-07-16 14:15:06 +08:00
|
|
|
{
|
2010-07-16 16:28:11 +08:00
|
|
|
if (!g_sharedTextureCache)
|
|
|
|
g_sharedTextureCache = new CCTextureCache();
|
2010-07-16 14:15:06 +08:00
|
|
|
|
2010-07-16 16:28:11 +08:00
|
|
|
return g_sharedTextureCache;
|
2010-07-16 14:15:06 +08:00
|
|
|
}
|
|
|
|
|
2010-07-16 16:28:11 +08:00
|
|
|
CCTextureCache::CCTextureCache()
|
2010-07-16 14:15:06 +08:00
|
|
|
{
|
2010-08-03 12:07:32 +08:00
|
|
|
NSAssert(g_sharedTextureCache == NULL, "Attempted to allocate a second instance of a singleton.");
|
2010-07-21 11:13:32 +08:00
|
|
|
|
|
|
|
m_pTextures = new NSMutableDictionary<std::string, CCTexture2D*>();
|
|
|
|
m_pTextures->retain();
|
2010-07-26 17:32:23 +08:00
|
|
|
m_pDictLock = new NSLock();
|
|
|
|
m_pContextLock = new NSLock();
|
2010-07-16 14:15:06 +08:00
|
|
|
}
|
|
|
|
|
2010-07-16 16:28:11 +08:00
|
|
|
CCTextureCache::~CCTextureCache()
|
2010-07-16 14:15:06 +08:00
|
|
|
{
|
2010-07-16 16:28:11 +08:00
|
|
|
CCLOG("cocos2d: deallocing CCTextureCache.");
|
|
|
|
/// @todo release
|
2010-08-02 10:58:00 +08:00
|
|
|
// [textures release];
|
|
|
|
// [dictLock release];
|
|
|
|
// [contextLock release];
|
|
|
|
// [auxEAGLcontext release];
|
|
|
|
// auxEAGLcontext = nil;
|
|
|
|
// sharedTextureCache = nil;
|
|
|
|
// [super dealloc];
|
2010-07-16 14:15:06 +08:00
|
|
|
}
|
|
|
|
|
2010-07-16 16:28:11 +08:00
|
|
|
void CCTextureCache::purgeSharedTextureCache()
|
2010-07-16 14:15:06 +08:00
|
|
|
{
|
2010-07-16 16:28:11 +08:00
|
|
|
CCX_SAFE_RELEASE(g_sharedTextureCache);
|
2010-07-16 14:15:06 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-08-06 16:05:19 +08:00
|
|
|
char * CCTextureCache::description()
|
2010-07-16 14:15:06 +08:00
|
|
|
{
|
2010-08-06 16:05:19 +08:00
|
|
|
char *ret = new char[100];
|
|
|
|
sprintf_s(ret, 100, "<CCTextureCache | Number of textures = %u>", m_pTextures->count());
|
2010-07-21 11:13:32 +08:00
|
|
|
return ret;
|
2010-07-16 14:15:06 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-07-16 16:28:11 +08:00
|
|
|
// TextureCache - Add Images
|
|
|
|
void CCTextureCache::addImageWithAsyncObject(CCAsyncObject* async)
|
2010-07-16 14:15:06 +08:00
|
|
|
{
|
2010-07-21 17:40:10 +08:00
|
|
|
/** @todo EAGLContext
|
2010-07-16 14:15:06 +08:00
|
|
|
NSAutoreleasePool *autoreleasepool = [[NSAutoreleasePool alloc] init];
|
|
|
|
|
|
|
|
// textures will be created on the main OpenGL context
|
|
|
|
// it seems that in SDK 2.2.x there can't be 2 threads creating textures at the same time
|
|
|
|
// the lock is used for this purpose: issue #472
|
|
|
|
[contextLock lock];
|
|
|
|
if( auxEAGLcontext == nil ) {
|
|
|
|
auxEAGLcontext = [[EAGLContext alloc]
|
|
|
|
initWithAPI:kEAGLRenderingAPIOpenGLES1
|
|
|
|
sharegroup:[[[[CCDirector sharedDirector] openGLView] context] sharegroup]];
|
|
|
|
|
|
|
|
if( ! auxEAGLcontext )
|
|
|
|
CCLOG(@"cocos2d: TextureCache: Could not create EAGL context");
|
|
|
|
}
|
|
|
|
|
|
|
|
if( [EAGLContext setCurrentContext:auxEAGLcontext] ) {
|
|
|
|
|
|
|
|
// load / create the texture
|
|
|
|
CCTexture2D *tex = [self addImage:async.data];
|
|
|
|
|
|
|
|
// The callback will be executed on the main thread
|
|
|
|
[async.target performSelectorOnMainThread:async.selector withObject:tex waitUntilDone:NO];
|
|
|
|
|
|
|
|
[EAGLContext setCurrentContext:nil];
|
|
|
|
} else {
|
|
|
|
CCLOG(@"cocos2d: TetureCache: EAGLContext error");
|
|
|
|
}
|
|
|
|
[contextLock unlock];
|
|
|
|
|
2010-07-16 16:28:11 +08:00
|
|
|
[autoreleasepool release];*/
|
2010-07-16 14:15:06 +08:00
|
|
|
}
|
|
|
|
|
2010-07-23 14:23:53 +08:00
|
|
|
/** @todo selector, NSThread*/
|
2010-07-22 11:15:25 +08:00
|
|
|
void CCTextureCache::addImageAsync(const char* filename, NSObject *target, fpAsyncCallback func)
|
2010-07-16 14:15:06 +08:00
|
|
|
{
|
2010-07-22 11:15:25 +08:00
|
|
|
NSAssert(filename != NULL , "TextureCache: fileimage MUST not be nill");
|
2010-07-16 14:15:06 +08:00
|
|
|
|
|
|
|
// optimization
|
|
|
|
|
2010-08-06 14:32:32 +08:00
|
|
|
// CCTexture2D * tex;
|
2010-08-02 10:58:00 +08:00
|
|
|
//
|
|
|
|
// if ( (tex = m_pTextures->objectForKey(filename)) )
|
|
|
|
|
|
|
|
// {
|
|
|
|
|
|
|
|
// target->
|
|
|
|
|
|
|
|
// }
|
|
|
|
|
|
|
|
//
|
|
|
|
// if( (tex=[textures objectForKey: filename] ) ) {
|
|
|
|
// [target performSelector:selector withObject:tex];
|
|
|
|
// return;
|
|
|
|
// }
|
|
|
|
//
|
|
|
|
// // schedule the load
|
|
|
|
//
|
|
|
|
// CCAsyncObject *asyncObject = [[CCAsyncObject alloc] init];
|
|
|
|
// asyncObject.selector = selector;
|
|
|
|
// asyncObject.target = target;
|
|
|
|
// asyncObject.data = filename;
|
|
|
|
//
|
|
|
|
// [NSThread detachNewThreadSelector:@selector(addImageWithAsyncObject:) toTarget:self withObject:asyncObject];
|
|
|
|
// [asyncObject release];
|
2010-07-21 11:13:32 +08:00
|
|
|
}
|
2010-07-16 14:15:06 +08:00
|
|
|
|
2010-07-22 11:15:25 +08:00
|
|
|
CCTexture2D * CCTextureCache::addImage(const char * path)
|
2010-07-16 14:15:06 +08:00
|
|
|
{
|
2010-07-26 17:32:23 +08:00
|
|
|
NSAssert(path != NULL, "TextureCache: fileimage MUST not be NULL");
|
2010-07-16 14:15:06 +08:00
|
|
|
|
2010-08-06 14:32:32 +08:00
|
|
|
CCTexture2D * texture = NULL;
|
2010-07-26 17:32:23 +08:00
|
|
|
std::string temp(path);
|
2010-07-16 14:15:06 +08:00
|
|
|
|
|
|
|
// MUTEX:
|
|
|
|
// Needed since addImageAsync calls this method from a different thread
|
2010-07-26 17:32:23 +08:00
|
|
|
|
|
|
|
m_pDictLock->lock();
|
2010-07-16 14:15:06 +08:00
|
|
|
|
2010-08-06 14:32:32 +08:00
|
|
|
texture = m_pTextures->objectForKey(temp);
|
2010-07-16 14:15:06 +08:00
|
|
|
|
2010-08-06 14:32:32 +08:00
|
|
|
if( ! texture ) {
|
2010-07-16 14:15:06 +08:00
|
|
|
|
|
|
|
// Split up directory and filename
|
2010-07-26 17:32:23 +08:00
|
|
|
std::string fullpath(CCFileUtils::fullPathFromRelativePath(path));
|
2010-07-16 14:15:06 +08:00
|
|
|
|
|
|
|
// all images are handled by UIImage except PVR extension that is handled by our own handler
|
2010-07-26 17:32:23 +08:00
|
|
|
// if ( [[path lowercaseString] hasSuffix:@".pvr"] )
|
2010-08-02 10:58:00 +08:00
|
|
|
for (unsigned int i = 0; i < temp.length(); ++i)
|
2010-07-26 17:32:23 +08:00
|
|
|
temp[i] = tolower(temp[i]);
|
2010-08-06 14:32:32 +08:00
|
|
|
if (std::string::npos != temp.find(".pvr"))
|
2010-08-02 10:58:00 +08:00
|
|
|
{
|
2010-08-06 14:32:32 +08:00
|
|
|
texture = this->addPVRTCImage(fullpath.c_str());
|
2010-08-02 10:58:00 +08:00
|
|
|
}
|
2010-07-26 17:32:23 +08:00
|
|
|
else
|
|
|
|
{
|
2010-07-16 14:15:06 +08:00
|
|
|
// prevents overloading the autorelease pool
|
2010-07-26 17:32:23 +08:00
|
|
|
UIImage * image = new UIImage();
|
2010-07-30 18:21:00 +08:00
|
|
|
NSAssert(image->initWithContentsOfFile(fullpath), "")
|
2010-08-02 17:38:42 +08:00
|
|
|
CCLOG("cocos2d: Initialize image file %s error!",fullpath);
|
2010-08-06 14:32:32 +08:00
|
|
|
texture = new CCTexture2D();
|
|
|
|
texture->initWithImage(image);
|
2010-07-26 17:32:23 +08:00
|
|
|
CCX_SAFE_DELETE(image);// image->release();
|
2010-07-16 14:15:06 +08:00
|
|
|
|
2010-08-06 14:32:32 +08:00
|
|
|
if( texture )
|
|
|
|
m_pTextures->setObject(texture, path);
|
2010-07-16 14:15:06 +08:00
|
|
|
else
|
2010-07-26 17:32:23 +08:00
|
|
|
CCLOG("cocos2d: Couldn't add image:%s in CCTextureCache", path);
|
2010-07-16 14:15:06 +08:00
|
|
|
|
2010-08-06 14:32:32 +08:00
|
|
|
texture->release();
|
2010-07-16 14:15:06 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-08-04 15:46:12 +08:00
|
|
|
m_pDictLock->unlock();
|
2010-07-26 17:32:23 +08:00
|
|
|
|
2010-08-06 14:32:32 +08:00
|
|
|
return texture;
|
2010-07-16 14:15:06 +08:00
|
|
|
}
|
|
|
|
|
2010-07-22 11:15:25 +08:00
|
|
|
CCTexture2D* CCTextureCache::addPVRTCImage(const char* path, int bpp, bool hasAlpha, int width)
|
2010-07-16 14:15:06 +08:00
|
|
|
{
|
2010-07-16 16:28:11 +08:00
|
|
|
|
2010-07-22 11:15:25 +08:00
|
|
|
NSAssert(path != NULL, "TextureCache: fileimage MUST not be nill");
|
2010-07-16 16:28:11 +08:00
|
|
|
NSAssert( bpp==2 || bpp==4, "TextureCache: bpp must be either 2 or 4");
|
2010-07-16 14:15:06 +08:00
|
|
|
|
2010-08-06 14:32:32 +08:00
|
|
|
CCTexture2D * texture;
|
2010-07-26 17:32:23 +08:00
|
|
|
std::string temp(path);
|
2010-08-06 14:32:32 +08:00
|
|
|
if ( (texture = m_pTextures->objectForKey(temp)) )
|
2010-08-02 10:58:00 +08:00
|
|
|
{
|
2010-08-06 14:32:32 +08:00
|
|
|
return texture;
|
2010-08-02 10:58:00 +08:00
|
|
|
}
|
2010-07-26 17:32:23 +08:00
|
|
|
|
2010-07-16 14:15:06 +08:00
|
|
|
// Split up directory and filename
|
2010-07-26 17:32:23 +08:00
|
|
|
std::string fullpath( CCFileUtils::fullPathFromRelativePath(path) );
|
2010-07-16 14:15:06 +08:00
|
|
|
|
2010-07-28 13:48:13 +08:00
|
|
|
NSData * data = NSData::dataWithContentsOfFile(fullpath);
|
2010-08-06 14:32:32 +08:00
|
|
|
texture = new CCTexture2D();
|
|
|
|
texture->initWithPVRTCData(data->bytes(), 0, bpp, hasAlpha, width);
|
|
|
|
if( texture )
|
|
|
|
m_pTextures->setObject(texture, temp);
|
2010-07-16 14:15:06 +08:00
|
|
|
else
|
2010-07-26 17:32:23 +08:00
|
|
|
CCLOG("cocos2d: Couldn't add PVRTCImage:%s in CCTextureCache",path);
|
|
|
|
|
|
|
|
CCX_SAFE_DELETE(data);
|
2010-07-16 14:15:06 +08:00
|
|
|
|
2010-08-06 14:32:32 +08:00
|
|
|
texture->autorelease();
|
|
|
|
return texture;
|
2010-07-16 14:15:06 +08:00
|
|
|
}
|
|
|
|
|
2010-07-22 11:15:25 +08:00
|
|
|
CCTexture2D * CCTextureCache::addPVRTCImage(const char* fileimage)
|
2010-07-16 14:15:06 +08:00
|
|
|
{
|
2010-07-22 11:15:25 +08:00
|
|
|
NSAssert(fileimage != NULL, "TextureCache: fileimage MUST not be nill");
|
2010-07-16 14:15:06 +08:00
|
|
|
|
2010-08-06 14:32:32 +08:00
|
|
|
CCTexture2D * texture;
|
2010-07-22 11:20:52 +08:00
|
|
|
std::string key(fileimage);
|
2010-08-06 14:32:32 +08:00
|
|
|
if( (texture = m_pTextures->objectForKey(key)) )
|
2010-07-21 11:13:32 +08:00
|
|
|
{
|
2010-08-06 14:32:32 +08:00
|
|
|
return texture;
|
2010-07-16 14:15:06 +08:00
|
|
|
}
|
|
|
|
|
2010-08-06 14:32:32 +08:00
|
|
|
texture = new CCTexture2D();
|
|
|
|
texture = texture->initWithPVRTCFile(fileimage);
|
|
|
|
if( texture )
|
|
|
|
m_pTextures-> setObject( texture, key);
|
2010-07-16 14:15:06 +08:00
|
|
|
else
|
2010-07-21 11:13:32 +08:00
|
|
|
CCLOG("cocos2d: Couldn't add PVRTCImage:%s in CCTextureCache",fileimage);
|
|
|
|
|
2010-08-06 14:32:32 +08:00
|
|
|
texture->autorelease();
|
|
|
|
return texture;
|
2010-07-16 14:15:06 +08:00
|
|
|
}
|
|
|
|
|
2010-07-21 17:40:10 +08:00
|
|
|
/** @todo UIImage
|
2010-07-20 13:49:13 +08:00
|
|
|
-(CCTexture2D*) addCGImage: (CGImageRef) imageref forKey: (string & )key
|
2010-07-16 14:15:06 +08:00
|
|
|
{
|
|
|
|
NSAssert(imageref != nil, @"TextureCache: image MUST not be nill");
|
|
|
|
|
|
|
|
CCTexture2D * tex = nil;
|
|
|
|
|
|
|
|
// If key is nil, then create a new texture each time
|
|
|
|
if( key && (tex=[textures objectForKey: key] ) ) {
|
|
|
|
return tex;
|
|
|
|
}
|
|
|
|
|
|
|
|
// prevents overloading the autorelease pool
|
|
|
|
UIImage *image = [[UIImage alloc] initWithCGImage:imageref];
|
|
|
|
tex = [[CCTexture2D alloc] initWithImage: image];
|
|
|
|
[image release];
|
|
|
|
|
|
|
|
if(tex && key)
|
|
|
|
[textures setObject: tex forKey:key];
|
|
|
|
else
|
|
|
|
CCLOG(@"cocos2d: Couldn't add CGImage in CCTextureCache");
|
|
|
|
|
|
|
|
return [tex autorelease];
|
2010-07-16 16:28:11 +08:00
|
|
|
}*/
|
2010-07-16 14:15:06 +08:00
|
|
|
|
2010-07-16 16:28:11 +08:00
|
|
|
// TextureCache - Remove
|
2010-07-16 14:15:06 +08:00
|
|
|
|
2010-07-16 16:28:11 +08:00
|
|
|
void CCTextureCache::removeAllTextures()
|
2010-07-16 14:15:06 +08:00
|
|
|
{
|
2010-07-21 17:40:10 +08:00
|
|
|
m_pTextures->removeAllObjects();
|
2010-07-16 14:15:06 +08:00
|
|
|
}
|
|
|
|
|
2010-07-16 16:28:11 +08:00
|
|
|
void CCTextureCache::removeUnusedTextures()
|
2010-07-16 14:15:06 +08:00
|
|
|
{
|
2010-07-30 16:35:07 +08:00
|
|
|
std::vector<std::string> keys = m_pTextures->allKeys();
|
|
|
|
std::vector<std::string>::iterator it;
|
2010-08-02 10:58:00 +08:00
|
|
|
for (it = keys.begin(); it <= keys.end(); it++)
|
|
|
|
{
|
|
|
|
CCTexture2D *value = m_pTextures->objectForKey(*it);
|
|
|
|
if (value->retainCount() == 1)
|
|
|
|
{
|
|
|
|
CCLOG("cocos2d: CCTextureCache: removing unused texture: %s", (*it).c_str());
|
|
|
|
m_pTextures->removeObjectForKey(*it);
|
|
|
|
}
|
|
|
|
}
|
2010-07-16 14:15:06 +08:00
|
|
|
}
|
|
|
|
|
2010-08-06 14:32:32 +08:00
|
|
|
void CCTextureCache::removeTexture(CCTexture2D* texture)
|
2010-07-16 14:15:06 +08:00
|
|
|
{
|
2010-08-06 14:32:32 +08:00
|
|
|
if( ! texture )
|
2010-07-16 14:15:06 +08:00
|
|
|
return;
|
|
|
|
|
2010-08-06 14:32:32 +08:00
|
|
|
std::vector<std::string> keys = m_pTextures->allKeysForObject(texture);
|
2010-07-16 14:15:06 +08:00
|
|
|
|
2010-08-02 10:58:00 +08:00
|
|
|
for (unsigned int i = 0; i < keys.size(); i++)
|
|
|
|
{
|
|
|
|
m_pTextures->removeObjectForKey(keys[i]);
|
|
|
|
}
|
2010-07-16 14:15:06 +08:00
|
|
|
}
|
|
|
|
|
2010-07-22 11:15:25 +08:00
|
|
|
void CCTextureCache::removeTextureForKey(const std::string & textureKeyName)
|
2010-07-16 14:15:06 +08:00
|
|
|
{
|
2010-07-20 13:49:13 +08:00
|
|
|
if( textureKeyName.empty() )
|
2010-07-16 14:15:06 +08:00
|
|
|
return;
|
|
|
|
|
2010-07-21 17:40:10 +08:00
|
|
|
m_pTextures->removeObjectForKey(textureKeyName);
|
2010-07-16 14:15:06 +08:00
|
|
|
}
|
|
|
|
|
2010-08-04 15:46:12 +08:00
|
|
|
}//namespace cocos2d
|
2010-07-16 16:28:11 +08:00
|
|
|
|