2013-04-09 09:21:53 +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.
|
|
|
|
****************************************************************************/
|
|
|
|
#define __CC_PLATFORM_IMAGE_CPP__
|
|
|
|
#include "platform/CCImageCommon_cpp.h"
|
|
|
|
|
|
|
|
#include <string.h>
|
|
|
|
#include <vector>
|
|
|
|
#include <string>
|
|
|
|
#include <sstream>
|
|
|
|
#include <algorithm>
|
|
|
|
#include "platform/CCImage.h"
|
|
|
|
#include "platform/CCFileUtils.h"
|
|
|
|
#include "platform/CCCommon.h"
|
|
|
|
#include "CCStdC.h"
|
|
|
|
#include <map>
|
2013-05-15 12:38:56 +08:00
|
|
|
#include <SDL/SDL.h>
|
|
|
|
#include <SDL/SDL_ttf.h>
|
2013-04-09 09:21:53 +08:00
|
|
|
|
|
|
|
#define szFont_kenning 2
|
|
|
|
|
|
|
|
#define RSHIFT6(num) ((num)>>6)
|
|
|
|
|
|
|
|
using namespace std;
|
|
|
|
|
|
|
|
struct TextLine {
|
|
|
|
std::string sLineStr;
|
|
|
|
int iLineWidth;
|
|
|
|
};
|
|
|
|
|
|
|
|
NS_CC_BEGIN
|
|
|
|
|
|
|
|
class BitmapDC
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
BitmapDC()
|
|
|
|
{
|
|
|
|
iInterval = szFont_kenning;
|
|
|
|
m_pData = NULL;
|
|
|
|
reset();
|
|
|
|
}
|
|
|
|
|
|
|
|
~BitmapDC(void)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void reset() {
|
|
|
|
iMaxLineWidth = 0;
|
|
|
|
iMaxLineHeight = 0;
|
|
|
|
vLines.clear();
|
|
|
|
}
|
|
|
|
|
2013-05-15 12:38:56 +08:00
|
|
|
void buildLine(std::stringstream& ss, TTF_Font *face)
|
2013-04-09 09:21:53 +08:00
|
|
|
{
|
|
|
|
TextLine oTempLine;
|
|
|
|
ss << '\0';
|
|
|
|
oTempLine.sLineStr = ss.str();
|
|
|
|
|
2013-05-15 12:38:56 +08:00
|
|
|
int w, h;
|
|
|
|
TTF_SizeText(face, oTempLine.sLineStr.c_str(), &w, &h);
|
|
|
|
|
|
|
|
oTempLine.iLineWidth = w;
|
2013-04-09 09:21:53 +08:00
|
|
|
|
|
|
|
iMaxLineWidth = MAX(iMaxLineWidth, oTempLine.iLineWidth);
|
|
|
|
ss.clear();
|
|
|
|
ss.str("");
|
|
|
|
vLines.push_back(oTempLine);
|
|
|
|
}
|
|
|
|
|
2013-05-15 12:38:56 +08:00
|
|
|
bool divideString(TTF_Font *face, const char* sText, int iMaxWidth, int iMaxHeight) {
|
2013-04-09 09:21:53 +08:00
|
|
|
const char* pText = sText;
|
2013-05-15 12:38:56 +08:00
|
|
|
|
2013-04-09 09:21:53 +08:00
|
|
|
//init stringstream
|
|
|
|
std::stringstream ss;
|
2013-05-15 12:38:56 +08:00
|
|
|
int w, h;
|
2013-04-09 09:21:53 +08:00
|
|
|
|
|
|
|
while (*pText != '\0') {
|
|
|
|
if (*pText == '\n') {
|
2013-05-15 12:38:56 +08:00
|
|
|
buildLine(ss, face);
|
2013-04-09 09:21:53 +08:00
|
|
|
|
|
|
|
pText++;
|
|
|
|
continue;
|
|
|
|
}
|
2013-05-15 12:38:56 +08:00
|
|
|
|
2013-04-09 09:21:53 +08:00
|
|
|
//check its width
|
|
|
|
//divide it when exceeding
|
2013-05-15 12:38:56 +08:00
|
|
|
std::string s = ss.str();
|
|
|
|
s.push_back(*pText);
|
|
|
|
TTF_SizeText(face, s.c_str(), &w, &h);
|
2013-04-09 09:21:53 +08:00
|
|
|
|
2013-05-15 12:38:56 +08:00
|
|
|
if (iMaxWidth > 0 && (w > iMaxWidth)) {
|
|
|
|
buildLine(ss, face);
|
2013-04-09 09:21:53 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
ss << *pText;
|
|
|
|
pText++;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2013-05-15 12:38:56 +08:00
|
|
|
buildLine(ss, face);
|
2013-04-09 09:21:53 +08:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* compute the start pos of every line
|
|
|
|
*
|
|
|
|
* return >0 represent the start x pos of the line
|
|
|
|
* while -1 means fail
|
|
|
|
*
|
|
|
|
*/
|
2013-05-15 12:38:56 +08:00
|
|
|
int computeLineStart(TTF_Font *face, CCImage::ETextAlign eAlignMask, char cText,
|
2013-04-09 09:21:53 +08:00
|
|
|
int iLineIndex) {
|
2013-05-15 12:38:56 +08:00
|
|
|
return 0;
|
|
|
|
/*
|
2013-04-09 09:21:53 +08:00
|
|
|
int iRet;
|
|
|
|
int iError = FT_Load_Glyph(face, FT_Get_Char_Index(face, cText),
|
|
|
|
FT_LOAD_DEFAULT);
|
|
|
|
if (iError) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (eAlignMask == CCImage::kAlignCenter) {
|
|
|
|
iRet = (iMaxLineWidth - vLines[iLineIndex].iLineWidth) / 2
|
|
|
|
- RSHIFT6(face->glyph->metrics.horiBearingX );
|
|
|
|
|
|
|
|
} else if (eAlignMask == CCImage::kAlignRight) {
|
|
|
|
iRet = (iMaxLineWidth - vLines[iLineIndex].iLineWidth)
|
|
|
|
- RSHIFT6(face->glyph->metrics.horiBearingX );
|
|
|
|
} else {
|
|
|
|
// left or other situation
|
|
|
|
iRet = -RSHIFT6(face->glyph->metrics.horiBearingX );
|
|
|
|
}
|
|
|
|
return iRet;
|
2013-05-15 12:38:56 +08:00
|
|
|
*/
|
2013-04-09 09:21:53 +08:00
|
|
|
}
|
|
|
|
|
2013-05-15 12:38:56 +08:00
|
|
|
int computeLineStartY( TTF_Font *face, CCImage::ETextAlign eAlignMask, int txtHeight, int borderHeight ){
|
|
|
|
return 0;
|
|
|
|
/*
|
2013-04-09 09:21:53 +08:00
|
|
|
int iRet;
|
|
|
|
if (eAlignMask == CCImage::kAlignCenter || eAlignMask == CCImage::kAlignLeft ||
|
|
|
|
eAlignMask == CCImage::kAlignRight ) {
|
|
|
|
//vertical center
|
|
|
|
iRet = (borderHeight - txtHeight)/2 + RSHIFT6(face->size->metrics.ascender);
|
|
|
|
|
|
|
|
} else if (eAlignMask == CCImage::kAlignBottomRight ||
|
|
|
|
eAlignMask == CCImage::kAlignBottom ||
|
|
|
|
eAlignMask == CCImage::kAlignBottomLeft ) {
|
|
|
|
//vertical bottom
|
|
|
|
iRet = borderHeight - txtHeight + RSHIFT6(face->size->metrics.ascender);
|
|
|
|
} else {
|
|
|
|
// left or other situation
|
|
|
|
iRet = RSHIFT6(face->size->metrics.ascender);
|
|
|
|
}
|
|
|
|
return iRet;
|
2013-05-15 12:38:56 +08:00
|
|
|
*/
|
2013-04-09 09:21:53 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
bool getBitmap(const char *text, int nWidth, int nHeight, CCImage::ETextAlign eAlignMask, const char * pFontName, float fontSize) {
|
|
|
|
const char* pText = text;
|
2013-05-15 12:38:56 +08:00
|
|
|
// No need to release m_pData here as it is destroyed by CCImage.
|
2013-04-09 09:21:53 +08:00
|
|
|
|
|
|
|
unsigned char cTemp ;
|
|
|
|
int iY, iX, iTemp ;
|
|
|
|
uint32 offset, rowOffset ;
|
|
|
|
|
2013-05-15 12:38:56 +08:00
|
|
|
int iCurXCursor;
|
2013-04-09 09:21:53 +08:00
|
|
|
|
2013-05-15 12:38:56 +08:00
|
|
|
TTF_Font *face = TTF_OpenFont(pFontName, fontSize);
|
|
|
|
if(!face)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
2013-04-09 09:21:53 +08:00
|
|
|
|
2013-05-15 12:38:56 +08:00
|
|
|
divideString(face, text, nWidth, nHeight);
|
2013-04-09 09:21:53 +08:00
|
|
|
|
2013-05-15 12:38:56 +08:00
|
|
|
//compute the final line width
|
|
|
|
iMaxLineWidth = MAX(iMaxLineWidth, nWidth);
|
2013-04-09 09:21:53 +08:00
|
|
|
|
2013-05-15 12:38:56 +08:00
|
|
|
iMaxLineHeight = (int)fontSize;
|
|
|
|
iMaxLineHeight *= vLines.size();
|
2013-04-09 09:21:53 +08:00
|
|
|
|
2013-05-15 12:38:56 +08:00
|
|
|
//compute the final line height
|
|
|
|
iMaxLineHeight = MAX(iMaxLineHeight, nHeight);
|
2013-04-09 09:21:53 +08:00
|
|
|
|
2013-05-15 12:38:56 +08:00
|
|
|
uint bitmapSize = iMaxLineWidth * iMaxLineHeight*4 ;
|
2013-04-09 09:21:53 +08:00
|
|
|
|
2013-05-15 12:38:56 +08:00
|
|
|
m_pData = new unsigned char[bitmapSize];
|
|
|
|
memset(m_pData, 0, bitmapSize);
|
2013-04-09 09:21:53 +08:00
|
|
|
|
2013-05-15 12:38:56 +08:00
|
|
|
for (size_t l = 0; l < vLines.size(); l++) {
|
|
|
|
pText = vLines[l].sLineStr.c_str();
|
|
|
|
//initialize the origin cursor
|
|
|
|
iCurXCursor = computeLineStart(face, eAlignMask, *pText, l);
|
2013-04-09 09:21:53 +08:00
|
|
|
|
2013-05-15 12:38:56 +08:00
|
|
|
SDL_Color color = { 0xff, 0xff, 0xff, 0xff };
|
|
|
|
SDL_Surface *tSurf = TTF_RenderText_Solid(face, pText, color);
|
|
|
|
SDL_LockSurface(tSurf);
|
|
|
|
SDL_UnlockSurface(tSurf);
|
2013-04-09 09:21:53 +08:00
|
|
|
|
2013-05-15 12:38:56 +08:00
|
|
|
// We treat pixels as 32-bit words, since both source and target
|
|
|
|
// are rendered as such.
|
|
|
|
int *pixels = (int*) tSurf->pixels;
|
2013-04-09 09:21:53 +08:00
|
|
|
|
2013-05-15 12:38:56 +08:00
|
|
|
for(int i = 0; i < tSurf->h; ++i)
|
|
|
|
{
|
|
|
|
for(int j = 0; j < tSurf->w; ++j)
|
|
|
|
{
|
|
|
|
int targetOffset = (l * iMaxLineHeight + i) * iMaxLineWidth + j;
|
|
|
|
int sourceOffset = i * tSurf->w + j;
|
|
|
|
*(int*) &m_pData[targetOffset] = pixels[sourceOffset];
|
2013-04-09 09:21:53 +08:00
|
|
|
}
|
|
|
|
}
|
2013-05-15 12:38:56 +08:00
|
|
|
SDL_FreeSurface(tSurf);
|
|
|
|
}
|
2013-04-09 09:21:53 +08:00
|
|
|
|
2013-05-15 12:38:56 +08:00
|
|
|
//clear all lines
|
|
|
|
vLines.clear();
|
2013-04-09 09:21:53 +08:00
|
|
|
|
2013-05-15 12:38:56 +08:00
|
|
|
TTF_CloseFont(face);
|
2013-04-09 09:21:53 +08:00
|
|
|
|
2013-05-15 12:38:56 +08:00
|
|
|
return true;
|
2013-04-09 09:21:53 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
public:
|
|
|
|
unsigned char *m_pData;
|
|
|
|
int libError;
|
|
|
|
vector<TextLine> vLines;
|
|
|
|
int iInterval;
|
|
|
|
int iMaxLineWidth;
|
|
|
|
int iMaxLineHeight;
|
|
|
|
};
|
|
|
|
|
|
|
|
static BitmapDC& sharedBitmapDC()
|
|
|
|
{
|
|
|
|
static BitmapDC s_BmpDC;
|
|
|
|
return s_BmpDC;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CCImage::initWithString(
|
|
|
|
const char * pText,
|
|
|
|
int nWidth/* = 0*/,
|
|
|
|
int nHeight/* = 0*/,
|
|
|
|
ETextAlign eAlignMask/* = kAlignCenter*/,
|
|
|
|
const char * pFontName/* = nil*/,
|
|
|
|
int nSize/* = 0*/)
|
|
|
|
{
|
|
|
|
bool bRet = false;
|
|
|
|
do
|
|
|
|
{
|
|
|
|
CC_BREAK_IF(! pText);
|
|
|
|
BitmapDC &dc = sharedBitmapDC();
|
|
|
|
|
|
|
|
std::string fullFontName = pFontName;
|
|
|
|
std::string lowerCasePath = fullFontName;
|
|
|
|
std::transform(lowerCasePath.begin(), lowerCasePath.end(), lowerCasePath.begin(), ::tolower);
|
|
|
|
|
|
|
|
if ( lowerCasePath.find(".ttf") != std::string::npos ) {
|
|
|
|
fullFontName = CCFileUtils::sharedFileUtils()->fullPathForFilename(pFontName);
|
|
|
|
}
|
|
|
|
//CCLog("-----pText=%s and Font File is %s nWidth= %d,nHeight=%d",pText,fullFontName.c_str(),nWidth,nHeight);
|
|
|
|
|
2013-05-15 12:38:56 +08:00
|
|
|
CC_BREAK_IF(! dc.getBitmap(pText, nWidth, nHeight, eAlignMask, fullFontName.c_str(), nSize * 3));
|
2013-04-09 09:21:53 +08:00
|
|
|
//CCLog("---- dc.getBitmap is Succesfull... \n");
|
|
|
|
|
|
|
|
// assign the dc.m_pData to m_pData in order to save time
|
|
|
|
m_pData = dc.m_pData;
|
|
|
|
CC_BREAK_IF(! m_pData);
|
|
|
|
|
|
|
|
m_nWidth = (short)dc.iMaxLineWidth;
|
|
|
|
m_nHeight = (short)dc.iMaxLineHeight;
|
|
|
|
m_bHasAlpha = true;
|
|
|
|
m_bPreMulti = true;
|
|
|
|
m_nBitsPerComponent = 8;
|
|
|
|
|
|
|
|
bRet = true;
|
|
|
|
|
|
|
|
dc.reset();
|
|
|
|
|
|
|
|
} while (0);
|
|
|
|
|
|
|
|
return bRet;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_CC_END
|
|
|
|
|