2011-02-23 18:22:05 +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 <Foundation/Foundation.h>
|
|
|
|
#include <UIKit/UIKit.h>
|
2011-03-09 16:19:20 +08:00
|
|
|
#include "CCImage.h"
|
2011-03-29 17:32:52 +08:00
|
|
|
#include "CCFileUtils.h"
|
2011-03-09 16:19:20 +08:00
|
|
|
#include <string>
|
2011-02-23 18:22:05 +08:00
|
|
|
|
|
|
|
typedef struct
|
|
|
|
{
|
2011-03-04 17:02:22 +08:00
|
|
|
unsigned int height;
|
|
|
|
unsigned int width;
|
|
|
|
int bitsPerComponent;
|
|
|
|
bool hasAlpha;
|
|
|
|
bool isPremultipliedAlpha;
|
|
|
|
unsigned char* data;
|
2011-02-23 18:22:05 +08:00
|
|
|
} tImageInfo;
|
|
|
|
|
|
|
|
static unsigned int nextPOT(unsigned int x)
|
|
|
|
{
|
2011-03-04 17:02:22 +08:00
|
|
|
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;
|
2011-02-23 18:22:05 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
typedef enum {
|
2011-03-04 17:02:22 +08:00
|
|
|
kCCTexture2DPixelFormat_Automatic = 0,
|
|
|
|
//! 32-bit texture: RGBA8888
|
|
|
|
kCCTexture2DPixelFormat_RGBA8888,
|
|
|
|
//! 24-bit texture: RGBA888
|
|
|
|
kCCTexture2DPixelFormat_RGB888,
|
|
|
|
//! 16-bit texture without Alpha channel
|
|
|
|
kCCTexture2DPixelFormat_RGB565,
|
|
|
|
//! 8-bit textures used as masks
|
|
|
|
kCCTexture2DPixelFormat_A8,
|
|
|
|
//! 16-bit textures: RGBA4444
|
|
|
|
kCCTexture2DPixelFormat_RGBA4444,
|
|
|
|
//! 16-bit textures: RGB5A1
|
|
|
|
kCCTexture2DPixelFormat_RGB5A1,
|
|
|
|
|
|
|
|
//! Default texture format: RGBA8888
|
|
|
|
kCCTexture2DPixelFormat_Default = kCCTexture2DPixelFormat_RGBA8888,
|
|
|
|
|
|
|
|
// backward compatibility stuff
|
|
|
|
kTexture2DPixelFormat_Automatic = kCCTexture2DPixelFormat_Automatic,
|
|
|
|
kTexture2DPixelFormat_RGBA8888 = kCCTexture2DPixelFormat_RGBA8888,
|
|
|
|
kTexture2DPixelFormat_RGB888 = kCCTexture2DPixelFormat_RGB888,
|
|
|
|
kTexture2DPixelFormat_RGB565 = kCCTexture2DPixelFormat_RGB565,
|
|
|
|
kTexture2DPixelFormat_A8 = kCCTexture2DPixelFormat_A8,
|
|
|
|
kTexture2DPixelFormat_RGBA4444 = kCCTexture2DPixelFormat_RGBA4444,
|
|
|
|
kTexture2DPixelFormat_RGB5A1 = kCCTexture2DPixelFormat_RGB5A1,
|
|
|
|
kTexture2DPixelFormat_Default = kCCTexture2DPixelFormat_Default
|
|
|
|
|
2011-02-23 18:22:05 +08:00
|
|
|
} CCTexture2DPixelFormat;
|
|
|
|
|
|
|
|
static bool _initPremultipliedATextureWithImage(CGImageRef image, NSUInteger POTWide, NSUInteger POTHigh, tImageInfo *pImageInfo)
|
|
|
|
{
|
2011-03-04 17:02:22 +08:00
|
|
|
NSUInteger i;
|
|
|
|
CGContextRef context = nil;
|
|
|
|
unsigned char* data = nil;;
|
|
|
|
CGColorSpaceRef colorSpace;
|
|
|
|
unsigned char* tempData;
|
|
|
|
unsigned int* inPixel32;
|
|
|
|
unsigned short* outPixel16;
|
|
|
|
bool hasAlpha;
|
|
|
|
CGImageAlphaInfo info;
|
|
|
|
CGSize imageSize;
|
|
|
|
CCTexture2DPixelFormat pixelFormat;
|
|
|
|
|
|
|
|
info = CGImageGetAlphaInfo(image);
|
|
|
|
hasAlpha = ((info == kCGImageAlphaPremultipliedLast) || (info == kCGImageAlphaPremultipliedFirst) || (info == kCGImageAlphaLast) || (info == kCGImageAlphaFirst) ? YES : NO);
|
|
|
|
|
|
|
|
size_t bpp = CGImageGetBitsPerComponent(image);
|
|
|
|
colorSpace = CGImageGetColorSpace(image);
|
|
|
|
|
|
|
|
if(colorSpace) {
|
|
|
|
if(hasAlpha || bpp >= 8)
|
|
|
|
pixelFormat = kCCTexture2DPixelFormat_Default;
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
|
|
|
imageSize.width = CGImageGetWidth(image);
|
|
|
|
imageSize.height = CGImageGetHeight(image);
|
|
|
|
//imageSize = CGSizeMake(CGImageGetWidth(image), CGImageGetHeight(image));
|
|
|
|
|
|
|
|
// Create the bitmap graphics context
|
|
|
|
|
|
|
|
switch(pixelFormat) {
|
|
|
|
case kCCTexture2DPixelFormat_RGBA8888:
|
|
|
|
case kCCTexture2DPixelFormat_RGBA4444:
|
|
|
|
case kCCTexture2DPixelFormat_RGB5A1:
|
|
|
|
colorSpace = CGColorSpaceCreateDeviceRGB();
|
|
|
|
//data = malloc(POTHigh * POTWide * 4);
|
|
|
|
data = new unsigned char[POTHigh * POTWide * 4];
|
|
|
|
info = hasAlpha ? kCGImageAlphaPremultipliedLast : kCGImageAlphaNoneSkipLast;
|
|
|
|
// info = kCGImageAlphaPremultipliedLast; // issue #886. This patch breaks BMP images.
|
|
|
|
context = CGBitmapContextCreate(data, POTWide, POTHigh, 8, 4 * POTWide, colorSpace, info | kCGBitmapByteOrder32Big);
|
|
|
|
CGColorSpaceRelease(colorSpace);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case kCCTexture2DPixelFormat_RGB565:
|
|
|
|
colorSpace = CGColorSpaceCreateDeviceRGB();
|
|
|
|
//data = malloc(POTHigh * POTWide * 4);
|
|
|
|
data = new unsigned char[POTHigh * POTWide * 4];
|
|
|
|
info = kCGImageAlphaNoneSkipLast;
|
|
|
|
context = CGBitmapContextCreate(data, POTWide, POTHigh, 8, 4 * POTWide, colorSpace, info | kCGBitmapByteOrder32Big);
|
|
|
|
CGColorSpaceRelease(colorSpace);
|
|
|
|
break;
|
|
|
|
case kCCTexture2DPixelFormat_A8:
|
|
|
|
//data = malloc(POTHigh * POTWide);
|
|
|
|
data = new unsigned char[POTHigh * POTWide];
|
|
|
|
info = kCGImageAlphaOnly;
|
|
|
|
context = CGBitmapContextCreate(data, POTWide, POTHigh, 8, POTWide, NULL, info);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
//[NSException raise:NSInternalInconsistencyException format:@"Invalid pixel format"];
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
CGRect rect;
|
|
|
|
rect.size.width = POTWide;
|
|
|
|
rect.size.height = POTHigh;
|
|
|
|
rect.origin.x = 0;
|
|
|
|
rect.origin.y = 0;
|
|
|
|
|
|
|
|
//CGContextClearRect(context, CGRectMake(0, 0, POTWide, POTHigh));
|
|
|
|
CGContextClearRect(context, rect);
|
|
|
|
CGContextTranslateCTM(context, 0, POTHigh - imageSize.height);
|
|
|
|
//CGContextDrawImage(context, CGRectMake(0, 0, CGImageGetWidth(image), CGImageGetHeight(image)), image);
|
|
|
|
rect.size.width = CGImageGetWidth(image);
|
|
|
|
rect.size.height = CGImageGetHeight(image);
|
|
|
|
rect.origin.x = 0;
|
|
|
|
rect.origin.y = 0;
|
|
|
|
//CGContextDrawImage(context, CGRectMake(0, 0, CGImageGetWidth(image), CGImageGetHeight(image)), image);
|
|
|
|
CGContextDrawImage(context, rect, image);
|
|
|
|
|
|
|
|
// Repack the pixel data into the right format
|
|
|
|
|
|
|
|
if(pixelFormat == kCCTexture2DPixelFormat_RGB565) {
|
|
|
|
//Convert "RRRRRRRRRGGGGGGGGBBBBBBBBAAAAAAAA" to "RRRRRGGGGGGBBBBB"
|
|
|
|
//tempData = malloc(POTHigh * POTWide * 2);
|
|
|
|
tempData = new unsigned char[POTHigh * POTWide * 2];
|
|
|
|
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);
|
|
|
|
//free(data);
|
|
|
|
delete[] data;
|
|
|
|
data = tempData;
|
|
|
|
|
|
|
|
}
|
|
|
|
else if (pixelFormat == kCCTexture2DPixelFormat_RGBA4444) {
|
|
|
|
//Convert "RRRRRRRRRGGGGGGGGBBBBBBBBAAAAAAAA" to "RRRRGGGGBBBBAAAA"
|
|
|
|
//tempData = malloc(POTHigh * POTWide * 2);
|
|
|
|
tempData = new unsigned char[POTHigh * POTWide * 2];
|
|
|
|
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
|
|
|
|
|
|
|
|
|
|
|
|
//free(data);
|
|
|
|
delete[] data;
|
|
|
|
data = tempData;
|
|
|
|
|
|
|
|
}
|
|
|
|
else if (pixelFormat == kCCTexture2DPixelFormat_RGB5A1) {
|
|
|
|
//Convert "RRRRRRRRRGGGGGGGGBBBBBBBBAAAAAAAA" to "RRRRRGGGGGBBBBBA"
|
|
|
|
//tempData = malloc(POTHigh * POTWide * 2);
|
|
|
|
tempData = new unsigned char[POTHigh * POTWide * 2];
|
|
|
|
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
|
|
|
|
|
|
|
|
//free(data);
|
|
|
|
delete[] data;
|
|
|
|
data = tempData;
|
|
|
|
}
|
|
|
|
//self = [self initWithData:data pixelFormat:pixelFormat pixelsWide:POTWide pixelsHigh:POTHigh contentSize:imageSize];
|
|
|
|
|
|
|
|
// should be after calling super init
|
|
|
|
//s_imageInfo.isPremultipliedAlpha = (info == kCGImageAlphaPremultipliedLast || info == kCGImageAlphaPremultipliedFirst);
|
|
|
|
pImageInfo->isPremultipliedAlpha = true;
|
|
|
|
pImageInfo->hasAlpha = true;
|
|
|
|
//s_imageInfo.hasAlpha = hasAlpha;
|
|
|
|
pImageInfo->bitsPerComponent = bpp;
|
|
|
|
pImageInfo->width = imageSize.width;
|
|
|
|
pImageInfo->height = imageSize.height;
|
|
|
|
|
|
|
|
if (pImageInfo->data)
|
|
|
|
{
|
|
|
|
delete []pImageInfo->data;
|
|
|
|
}
|
|
|
|
pImageInfo->data = data;
|
|
|
|
|
|
|
|
CGContextRelease(context);
|
|
|
|
return true;
|
2011-02-23 18:22:05 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static bool _initWithImage(CGImageRef CGImage, tImageInfo *pImageinfo)
|
|
|
|
{
|
2011-03-04 17:02:22 +08:00
|
|
|
NSUInteger POTWide, POTHigh;
|
|
|
|
|
|
|
|
if(CGImage == NULL) {
|
|
|
|
//CCLOG(@"cocos2d: CCTexture2D. Can't create Texture. ccxImage is nil");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
//CCConfiguration *conf = [CCConfiguration sharedConfiguration];
|
|
|
|
// cocos2d::CCConfiguration *conf = cocos2d::CCConfiguration::sharedConfiguration();
|
|
|
|
|
|
|
|
POTWide = CGImageGetWidth(CGImage);
|
|
|
|
POTHigh = CGImageGetHeight(CGImage);
|
|
|
|
|
|
|
|
// unsigned maxTextureSize = conf->getMaxTextureSize();
|
|
|
|
// if( POTHigh > maxTextureSize || POTWide > maxTextureSize ) {
|
|
|
|
// return false;
|
|
|
|
// }
|
|
|
|
|
|
|
|
// always load premultiplied images
|
|
|
|
_initPremultipliedATextureWithImage(CGImage, POTWide, POTHigh, pImageinfo);
|
|
|
|
|
|
|
|
return true;
|
2011-02-23 18:22:05 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static bool _initWithFile(const char* path, tImageInfo *pImageinfo)
|
|
|
|
{
|
2011-03-04 17:02:22 +08:00
|
|
|
CGImageRef CGImage;
|
|
|
|
UIImage *jpg;
|
|
|
|
UIImage *png;
|
|
|
|
bool ret;
|
|
|
|
|
|
|
|
// convert jpg to png before loading the texture
|
|
|
|
|
|
|
|
NSString *fullPath = [NSString stringWithUTF8String:path];
|
|
|
|
jpg = [[UIImage alloc] initWithContentsOfFile: fullPath];
|
|
|
|
png = [[UIImage alloc] initWithData:UIImagePNGRepresentation(jpg)];
|
|
|
|
CGImage = png.CGImage;
|
|
|
|
|
|
|
|
ret = _initWithImage(CGImage, pImageinfo);
|
|
|
|
|
|
|
|
[png release];
|
|
|
|
[jpg release];
|
|
|
|
|
|
|
|
return ret;
|
2011-02-23 18:22:05 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static bool _initWithData(void * pBuffer, int length, tImageInfo *pImageinfo)
|
|
|
|
{
|
2011-03-04 17:02:22 +08:00
|
|
|
bool ret = false;
|
|
|
|
|
|
|
|
if (pBuffer) {
|
|
|
|
CGImageRef CGImage;
|
|
|
|
NSData *data;
|
|
|
|
|
|
|
|
data = [NSData dataWithBytes:pBuffer length:length];
|
|
|
|
CGImage = [[UIImage imageWithData:data] CGImage];
|
|
|
|
|
|
|
|
ret = _initWithImage(CGImage, pImageinfo);
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
2011-02-23 18:22:05 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static bool _isValidFontName(const char *fontName)
|
|
|
|
{
|
2011-03-04 17:02:22 +08:00
|
|
|
bool ret = false;
|
|
|
|
|
|
|
|
NSString *fontNameNS = [NSString stringWithUTF8String:fontName];
|
|
|
|
|
|
|
|
for (NSString *familiName in [UIFont familyNames]) {
|
|
|
|
if ([familiName isEqualToString:fontNameNS]) {
|
|
|
|
ret = true;
|
|
|
|
goto out;
|
2011-02-23 18:22:05 +08:00
|
|
|
}
|
|
|
|
|
2011-03-04 17:02:22 +08:00
|
|
|
for(NSString *font in [UIFont fontNamesForFamilyName: familiName]){
|
|
|
|
if ([font isEqualToString: fontNameNS]){
|
|
|
|
ret = true;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
out:
|
|
|
|
return ret;
|
2011-02-23 18:22:05 +08:00
|
|
|
}
|
|
|
|
|
2011-03-09 16:19:20 +08:00
|
|
|
static bool _initWithString(const char * pText, cocos2d::CCImage::ETextAlign eAlign, const char * pFontName, int nSize, tImageInfo* pInfo)
|
2011-02-23 18:22:05 +08:00
|
|
|
{
|
2011-03-04 17:02:22 +08:00
|
|
|
bool bRet = false;
|
|
|
|
do
|
|
|
|
{
|
2011-03-09 16:19:20 +08:00
|
|
|
CC_BREAK_IF(! pText || ! pInfo);
|
2011-03-04 17:02:22 +08:00
|
|
|
|
|
|
|
NSString * string = [NSString stringWithUTF8String:pText];
|
|
|
|
|
|
|
|
// create the font
|
|
|
|
NSString * fntName = _isValidFontName(pFontName) ? [NSString stringWithUTF8String:pFontName] : @"MarkerFelt-Wide";
|
|
|
|
UIFont * font = [UIFont fontWithName:fntName size:nSize];
|
|
|
|
if (! font)
|
2011-02-23 18:22:05 +08:00
|
|
|
{
|
2011-03-04 17:02:22 +08:00
|
|
|
font = [UIFont systemFontOfSize:nSize];
|
2011-02-23 18:22:05 +08:00
|
|
|
}
|
2011-03-09 16:19:20 +08:00
|
|
|
CC_BREAK_IF(! font);
|
2011-03-04 17:02:22 +08:00
|
|
|
|
|
|
|
// measure text size with specified font and determine the rectangle to draw text in
|
|
|
|
unsigned uHoriFlag = eAlign & 0x0f;
|
|
|
|
unsigned uVertFlag = (eAlign & 0xf0) >> 4;
|
|
|
|
|
|
|
|
CGSize textSize; // the size which total text layout need
|
|
|
|
CGSize canvasSize; // the size which malloc for drawtext
|
|
|
|
CGRect textRect; // the rectangle which draw text in
|
|
|
|
|
|
|
|
if (0 >= pInfo->width)
|
2011-02-23 18:22:05 +08:00
|
|
|
{
|
2011-03-04 17:02:22 +08:00
|
|
|
// the content width no limit, use 0x7fffffff as a big enough bounds
|
|
|
|
CGSize contentSize = CGSizeMake(0x7fffffff, 0x7fffffff);
|
|
|
|
textSize = [string sizeWithFont:font constrainedToSize:contentSize];
|
|
|
|
|
|
|
|
canvasSize = textSize;
|
|
|
|
pInfo->width = (size_t) textSize.width;
|
|
|
|
pInfo->height = (size_t) textSize.height;
|
|
|
|
|
|
|
|
textRect = CGRectMake(0, 0, textSize.width, textSize.height);
|
2011-02-23 18:22:05 +08:00
|
|
|
}
|
2011-03-04 17:02:22 +08:00
|
|
|
else
|
2011-02-23 18:22:05 +08:00
|
|
|
{
|
2011-03-04 17:02:22 +08:00
|
|
|
// the content height no limit, use 0x7fffffff as a big enough height
|
|
|
|
CGSize contentSize = CGSizeMake(pInfo->width, 0x7fffffff);
|
|
|
|
textSize = [string sizeWithFont:font constrainedToSize:contentSize];
|
|
|
|
|
|
|
|
if (0 >= pInfo->height)
|
|
|
|
{
|
|
|
|
canvasSize = textSize;
|
|
|
|
pInfo->height = (size_t) textSize.height;
|
|
|
|
|
|
|
|
textRect = CGRectMake(0, 0, textSize.width, textSize.height);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
canvasSize.width = pInfo->width;
|
|
|
|
if (textSize.height <= pInfo->height)
|
|
|
|
{
|
|
|
|
canvasSize.height = pInfo->height;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// the height of text larger than the specified
|
|
|
|
// let canvas's height larger a little,
|
|
|
|
// make sure the last line of text in specified rectangle draw in canvas
|
|
|
|
size_t lineHeight = (size_t)font.lineHeight;
|
|
|
|
canvasSize.height = ((pInfo->height + lineHeight - 1) / lineHeight) * lineHeight;
|
|
|
|
}
|
2011-02-23 18:22:05 +08:00
|
|
|
|
2011-03-04 17:02:22 +08:00
|
|
|
canvasSize = CGSizeMake(pInfo->width, MAX(pInfo->height, (size_t)textSize.height));
|
|
|
|
|
|
|
|
textRect.size = textSize;
|
|
|
|
textRect.origin.x = (1 == uHoriFlag) ? 0 // align to left
|
|
|
|
: (2 == uHoriFlag) ? pInfo->width - textSize.width // align to right
|
|
|
|
: (pInfo->width - textSize.width) / 2; // align to center
|
2011-02-23 18:22:05 +08:00
|
|
|
|
2011-03-04 17:02:22 +08:00
|
|
|
textRect.origin.y = (1 == uVertFlag || textSize.height >= pInfo->height) ? 0 // align to top
|
|
|
|
: (2 == uVertFlag) ? pInfo->height - textSize.height // align to bottom
|
|
|
|
: (pInfo->height - textSize.height) / 2; // align to center
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
// malloc space to store pixels data
|
|
|
|
size_t width = (size_t)canvasSize.width;
|
|
|
|
size_t height = (size_t)canvasSize.height;
|
|
|
|
unsigned char* data = new unsigned char[width * height * 4];
|
2011-03-09 16:19:20 +08:00
|
|
|
CC_BREAK_IF(! data);
|
2011-02-23 18:22:05 +08:00
|
|
|
memset(data, 0, height * width * 4);
|
2011-03-04 17:02:22 +08:00
|
|
|
|
|
|
|
// draw text
|
|
|
|
CGColorSpaceRef colorSpace = CGColorSpaceCreateDeviceRGB();
|
|
|
|
|
|
|
|
CGContextRef context = CGBitmapContextCreate(data, width, height, 8, width * 4, colorSpace, kCGImageAlphaPremultipliedLast | kCGBitmapByteOrder32Big);
|
2011-02-23 18:22:05 +08:00
|
|
|
CGColorSpaceRelease(colorSpace);
|
2011-05-12 08:24:29 +08:00
|
|
|
CC_BREAK_IF(!context);
|
2011-03-04 17:02:22 +08:00
|
|
|
|
2011-02-23 18:22:05 +08:00
|
|
|
CGContextSetRGBFillColor(context, 1, 1, 1, 1);
|
|
|
|
CGContextTranslateCTM(context, 0.0f, height);
|
|
|
|
CGContextScaleCTM(context, 1.0f, -1.0f); //NOTE: NSString draws in UIKit referential i.e. renders upside-down compared to CGBitmapContext referential
|
|
|
|
UIGraphicsPushContext(context);
|
2011-03-04 17:02:22 +08:00
|
|
|
|
|
|
|
UITextAlignment align = (2 == uHoriFlag) ? UITextAlignmentRight
|
|
|
|
: (3 == uHoriFlag) ? UITextAlignmentCenter
|
|
|
|
: UITextAlignmentLeft;
|
|
|
|
[string drawInRect:textRect withFont:font lineBreakMode:UILineBreakModeWordWrap alignment:align];
|
|
|
|
|
2011-02-23 18:22:05 +08:00
|
|
|
UIGraphicsPopContext();
|
2011-03-04 17:02:22 +08:00
|
|
|
|
2011-02-23 18:22:05 +08:00
|
|
|
CGContextRelease(context);
|
2011-03-04 17:02:22 +08:00
|
|
|
|
2011-02-23 18:22:05 +08:00
|
|
|
// output params
|
|
|
|
pInfo->data = data;
|
|
|
|
pInfo->hasAlpha = true;
|
|
|
|
pInfo->isPremultipliedAlpha = true;
|
|
|
|
pInfo->bitsPerComponent = 8;
|
2011-03-04 17:02:22 +08:00
|
|
|
bRet = true;
|
|
|
|
} while (0);
|
|
|
|
|
|
|
|
return bRet;
|
2011-02-23 18:22:05 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
NS_CC_BEGIN;
|
|
|
|
|
2011-03-09 16:19:20 +08:00
|
|
|
CCImage::CCImage()
|
2011-02-23 18:22:05 +08:00
|
|
|
: m_nWidth(0)
|
|
|
|
, m_nHeight(0)
|
|
|
|
, m_nBitsPerComponent(0)
|
2011-03-28 13:41:14 +08:00
|
|
|
, m_pData(0)
|
2011-02-23 18:22:05 +08:00
|
|
|
, m_bHasAlpha(false)
|
|
|
|
, m_bPreMulti(false)
|
|
|
|
{
|
2011-03-04 17:02:22 +08:00
|
|
|
|
2011-02-23 18:22:05 +08:00
|
|
|
}
|
|
|
|
|
2011-03-28 13:41:14 +08:00
|
|
|
CCImage::~CCImage()
|
|
|
|
{
|
|
|
|
CC_SAFE_DELETE_ARRAY(m_pData);
|
|
|
|
}
|
|
|
|
|
2011-03-09 16:19:20 +08:00
|
|
|
bool CCImage::initWithImageFile(const char * strPath, EImageFormat eImgFmt/* = eFmtPng*/)
|
2011-02-23 18:22:05 +08:00
|
|
|
{
|
2011-03-29 17:32:52 +08:00
|
|
|
CCFileData data(CCFileUtils::fullPathFromRelativePath(strPath), "rb");
|
|
|
|
return initWithImageData(data.getBuffer(), data.getSize(), eImgFmt);
|
2011-02-23 18:22:05 +08:00
|
|
|
}
|
|
|
|
|
2011-03-09 16:19:20 +08:00
|
|
|
bool CCImage::initWithImageData(void * pData, int nDataLen, EImageFormat eFmt/* = eSrcFmtPng*/)
|
2011-02-23 18:22:05 +08:00
|
|
|
{
|
2011-03-04 17:02:22 +08:00
|
|
|
bool bRet = false;
|
|
|
|
tImageInfo info = {0};
|
|
|
|
do
|
|
|
|
{
|
2011-03-09 16:19:20 +08:00
|
|
|
CC_BREAK_IF(! pData || nDataLen <= 0);
|
2011-03-04 17:02:22 +08:00
|
|
|
bRet = _initWithData(pData, nDataLen, &info);
|
|
|
|
} while (0);
|
|
|
|
if (bRet)
|
|
|
|
{
|
2011-03-28 13:41:14 +08:00
|
|
|
m_nHeight = (short)info.height;
|
|
|
|
m_nWidth = (short)info.width;
|
2011-02-23 18:22:05 +08:00
|
|
|
m_nBitsPerComponent = info.bitsPerComponent;
|
|
|
|
m_bHasAlpha = info.hasAlpha;
|
|
|
|
m_bPreMulti = info.isPremultipliedAlpha;
|
2011-03-28 13:41:14 +08:00
|
|
|
m_pData = info.data;
|
2011-03-04 17:02:22 +08:00
|
|
|
}
|
|
|
|
return bRet;
|
|
|
|
}
|
2011-02-23 18:22:05 +08:00
|
|
|
|
2011-03-09 16:19:20 +08:00
|
|
|
bool CCImage::initWithString(
|
2011-03-04 17:02:22 +08:00
|
|
|
const char * pText,
|
|
|
|
int nWidth /* = 0 */,
|
|
|
|
int nHeight /* = 0 */,
|
|
|
|
ETextAlign eAlignMask /* = kAlignCenter */,
|
|
|
|
const char * pFontName /* = nil */,
|
|
|
|
int nSize /* = 0 */)
|
|
|
|
{
|
|
|
|
tImageInfo info = {0};
|
|
|
|
info.width = nWidth;
|
|
|
|
info.height = nHeight;
|
|
|
|
|
|
|
|
if (! _initWithString(pText, eAlignMask, pFontName, nSize, &info))
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
2011-03-28 13:41:14 +08:00
|
|
|
m_nHeight = (short)info.height;
|
|
|
|
m_nWidth = (short)info.width;
|
2011-03-04 17:02:22 +08:00
|
|
|
m_nBitsPerComponent = info.bitsPerComponent;
|
|
|
|
m_bHasAlpha = info.hasAlpha;
|
|
|
|
m_bPreMulti = info.isPremultipliedAlpha;
|
2011-03-28 13:41:14 +08:00
|
|
|
m_pData = info.data;
|
2011-03-04 17:02:22 +08:00
|
|
|
|
|
|
|
return true;
|
2011-02-23 18:22:05 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
NS_CC_END;
|
|
|
|
|