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.
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
|
2010-07-15 18:15:00 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Support for RGBA_4_4_4_4 and RGBA_5_5_5_1 was copied from:
|
|
|
|
* https://devforums.apple.com/message/37855#37855 by a1studmuffin
|
|
|
|
*/
|
|
|
|
|
2010-08-02 10:58:00 +08:00
|
|
|
#include "CCTexture2D.h"
|
2010-07-15 18:15:00 +08:00
|
|
|
#include <GLES/glext.h>
|
|
|
|
|
|
|
|
#include "ccConfig.h"
|
|
|
|
#include "ccMacros.h"
|
|
|
|
#include "CCTexture2D.h"
|
2010-07-20 14:29:18 +08:00
|
|
|
#include "CCConfiguration.h"
|
2010-07-27 13:47:29 +08:00
|
|
|
#include "platform/platform.h"
|
2010-08-25 17:21:38 +08:00
|
|
|
#include "CCXUIImage.h"
|
2010-07-15 18:15:00 +08:00
|
|
|
|
2010-08-31 16:53:25 +08:00
|
|
|
#ifdef _POWERVR_SUPPORT_
|
|
|
|
#include "CCPVRTexture.h"
|
|
|
|
#endif
|
|
|
|
|
2010-08-04 15:46:12 +08:00
|
|
|
namespace cocos2d {
|
2010-07-15 18:15:00 +08:00
|
|
|
|
|
|
|
#if CC_FONT_LABEL_SUPPORT
|
|
|
|
// FontLabel support
|
|
|
|
#endif// CC_FONT_LABEL_SUPPORT
|
|
|
|
|
|
|
|
|
|
|
|
static unsigned int nextPOT(unsigned int x)
|
|
|
|
{
|
|
|
|
x = x - 1;
|
|
|
|
x = x | (x >> 1);
|
|
|
|
x = x | (x >> 2);
|
|
|
|
x = x | (x >> 4);
|
|
|
|
x = x | (x >> 8);
|
|
|
|
x = x | (x >>16);
|
|
|
|
return x + 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
//CLASS IMPLEMENTATIONS:
|
|
|
|
|
|
|
|
|
|
|
|
// If the image has alpha, you can create RGBA8 (32-bit) or RGBA4 (16-bit) or RGB5A1 (16-bit)
|
|
|
|
// Default is: RGBA8888 (32-bit textures)
|
|
|
|
static CCTexture2DPixelFormat g_defaultAlphaPixelFormat = kCCTexture2DPixelFormat_Default;
|
|
|
|
|
|
|
|
CCTexture2D::CCTexture2D()
|
|
|
|
{
|
2010-08-12 17:01:51 +08:00
|
|
|
m_uName = 0;
|
2010-07-15 18:15:00 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
CCTexture2D::~CCTexture2D()
|
|
|
|
{
|
|
|
|
CCLOGINFO("cocos2d: deallocing CCTexture2D %u.", m_uName);
|
|
|
|
if(m_uName)
|
|
|
|
glDeleteTextures(1, &m_uName);
|
|
|
|
}
|
|
|
|
|
|
|
|
CCTexture2DPixelFormat CCTexture2D::getPixelFormat()
|
|
|
|
{
|
|
|
|
return m_ePixelFormat;
|
|
|
|
}
|
|
|
|
|
2010-08-02 10:58:00 +08:00
|
|
|
unsigned int CCTexture2D::getPixelsWide()
|
2010-07-15 18:15:00 +08:00
|
|
|
{
|
|
|
|
return m_uPixelsWide;
|
|
|
|
}
|
|
|
|
|
2010-08-02 10:58:00 +08:00
|
|
|
unsigned int CCTexture2D::getPixelsHigh()
|
2010-07-15 18:15:00 +08:00
|
|
|
{
|
|
|
|
return m_uPixelsHigh;
|
|
|
|
}
|
|
|
|
|
|
|
|
GLuint CCTexture2D::getName()
|
|
|
|
{
|
|
|
|
return m_uName;
|
|
|
|
}
|
|
|
|
|
|
|
|
CGSize CCTexture2D::getContentSize()
|
|
|
|
{
|
|
|
|
return m_tContentSize;
|
|
|
|
}
|
|
|
|
|
|
|
|
GLfloat CCTexture2D::getMaxS()
|
|
|
|
{
|
|
|
|
return m_fMaxS;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CCTexture2D::setMaxS(GLfloat maxS)
|
|
|
|
{
|
|
|
|
m_fMaxS = maxS;
|
|
|
|
}
|
|
|
|
|
|
|
|
GLfloat CCTexture2D::getMaxT()
|
|
|
|
{
|
|
|
|
return m_fMaxT;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CCTexture2D::setMaxT(GLfloat maxT)
|
|
|
|
{
|
|
|
|
m_fMaxT = maxT;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CCTexture2D::getHasPremultipliedAlpha()
|
|
|
|
{
|
|
|
|
return m_bHasPremultipliedAlpha;
|
|
|
|
}
|
|
|
|
|
2010-08-02 10:58:00 +08:00
|
|
|
CCTexture2D * CCTexture2D::initWithData(const void *data, CCTexture2DPixelFormat pixelFormat, unsigned int pixelsWide, unsigned int pixelsHigh, CGSize contentSize)
|
2010-07-15 18:15:00 +08:00
|
|
|
{
|
|
|
|
glGenTextures(1, &m_uName);
|
|
|
|
glBindTexture(GL_TEXTURE_2D, m_uName);
|
|
|
|
|
|
|
|
this->setAntiAliasTexParameters();
|
|
|
|
|
|
|
|
// Specify OpenGL texture image
|
|
|
|
|
|
|
|
switch(pixelFormat)
|
|
|
|
{
|
|
|
|
case kCCTexture2DPixelFormat_RGBA8888:
|
|
|
|
glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, pixelsWide, pixelsHigh, 0, GL_RGBA, GL_UNSIGNED_BYTE, data);
|
|
|
|
break;
|
|
|
|
case kCCTexture2DPixelFormat_RGBA4444:
|
|
|
|
glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, pixelsWide, pixelsHigh, 0, GL_RGBA, GL_UNSIGNED_SHORT_4_4_4_4, data);
|
|
|
|
break;
|
|
|
|
case kCCTexture2DPixelFormat_RGB5A1:
|
|
|
|
glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, pixelsWide, pixelsHigh, 0, GL_RGBA, GL_UNSIGNED_SHORT_5_5_5_1, data);
|
|
|
|
break;
|
|
|
|
case kCCTexture2DPixelFormat_RGB565:
|
|
|
|
glTexImage2D(GL_TEXTURE_2D, 0, GL_RGB, pixelsWide, pixelsHigh, 0, GL_RGB, GL_UNSIGNED_SHORT_5_6_5, data);
|
|
|
|
break;
|
|
|
|
case kCCTexture2DPixelFormat_A8:
|
|
|
|
glTexImage2D(GL_TEXTURE_2D, 0, GL_ALPHA, pixelsWide, pixelsHigh, 0, GL_ALPHA, GL_UNSIGNED_BYTE, data);
|
|
|
|
break;
|
|
|
|
default:;
|
|
|
|
NSAssert(0, "NSInternalInconsistencyException");
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
m_tContentSize = contentSize;
|
|
|
|
m_uPixelsWide = pixelsWide;
|
|
|
|
m_uPixelsHigh = pixelsHigh;
|
|
|
|
m_ePixelFormat = pixelFormat;
|
2010-08-31 11:20:37 +08:00
|
|
|
m_fMaxS = contentSize.width / (float)(pixelsWide);
|
|
|
|
m_fMaxT = contentSize.height / (float)(pixelsHigh);
|
2010-07-15 18:15:00 +08:00
|
|
|
|
|
|
|
m_bHasPremultipliedAlpha = false;
|
|
|
|
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-08-06 16:05:19 +08:00
|
|
|
char * CCTexture2D::description(void)
|
2010-07-15 18:15:00 +08:00
|
|
|
{
|
2010-08-06 16:05:19 +08:00
|
|
|
char *ret = new char[100];
|
2010-09-03 11:47:35 +08:00
|
|
|
sprintf(ret, "<CCTexture2D | Name = %u | Dimensions = %u x %u | Coordinates = (%.2f, %.2f)>", m_uName, m_uPixelsWide, m_uPixelsHigh, m_fMaxS, m_fMaxT);
|
2010-07-15 18:15:00 +08:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
// implementation CCTexture2D (Image)
|
|
|
|
|
2010-07-26 17:32:23 +08:00
|
|
|
CCTexture2D* CCTexture2D::initWithImage(UIImage * uiImage)
|
2010-07-15 18:15:00 +08:00
|
|
|
{
|
2010-08-02 10:58:00 +08:00
|
|
|
unsigned int POTWide, POTHigh;
|
2010-07-15 18:15:00 +08:00
|
|
|
|
2010-07-26 17:32:23 +08:00
|
|
|
if(uiImage == NULL)
|
|
|
|
{
|
|
|
|
CCLOG("cocos2d: CCTexture2D. Can't create Texture. UIImage is nil");
|
|
|
|
this->release();
|
|
|
|
return NULL;
|
|
|
|
}
|
2010-07-15 18:15:00 +08:00
|
|
|
|
2010-07-26 17:32:23 +08:00
|
|
|
CCConfiguration *conf = CCConfiguration::sharedConfiguration();
|
2010-07-15 18:15:00 +08:00
|
|
|
|
2010-07-27 19:52:33 +08:00
|
|
|
#if CC_TEXTURE_NPOT_SUPPORT
|
2010-07-26 17:32:23 +08:00
|
|
|
if( conf->isSupportsNPOT() )
|
|
|
|
{
|
|
|
|
POTWide = uiImage->width();
|
|
|
|
POTHigh = uiImage->height();
|
|
|
|
}
|
|
|
|
else
|
2010-07-15 18:15:00 +08:00
|
|
|
#endif
|
2010-07-26 17:32:23 +08:00
|
|
|
{
|
|
|
|
POTWide = nextPOT(uiImage->width());
|
|
|
|
POTHigh = nextPOT(uiImage->height());
|
|
|
|
}
|
2010-07-15 18:15:00 +08:00
|
|
|
|
2010-07-26 17:32:23 +08:00
|
|
|
unsigned maxTextureSize = conf->getMaxTextureSize();
|
|
|
|
if( POTHigh > maxTextureSize || POTWide > maxTextureSize )
|
|
|
|
{
|
|
|
|
CCLOG("cocos2d: WARNING: Image (%u x %u) is bigger than the supported %u x %u", POTWide, POTHigh, maxTextureSize, maxTextureSize);
|
|
|
|
this->release();
|
|
|
|
return NULL;
|
|
|
|
}
|
2010-07-15 18:15:00 +08:00
|
|
|
|
2010-07-26 17:32:23 +08:00
|
|
|
// always load premultiplied images
|
|
|
|
this->initPremultipliedATextureWithImage(uiImage, POTWide, POTHigh);
|
|
|
|
return this;
|
2010-07-15 18:15:00 +08:00
|
|
|
}
|
2010-08-02 10:58:00 +08:00
|
|
|
CCTexture2D * CCTexture2D::initPremultipliedATextureWithImage(UIImage *image, unsigned int POTWide, unsigned int POTHigh)
|
2010-07-15 18:15:00 +08:00
|
|
|
{
|
2010-08-02 10:58:00 +08:00
|
|
|
unsigned int i;
|
2010-07-27 17:55:06 +08:00
|
|
|
void* data = NULL;
|
2010-07-27 19:52:33 +08:00
|
|
|
void* tempData =NULL;
|
|
|
|
unsigned int* inPixel32 = NULL;
|
|
|
|
unsigned short* outPixel16 = NULL;
|
|
|
|
bool hasAlpha;
|
2010-07-27 17:55:06 +08:00
|
|
|
CGSize imageSize;
|
|
|
|
CCTexture2DPixelFormat pixelFormat;
|
2010-07-15 18:15:00 +08:00
|
|
|
|
2010-07-27 17:55:06 +08:00
|
|
|
hasAlpha = image->isAlphaPixelFormat();
|
2010-07-15 18:15:00 +08:00
|
|
|
|
2010-07-27 17:55:06 +08:00
|
|
|
size_t bpp = image->CGImageGetBitsPerComponent();
|
|
|
|
int colorSpace = image->CGImageGetColorSpace();
|
2010-07-15 18:15:00 +08:00
|
|
|
|
2010-07-27 17:55:06 +08:00
|
|
|
if(colorSpace)
|
|
|
|
{
|
|
|
|
if(hasAlpha || bpp >= 8)
|
|
|
|
pixelFormat = defaultAlphaPixelFormat();
|
|
|
|
else
|
|
|
|
{
|
|
|
|
CCLOG("cocos2d: CCTexture2D: Using RGB565 texture since image has no alpha");
|
|
|
|
pixelFormat = kCCTexture2DPixelFormat_RGB565;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// NOTE: No colorspace means a mask image
|
|
|
|
CCLOG("cocos2d: CCTexture2D: Using A8 texture since image is a mask");
|
|
|
|
pixelFormat = kCCTexture2DPixelFormat_A8;
|
|
|
|
}
|
2010-07-15 18:15:00 +08:00
|
|
|
|
2010-08-31 11:20:37 +08:00
|
|
|
imageSize = CGSizeMake((float)(image->width()), (float)(image->height()));
|
2010-07-27 17:55:06 +08:00
|
|
|
|
|
|
|
// Create the bitmap graphics context
|
|
|
|
|
|
|
|
switch(pixelFormat) {
|
|
|
|
case kCCTexture2DPixelFormat_RGBA8888:
|
|
|
|
case kCCTexture2DPixelFormat_RGBA4444:
|
|
|
|
case kCCTexture2DPixelFormat_RGB5A1:
|
|
|
|
// colorSpace = CGColorSpaceCreateDeviceRGB();
|
2010-07-27 18:31:52 +08:00
|
|
|
// data = malloc(POTHigh * POTWide * 4);
|
2010-08-02 10:58:00 +08:00
|
|
|
// info = hasAlpha ? kCGImageAlphaPremultipliedLast : kCGImageAlphaNoneSkipLast;
|
|
|
|
// context = CGBitmapContextCreate(data, POTWide, POTHigh, 8, 4 * POTWide, colorSpace, info | kCGBitmapByteOrder32Big);
|
|
|
|
// CGColorSpaceRelease(colorSpace);
|
2010-07-27 18:31:52 +08:00
|
|
|
// break;
|
2010-07-27 17:55:06 +08:00
|
|
|
case kCCTexture2DPixelFormat_RGB565:
|
|
|
|
// colorSpace = CGColorSpaceCreateDeviceRGB();
|
2010-07-27 18:31:52 +08:00
|
|
|
// data = malloc(POTHigh * POTWide * 4);
|
2010-08-02 10:58:00 +08:00
|
|
|
// info = kCGImageAlphaNoneSkipLast;
|
|
|
|
// context = CGBitmapContextCreate(data, POTWide, POTHigh, 8, 4 * POTWide, colorSpace, info | kCGBitmapByteOrder32Big);
|
|
|
|
// CGColorSpaceRelease(colorSpace);
|
2010-07-27 18:31:52 +08:00
|
|
|
// break;
|
2010-07-27 17:55:06 +08:00
|
|
|
case kCCTexture2DPixelFormat_A8:
|
2010-07-27 18:31:52 +08:00
|
|
|
// data = malloc(POTHigh * POTWide);
|
2010-08-02 10:58:00 +08:00
|
|
|
// info = kCGImageAlphaOnly;
|
|
|
|
// context = CGBitmapContextCreate(data, POTWide, POTHigh, 8, POTWide, NULL, info);
|
|
|
|
|
2010-08-31 11:20:37 +08:00
|
|
|
tempData = (void*)(image->getRGBA8888Data());
|
2010-08-02 10:58:00 +08:00
|
|
|
NSAssert(tempData != NULL, "NULL image data.");
|
2010-07-27 19:52:33 +08:00
|
|
|
if(image->width() == POTWide && image->height() == POTHigh)
|
2010-07-27 18:31:52 +08:00
|
|
|
{
|
|
|
|
data = tempData;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2010-08-02 10:16:35 +08:00
|
|
|
data = new UINT8[POTHigh * POTWide * 4];
|
2010-07-27 18:31:52 +08:00
|
|
|
memset(data, 0, POTHigh * POTWide * 4);
|
|
|
|
|
|
|
|
UINT8* pPixelData = (UINT8*) tempData;
|
|
|
|
UINT8* pTargetData = (UINT8*) data;
|
|
|
|
|
2010-08-02 10:58:00 +08:00
|
|
|
for(unsigned int y=0; y<image->height(); ++y)
|
2010-07-27 18:31:52 +08:00
|
|
|
{
|
2010-07-27 19:52:33 +08:00
|
|
|
memcpy(pTargetData+POTWide*4*y, pPixelData+(image->width())*4*y, (image->width())*4);
|
2010-07-27 18:31:52 +08:00
|
|
|
}
|
2010-08-12 17:39:25 +08:00
|
|
|
delete []tempData;
|
2010-08-02 10:58:00 +08:00
|
|
|
}
|
2010-07-27 17:55:06 +08:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
NSAssert(0, "Invalid pixel format");
|
|
|
|
//[NSException raise:NSInternalInconsistencyException format:@"Invalid pixel format"];
|
|
|
|
}
|
2010-07-15 18:15:00 +08:00
|
|
|
|
|
|
|
|
2010-08-02 10:58:00 +08:00
|
|
|
// CGContextClearRect(context, CGRectMake(0, 0, POTWide, POTHigh));
|
|
|
|
// CGContextTranslateCTM(context, 0, POTHigh - imageSize.height);
|
|
|
|
// CGContextDrawImage(context, CGRectMake(0, 0, CGImageGetWidth(image), CGImageGetHeight(image)), image);
|
2010-07-15 18:15:00 +08:00
|
|
|
|
2010-07-27 17:55:06 +08:00
|
|
|
// Repack the pixel data into the right format
|
|
|
|
|
|
|
|
if(pixelFormat == kCCTexture2DPixelFormat_RGB565) {
|
|
|
|
//Convert "RRRRRRRRRGGGGGGGGBBBBBBBBAAAAAAAA" to "RRRRRGGGGGGBBBBB"
|
2010-08-02 10:16:35 +08:00
|
|
|
tempData = new UINT8[POTHigh * POTWide * 2];
|
2010-07-27 17:55:06 +08:00
|
|
|
inPixel32 = (unsigned int*)data;
|
|
|
|
outPixel16 = (unsigned short*)tempData;
|
|
|
|
for(i = 0; i < POTWide * POTHigh; ++i, ++inPixel32)
|
|
|
|
*outPixel16++ = ((((*inPixel32 >> 0) & 0xFF) >> 3) << 11) | ((((*inPixel32 >> 8) & 0xFF) >> 2) << 5) | ((((*inPixel32 >> 16) & 0xFF) >> 3) << 0);
|
2010-08-24 09:49:38 +08:00
|
|
|
delete [] data;
|
2010-07-27 17:55:06 +08:00
|
|
|
data = tempData;
|
|
|
|
}
|
|
|
|
else if (pixelFormat == kCCTexture2DPixelFormat_RGBA4444) {
|
|
|
|
//Convert "RRRRRRRRRGGGGGGGGBBBBBBBBAAAAAAAA" to "RRRRGGGGBBBBAAAA"
|
2010-08-02 10:16:35 +08:00
|
|
|
tempData = new UINT8[POTHigh * POTWide * 2];
|
2010-07-27 17:55:06 +08:00
|
|
|
inPixel32 = (unsigned int*)data;
|
|
|
|
outPixel16 = (unsigned short*)tempData;
|
|
|
|
for(i = 0; i < POTWide * POTHigh; ++i, ++inPixel32)
|
|
|
|
*outPixel16++ =
|
|
|
|
((((*inPixel32 >> 0) & 0xFF) >> 4) << 12) | // R
|
|
|
|
((((*inPixel32 >> 8) & 0xFF) >> 4) << 8) | // G
|
|
|
|
((((*inPixel32 >> 16) & 0xFF) >> 4) << 4) | // B
|
|
|
|
((((*inPixel32 >> 24) & 0xFF) >> 4) << 0); // A
|
|
|
|
|
2010-08-24 09:49:38 +08:00
|
|
|
delete [] data;
|
2010-07-27 17:55:06 +08:00
|
|
|
data = tempData;
|
|
|
|
}
|
|
|
|
else if (pixelFormat == kCCTexture2DPixelFormat_RGB5A1) {
|
|
|
|
//Convert "RRRRRRRRRGGGGGGGGBBBBBBBBAAAAAAAA" to "RRRRRGGGGGBBBBBA"
|
2010-08-02 10:16:35 +08:00
|
|
|
tempData = new UINT8[POTHigh * POTWide * 2];
|
2010-07-27 17:55:06 +08:00
|
|
|
inPixel32 = (unsigned int*)data;
|
|
|
|
outPixel16 = (unsigned short*)tempData;
|
|
|
|
for(i = 0; i < POTWide * POTHigh; ++i, ++inPixel32)
|
|
|
|
*outPixel16++ =
|
|
|
|
((((*inPixel32 >> 0) & 0xFF) >> 3) << 11) | // R
|
|
|
|
((((*inPixel32 >> 8) & 0xFF) >> 3) << 6) | // G
|
|
|
|
((((*inPixel32 >> 16) & 0xFF) >> 3) << 1) | // B
|
|
|
|
((((*inPixel32 >> 24) & 0xFF) >> 7) << 0); // A
|
|
|
|
|
2010-08-12 17:39:25 +08:00
|
|
|
delete []data;
|
2010-07-27 17:55:06 +08:00
|
|
|
data = tempData;
|
|
|
|
}
|
|
|
|
|
2010-08-02 10:58:00 +08:00
|
|
|
if (data)
|
|
|
|
{
|
2010-07-27 17:55:06 +08:00
|
|
|
this->initWithData(data, pixelFormat, POTWide, POTHigh, imageSize);
|
2010-07-15 18:15:00 +08:00
|
|
|
|
2010-07-27 17:55:06 +08:00
|
|
|
// should be after calling super init
|
2010-08-04 17:19:37 +08:00
|
|
|
m_bHasPremultipliedAlpha = image->isPremultipliedAlpha();
|
2010-07-15 18:15:00 +08:00
|
|
|
|
2010-07-27 17:55:06 +08:00
|
|
|
//CGContextRelease(context);
|
2010-08-24 09:49:38 +08:00
|
|
|
delete [] data;
|
2010-08-02 10:58:00 +08:00
|
|
|
}
|
2010-07-27 17:55:06 +08:00
|
|
|
return this;
|
2010-07-26 17:32:23 +08:00
|
|
|
}
|
2010-07-15 18:15:00 +08:00
|
|
|
|
|
|
|
// implementation CCTexture2D (Text)
|
2010-08-16 15:02:17 +08:00
|
|
|
CCTexture2D * CCTexture2D::initWithString(const char *text, const char *fontName, float fontSize)
|
2010-07-15 18:15:00 +08:00
|
|
|
{
|
2010-08-16 15:02:17 +08:00
|
|
|
return initWithString(text, CGSizeMake(0,0), UITextAlignmentCenter, fontName, fontSize);
|
2010-07-26 17:32:23 +08:00
|
|
|
}
|
2010-08-16 15:02:17 +08:00
|
|
|
CCTexture2D * CCTexture2D::initWithString(const char *text, CGSize dimensions, UITextAlignment alignment, const char *fontName, float fontSize)
|
2010-08-16 11:59:53 +08:00
|
|
|
{
|
2010-08-16 15:02:17 +08:00
|
|
|
CCXBitmapDC *pBitmapDC = new CCXBitmapDC(text, dimensions, alignment, fontName, fontSize);
|
2010-07-15 18:15:00 +08:00
|
|
|
|
2010-08-17 16:12:46 +08:00
|
|
|
void* pBitData = pBitmapDC->getBuffer();
|
|
|
|
CGSize size = pBitmapDC->getSize();
|
2010-07-15 18:15:00 +08:00
|
|
|
|
2010-08-17 16:12:46 +08:00
|
|
|
initWithData(pBitData, kCCTexture2DPixelFormat_RGBA8888, (UINT)size.width, (UINT)size.height, size);
|
2010-07-15 18:15:00 +08:00
|
|
|
|
2010-08-16 11:59:53 +08:00
|
|
|
delete pBitmapDC;
|
2010-08-11 18:09:10 +08:00
|
|
|
return this;
|
|
|
|
}
|
2010-07-15 18:15:00 +08:00
|
|
|
|
|
|
|
|
|
|
|
// implementation CCTexture2D (Drawing)
|
|
|
|
|
|
|
|
void CCTexture2D::drawAtPoint(CGPoint point)
|
|
|
|
{
|
2010-07-16 10:34:42 +08:00
|
|
|
GLfloat coordinates[] = {
|
|
|
|
0.0f, m_fMaxT,
|
2010-07-15 18:15:00 +08:00
|
|
|
m_fMaxS,m_fMaxT,
|
|
|
|
0.0f, 0.0f,
|
|
|
|
m_fMaxS,0.0f };
|
|
|
|
|
|
|
|
GLfloat width = (GLfloat)m_uPixelsWide * m_fMaxS,
|
|
|
|
height = (GLfloat)m_uPixelsHigh * m_fMaxT;
|
|
|
|
|
|
|
|
#if 0
|
2010-07-16 10:34:42 +08:00
|
|
|
GLfloat vertices[] = {
|
|
|
|
-width / 2 + point.x, -height / 2 + point.y, 0.0f,
|
2010-07-15 18:15:00 +08:00
|
|
|
width / 2 + point.x, -height / 2 + point.y, 0.0f,
|
|
|
|
-width / 2 + point.x, height / 2 + point.y, 0.0f,
|
|
|
|
width / 2 + point.x, height / 2 + point.y, 0.0f };
|
|
|
|
|
|
|
|
#else // anchor is done by cocos2d automagically
|
2010-07-16 10:34:42 +08:00
|
|
|
GLfloat vertices[] = {
|
|
|
|
point.x, point.y, 0.0f,
|
2010-07-15 18:15:00 +08:00
|
|
|
width + point.x, point.y, 0.0f,
|
|
|
|
point.x, height + point.y, 0.0f,
|
|
|
|
width + point.x, height + point.y, 0.0f };
|
|
|
|
#endif
|
|
|
|
|
|
|
|
glBindTexture(GL_TEXTURE_2D, m_uName);
|
|
|
|
glVertexPointer(3, GL_FLOAT, 0, vertices);
|
|
|
|
glTexCoordPointer(2, GL_FLOAT, 0, coordinates);
|
|
|
|
glDrawArrays(GL_TRIANGLE_STRIP, 0, 4);
|
|
|
|
}
|
|
|
|
|
|
|
|
void CCTexture2D::drawInRect(CGRect rect)
|
|
|
|
{
|
2010-07-16 10:34:42 +08:00
|
|
|
GLfloat coordinates[] = {
|
|
|
|
0.0f, m_fMaxT,
|
2010-07-15 18:15:00 +08:00
|
|
|
m_fMaxS,m_fMaxT,
|
|
|
|
0.0f, 0.0f,
|
|
|
|
m_fMaxS,0.0f };
|
|
|
|
|
|
|
|
GLfloat vertices[] = { rect.origin.x, rect.origin.y, /*0.0f,*/
|
|
|
|
rect.origin.x + rect.size.width, rect.origin.y, /*0.0f,*/
|
|
|
|
rect.origin.x, rect.origin.y + rect.size.height, /*0.0f,*/
|
|
|
|
rect.origin.x + rect.size.width, rect.origin.y + rect.size.height, /*0.0f*/ };
|
|
|
|
|
|
|
|
glBindTexture(GL_TEXTURE_2D, m_uName);
|
|
|
|
glVertexPointer(2, GL_FLOAT, 0, vertices);
|
|
|
|
glTexCoordPointer(2, GL_FLOAT, 0, coordinates);
|
|
|
|
glDrawArrays(GL_TRIANGLE_STRIP, 0, 4);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// implementation CCTexture2D (PVRTC)
|
2010-08-31 16:53:25 +08:00
|
|
|
#ifdef _POWERVR_SUPPORT_
|
2010-07-15 18:15:00 +08:00
|
|
|
CCTexture2D * CCTexture2D::initWithPVRTCData(const void *data, int level, int bpp, bool hasAlpha, int length)
|
|
|
|
{
|
|
|
|
if( !(CCConfiguration::sharedConfiguration()->isSupportsPVRTC()) )
|
|
|
|
{
|
|
|
|
CCLOG("cocos2d: WARNING: PVRTC images is not supported.");
|
|
|
|
this->release();
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
glGenTextures(1, &m_uName);
|
|
|
|
glBindTexture(GL_TEXTURE_2D, m_uName);
|
|
|
|
|
|
|
|
this->setAntiAliasTexParameters();
|
|
|
|
|
|
|
|
GLenum format;
|
|
|
|
GLsizei size = length * length * bpp / 8;
|
|
|
|
if(hasAlpha) {
|
|
|
|
format = (bpp == 4) ? GL_COMPRESSED_RGBA_PVRTC_4BPPV1_IMG : GL_COMPRESSED_RGBA_PVRTC_2BPPV1_IMG;
|
|
|
|
} else {
|
|
|
|
format = (bpp == 4) ? GL_COMPRESSED_RGB_PVRTC_4BPPV1_IMG : GL_COMPRESSED_RGB_PVRTC_2BPPV1_IMG;
|
|
|
|
}
|
|
|
|
if(size < 32) {
|
|
|
|
size = 32;
|
|
|
|
}
|
|
|
|
glCompressedTexImage2D(GL_TEXTURE_2D, level, format, length, length, 0, size, data);
|
|
|
|
|
2010-08-31 11:20:37 +08:00
|
|
|
m_tContentSize = CGSizeMake((float)(length), (float)(length));
|
2010-07-15 18:15:00 +08:00
|
|
|
m_uPixelsWide = length;
|
|
|
|
m_uPixelsHigh = length;
|
|
|
|
m_fMaxS = 1.0f;
|
|
|
|
m_fMaxT = 1.0f;
|
|
|
|
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
2010-07-22 11:15:25 +08:00
|
|
|
CCTexture2D * CCTexture2D::initWithPVRTCFile(const char* file)
|
2010-07-15 18:15:00 +08:00
|
|
|
{
|
2010-08-02 10:58:00 +08:00
|
|
|
if (! CCConfiguration::sharedConfiguration()->isSupportsPVRTC())
|
|
|
|
{
|
2010-07-21 11:13:32 +08:00
|
|
|
CCLOG("cocos2d: WARNING: PVRTC images is not supported");
|
|
|
|
this->release();
|
|
|
|
return NULL;
|
2010-08-02 10:58:00 +08:00
|
|
|
}
|
2010-07-21 11:13:32 +08:00
|
|
|
|
|
|
|
CCPVRTexture *pvr = new CCPVRTexture();
|
|
|
|
pvr = pvr->initWithContentsOfFile(file);
|
|
|
|
if( pvr )
|
|
|
|
{
|
|
|
|
pvr->setRetainName(true); // don't dealloc texture on release
|
2010-07-15 18:15:00 +08:00
|
|
|
|
2010-07-21 11:13:32 +08:00
|
|
|
m_uName = pvr->getName(); // texture id
|
|
|
|
m_fMaxS = 1.0f;
|
|
|
|
m_fMaxT = 1.0f;
|
|
|
|
m_uPixelsWide = pvr->getWidth(); // width
|
|
|
|
m_uPixelsHigh = pvr->getHeight(); // height
|
2010-08-02 10:58:00 +08:00
|
|
|
/// be careful : unsigned int to float
|
2010-08-31 11:20:37 +08:00
|
|
|
m_tContentSize = CGSizeMake((float)(m_uPixelsWide), (float)(m_uPixelsHigh));
|
2010-07-21 11:13:32 +08:00
|
|
|
|
|
|
|
pvr->release();
|
|
|
|
|
|
|
|
this->setAntiAliasTexParameters();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
CCLOG("cocos2d: Couldn't load PVR image");
|
|
|
|
this->release();
|
|
|
|
return NULL;
|
2010-07-15 18:15:00 +08:00
|
|
|
}
|
|
|
|
return this;
|
|
|
|
}
|
2010-08-31 16:53:25 +08:00
|
|
|
#endif
|
2010-07-15 18:15:00 +08:00
|
|
|
|
|
|
|
//
|
|
|
|
// Use to apply MIN/MAG filter
|
|
|
|
//
|
|
|
|
// implementation CCTexture2D (GLFilter)
|
|
|
|
|
|
|
|
void CCTexture2D::generateMipmap()
|
|
|
|
{
|
|
|
|
NSAssert( m_uPixelsWide == nextPOT(m_uPixelsWide) && m_uPixelsHigh == nextPOT(m_uPixelsHigh), "Mimpap texture only works in POT textures");
|
|
|
|
glBindTexture( GL_TEXTURE_2D, this->m_uName );
|
2010-08-16 11:59:53 +08:00
|
|
|
glGenerateMipmapOES(GL_TEXTURE_2D);
|
2010-07-15 18:15:00 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void CCTexture2D::setTexParameters(ccTexParams *texParams)
|
|
|
|
{
|
|
|
|
NSAssert( (m_uPixelsWide == nextPOT(m_uPixelsWide) && m_uPixelsHigh == nextPOT(m_uPixelsHigh)) ||
|
|
|
|
(texParams->wrapS == GL_CLAMP_TO_EDGE && texParams->wrapT == GL_CLAMP_TO_EDGE),
|
|
|
|
"GL_CLAMP_TO_EDGE should be used in NPOT textures");
|
|
|
|
glBindTexture( GL_TEXTURE_2D, this->m_uName );
|
|
|
|
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, texParams->minFilter );
|
|
|
|
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, texParams->magFilter );
|
|
|
|
glTexParameteri( GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, texParams->wrapS );
|
|
|
|
glTexParameteri( GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, texParams->wrapT );
|
|
|
|
}
|
|
|
|
|
|
|
|
void CCTexture2D::setAliasTexParameters()
|
|
|
|
{
|
|
|
|
ccTexParams texParams = { GL_NEAREST, GL_NEAREST, GL_CLAMP_TO_EDGE, GL_CLAMP_TO_EDGE };
|
|
|
|
this->setTexParameters(&texParams);
|
|
|
|
}
|
|
|
|
|
|
|
|
void CCTexture2D::setAntiAliasTexParameters()
|
|
|
|
{
|
|
|
|
ccTexParams texParams = { GL_LINEAR, GL_LINEAR, GL_CLAMP_TO_EDGE, GL_CLAMP_TO_EDGE };
|
|
|
|
this->setTexParameters(&texParams);
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Texture options for images that contains alpha
|
|
|
|
//
|
|
|
|
// implementation CCTexture2D (PixelFormat)
|
|
|
|
|
|
|
|
void CCTexture2D::setDefaultAlphaPixelFormat(CCTexture2DPixelFormat format)
|
|
|
|
{
|
|
|
|
g_defaultAlphaPixelFormat = format;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
CCTexture2DPixelFormat CCTexture2D::defaultAlphaPixelFormat()
|
|
|
|
{
|
|
|
|
return g_defaultAlphaPixelFormat;
|
|
|
|
}
|
2010-08-02 10:58:00 +08:00
|
|
|
|
2010-08-04 15:46:12 +08:00
|
|
|
}//namespace cocos2d
|