2019-11-23 20:27:39 +08:00
|
|
|
/****************************************************************************
|
|
|
|
Copyright (c) 2013 Zynga Inc.
|
|
|
|
Copyright (c) 2013-2016 Chukong Technologies Inc.
|
|
|
|
Copyright (c) 2017-2018 Xiamen Yaji Software Co., Ltd.
|
2021-10-11 12:15:41 +08:00
|
|
|
Copyright (c) 2021 Bytedance Inc.
|
|
|
|
|
|
|
|
https://adxe.org
|
2019-11-23 20:27:39 +08: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.
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
#include "2d/CCFontFNT.h"
|
|
|
|
#include "2d/CCFontAtlas.h"
|
2020-01-06 09:35:52 +08:00
|
|
|
#include "2d/CCSpriteFrameCache.h"
|
2019-11-23 20:27:39 +08:00
|
|
|
#include "platform/CCFileUtils.h"
|
|
|
|
#include "base/CCConfiguration.h"
|
|
|
|
#include "base/CCDirector.h"
|
|
|
|
#include "base/CCMap.h"
|
|
|
|
#include "base/ccUTF8.h"
|
|
|
|
#include "renderer/CCTextureCache.h"
|
|
|
|
|
|
|
|
#include <cmath>
|
|
|
|
|
|
|
|
NS_CC_BEGIN
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @addtogroup GUI
|
|
|
|
* @{
|
|
|
|
* @addtogroup label
|
|
|
|
* @{
|
|
|
|
*/
|
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
enum
|
|
|
|
{
|
2019-11-23 20:27:39 +08:00
|
|
|
kLabelAutomaticWidth = -1,
|
|
|
|
};
|
|
|
|
|
|
|
|
struct _FontDefHashElement;
|
|
|
|
|
|
|
|
//
|
2021-12-25 10:04:45 +08:00
|
|
|
// FNTConfig Cache - free functions
|
2019-11-23 20:27:39 +08:00
|
|
|
//
|
2021-12-26 23:26:34 +08:00
|
|
|
static StringMap<BMFontConfiguration*>* s_configurations = nullptr;
|
2019-11-23 20:27:39 +08:00
|
|
|
|
2021-12-26 23:26:34 +08:00
|
|
|
BMFontConfiguration* FNTConfigLoadFile(std::string_view fntFile)
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
BMFontConfiguration* ret = nullptr;
|
|
|
|
|
|
|
|
if (s_configurations == nullptr)
|
|
|
|
{
|
2021-12-26 23:26:34 +08:00
|
|
|
s_configurations = new StringMap<BMFontConfiguration*>();
|
2019-11-23 20:27:39 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
ret = s_configurations->at(fntFile);
|
2021-12-25 10:04:45 +08:00
|
|
|
if (ret == nullptr)
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
ret = BMFontConfiguration::create(fntFile);
|
|
|
|
if (ret)
|
|
|
|
{
|
|
|
|
s_configurations->insert(fntFile, ret);
|
2021-12-25 10:04:45 +08:00
|
|
|
}
|
2019-11-23 20:27:39 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
2021-12-25 10:04:45 +08:00
|
|
|
// BitmapFontConfiguration
|
2019-11-23 20:27:39 +08:00
|
|
|
//
|
|
|
|
|
2021-12-26 23:26:34 +08:00
|
|
|
BMFontConfiguration* BMFontConfiguration::create(std::string_view FNTfile)
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
2021-12-25 10:04:45 +08:00
|
|
|
BMFontConfiguration* ret = new BMFontConfiguration();
|
2019-11-23 20:27:39 +08:00
|
|
|
if (ret->initWithFNTfile(FNTfile))
|
|
|
|
{
|
|
|
|
ret->autorelease();
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
CC_SAFE_DELETE(ret);
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2021-12-26 23:26:34 +08:00
|
|
|
bool BMFontConfiguration::initWithFNTfile(std::string_view FNTfile)
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
_characterSet = this->parseConfigFile(FNTfile);
|
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
if (!_characterSet)
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::set<unsigned int>* BMFontConfiguration::getCharacterSet() const
|
|
|
|
{
|
|
|
|
return _characterSet;
|
|
|
|
}
|
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
BMFontConfiguration::BMFontConfiguration() : _commonHeight(0), _characterSet(nullptr), _fontSize(0) {}
|
2019-11-23 20:27:39 +08:00
|
|
|
|
|
|
|
BMFontConfiguration::~BMFontConfiguration()
|
|
|
|
{
|
2021-12-25 10:04:45 +08:00
|
|
|
CCLOGINFO("deallocing BMFontConfiguration: %p", this);
|
2019-11-23 20:27:39 +08:00
|
|
|
this->purgeFontDefDictionary();
|
|
|
|
this->purgeKerningDictionary();
|
|
|
|
_atlasName.clear();
|
|
|
|
CC_SAFE_DELETE(_characterSet);
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string BMFontConfiguration::description() const
|
|
|
|
{
|
|
|
|
return StringUtils::format(
|
2021-12-25 10:04:45 +08:00
|
|
|
"<BMFontConfiguration = " CC_FORMAT_PRINTF_SIZE_T " | Glphys:%d Kernings:%d | Image = %s>", (size_t)this,
|
|
|
|
static_cast<int>(_fontDefDictionary.size()), static_cast<int>(_kerningDictionary.size()), _atlasName.c_str());
|
2019-11-23 20:27:39 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void BMFontConfiguration::purgeKerningDictionary()
|
|
|
|
{
|
|
|
|
_kerningDictionary.clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
void BMFontConfiguration::purgeFontDefDictionary()
|
|
|
|
{
|
|
|
|
_fontDefDictionary.clear();
|
|
|
|
}
|
|
|
|
|
2021-12-26 23:26:34 +08:00
|
|
|
std::set<unsigned int>* BMFontConfiguration::parseConfigFile(std::string_view controlFile)
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
std::string data = FileUtils::getInstance()->getStringFromFile(controlFile);
|
|
|
|
if (data.empty())
|
|
|
|
{
|
|
|
|
return nullptr;
|
|
|
|
}
|
2021-12-25 10:04:45 +08:00
|
|
|
if (data.size() >= (sizeof("BMP") - 1) && memcmp("BMF", data.c_str(), sizeof("BMP") - 1) == 0)
|
|
|
|
{
|
2019-11-23 20:27:39 +08:00
|
|
|
// Handle fnt file of binary format
|
|
|
|
std::set<unsigned int>* ret = parseBinaryConfigFile((unsigned char*)&data.front(), data.size(), controlFile);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
if (data[0] == 0)
|
|
|
|
{
|
2021-12-26 23:26:34 +08:00
|
|
|
CCLOG("cocos2d: Error parsing FNTfile %s", controlFile.data());
|
2019-11-23 20:27:39 +08:00
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
auto contents = data.c_str();
|
2021-12-25 10:04:45 +08:00
|
|
|
|
|
|
|
std::set<unsigned int>* validCharsString = new std::set<unsigned int>();
|
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
auto contentsLen = strlen(contents);
|
2021-12-25 10:04:45 +08:00
|
|
|
char line[512] = {0};
|
|
|
|
|
|
|
|
auto next = strchr(contents, '\n');
|
|
|
|
auto base = contents;
|
2019-11-23 20:27:39 +08:00
|
|
|
size_t parseCount = 0;
|
|
|
|
while (next)
|
|
|
|
{
|
2021-10-11 12:15:41 +08:00
|
|
|
size_t lineLength = ((int)(next - base));
|
2019-11-23 20:27:39 +08:00
|
|
|
memcpy(line, contents + parseCount, lineLength);
|
|
|
|
line[lineLength] = 0;
|
|
|
|
|
|
|
|
parseCount += lineLength + 1;
|
|
|
|
if (parseCount < contentsLen)
|
|
|
|
{
|
|
|
|
base = next + 1;
|
|
|
|
next = strchr(base, '\n');
|
2021-12-25 10:04:45 +08:00
|
|
|
}
|
2019-11-23 20:27:39 +08:00
|
|
|
else
|
|
|
|
{
|
|
|
|
next = nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (memcmp(line, "info face", 9) == 0)
|
|
|
|
{
|
|
|
|
// FIXME: info parsing is incomplete
|
|
|
|
// Not needed for the Hiero editors, but needed for the AngelCode editor
|
|
|
|
// [self parseInfoArguments:line];
|
|
|
|
this->parseInfoArguments(line);
|
|
|
|
}
|
|
|
|
// Check to see if the start of the line is something we are interested in
|
|
|
|
else if (memcmp(line, "common lineHeight", 17) == 0)
|
|
|
|
{
|
|
|
|
this->parseCommonArguments(line);
|
|
|
|
}
|
|
|
|
else if (memcmp(line, "page id", 7) == 0)
|
|
|
|
{
|
|
|
|
this->parseImageFileName(line, controlFile);
|
|
|
|
}
|
|
|
|
else if (memcmp(line, "chars c", 7) == 0)
|
|
|
|
{
|
|
|
|
// Ignore this line
|
|
|
|
}
|
|
|
|
else if (memcmp(line, "char", 4) == 0)
|
|
|
|
{
|
|
|
|
// Parse the current line and create a new CharDef
|
|
|
|
unsigned int charID = this->parseCharacterDefinition(line);
|
|
|
|
validCharsString->insert(charID);
|
|
|
|
}
|
|
|
|
else if (memcmp(line, "kerning first", 13) == 0)
|
|
|
|
{
|
|
|
|
this->parseKerningEntry(line);
|
|
|
|
}
|
|
|
|
}
|
2021-12-25 10:04:45 +08:00
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
return validCharsString;
|
|
|
|
}
|
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
std::set<unsigned int>* BMFontConfiguration::parseBinaryConfigFile(unsigned char* pData,
|
|
|
|
uint32_t size,
|
2021-12-26 23:26:34 +08:00
|
|
|
std::string_view controlFile)
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
/* based on http://www.angelcode.com/products/bmfont/doc/file_format.html file format */
|
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
std::set<unsigned int>* validCharsString = new std::set<unsigned int>();
|
2019-11-23 20:27:39 +08:00
|
|
|
|
2021-09-05 20:45:32 +08:00
|
|
|
uint32_t remains = size;
|
2019-11-23 20:27:39 +08:00
|
|
|
|
|
|
|
CCASSERT(pData[3] == 3, "Only version 3 is supported");
|
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
pData += 4;
|
|
|
|
remains -= 4;
|
2019-11-23 20:27:39 +08:00
|
|
|
|
|
|
|
while (remains > 0)
|
|
|
|
{
|
2021-12-25 10:04:45 +08:00
|
|
|
unsigned char blockId = pData[0];
|
|
|
|
pData += 1;
|
|
|
|
remains -= 1;
|
|
|
|
uint32_t blockSize = 0;
|
|
|
|
memcpy(&blockSize, pData, 4);
|
2019-11-23 20:27:39 +08:00
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
pData += 4;
|
|
|
|
remains -= 4;
|
2019-11-23 20:27:39 +08:00
|
|
|
|
|
|
|
if (blockId == 1)
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
fontSize 2 int 0
|
2021-12-25 10:04:45 +08:00
|
|
|
bitField 1 bits 2 bit 0: smooth, bit 1: unicode, bit 2: italic, bit 3: bold, bit 4:
|
|
|
|
fixedHeight, bits 5-7: reserved charSet 1 uint 3 stretchH 2 uint 4 aa 1 uint 6
|
2019-11-23 20:27:39 +08:00
|
|
|
paddingUp 1 uint 7
|
|
|
|
paddingRight 1 uint 8
|
|
|
|
paddingDown 1 uint 9
|
|
|
|
paddingLeft 1 uint 10
|
|
|
|
spacingHoriz 1 uint 11
|
|
|
|
spacingVert 1 uint 12
|
|
|
|
outline 1 uint 13 added with version 2
|
|
|
|
fontName n+1 string 14 null terminated string with length n
|
|
|
|
*/
|
|
|
|
|
|
|
|
memcpy(&_fontSize, pData, 2);
|
2021-12-25 10:04:45 +08:00
|
|
|
_padding.top = (unsigned char)pData[7];
|
|
|
|
_padding.right = (unsigned char)pData[8];
|
2019-11-23 20:27:39 +08:00
|
|
|
_padding.bottom = (unsigned char)pData[9];
|
2021-12-25 10:04:45 +08:00
|
|
|
_padding.left = (unsigned char)pData[10];
|
2019-11-23 20:27:39 +08:00
|
|
|
}
|
2021-12-25 10:04:45 +08:00
|
|
|
else if (blockId == 2)
|
|
|
|
{
|
2019-11-23 20:27:39 +08:00
|
|
|
/*
|
|
|
|
lineHeight 2 uint 0
|
|
|
|
base 2 uint 2
|
|
|
|
scaleW 2 uint 4
|
|
|
|
scaleH 2 uint 6
|
|
|
|
pages 2 uint 8
|
|
|
|
bitField 1 bits 10 bits 0-6: reserved, bit 7: packed
|
|
|
|
alphaChnl 1 uint 11
|
|
|
|
redChnl 1 uint 12
|
|
|
|
greenChnl 1 uint 13
|
|
|
|
blueChnl 1 uint 14
|
|
|
|
*/
|
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
uint16_t lineHeight = 0;
|
|
|
|
memcpy(&lineHeight, pData, 2);
|
2019-11-23 20:27:39 +08:00
|
|
|
_commonHeight = lineHeight;
|
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
uint16_t scaleW = 0;
|
|
|
|
memcpy(&scaleW, pData + 4, 2);
|
|
|
|
uint16_t scaleH = 0;
|
|
|
|
memcpy(&scaleH, pData + 6, 2);
|
2019-11-23 20:27:39 +08:00
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
CCASSERT(scaleW <= Configuration::getInstance()->getMaxTextureSize() &&
|
|
|
|
scaleH <= Configuration::getInstance()->getMaxTextureSize(),
|
|
|
|
"CCLabelBMFont: page can't be larger than supported");
|
2019-11-23 20:27:39 +08:00
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
uint16_t pages = 0;
|
|
|
|
memcpy(&pages, pData + 8, 2);
|
2019-11-23 20:27:39 +08:00
|
|
|
CCASSERT(pages == 1, "CCBitfontAtlas: only supports 1 page");
|
|
|
|
}
|
|
|
|
else if (blockId == 3)
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
pageNames p*(n+1) strings 0 p null terminated strings, each with length n
|
|
|
|
*/
|
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
const char* value = (const char*)pData;
|
2019-11-23 20:27:39 +08:00
|
|
|
CCASSERT(strlen(value) < blockSize, "Block size should be less then string");
|
|
|
|
|
|
|
|
_atlasName = FileUtils::getInstance()->fullPathFromRelativeFile(value, controlFile);
|
|
|
|
}
|
2021-12-25 10:04:45 +08:00
|
|
|
else if (blockId == 4)
|
|
|
|
{
|
2019-11-23 20:27:39 +08:00
|
|
|
/*
|
|
|
|
id 4 uint 0+c*20 These fields are repeated until all characters have been described
|
|
|
|
x 2 uint 4+c*20
|
|
|
|
y 2 uint 6+c*20
|
|
|
|
width 2 uint 8+c*20
|
|
|
|
height 2 uint 10+c*20
|
|
|
|
xoffset 2 int 12+c*20
|
|
|
|
yoffset 2 int 14+c*20
|
|
|
|
xadvance 2 int 16+c*20
|
|
|
|
page 1 uint 18+c*20
|
|
|
|
chnl 1 uint 19+c*20
|
|
|
|
*/
|
|
|
|
|
2021-09-05 20:45:32 +08:00
|
|
|
uint32_t count = blockSize / 20;
|
2019-11-23 20:27:39 +08:00
|
|
|
|
2021-09-05 20:45:32 +08:00
|
|
|
for (uint32_t i = 0; i < count; i++)
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
2021-12-25 10:04:45 +08:00
|
|
|
uint32_t charId = 0;
|
|
|
|
memcpy(&charId, pData + (i * 20), 4);
|
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
BMFontDef& fontDef = _fontDefDictionary[charId];
|
2021-12-25 10:04:45 +08:00
|
|
|
fontDef.charID = charId;
|
2019-11-23 20:27:39 +08:00
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
uint16_t charX = 0;
|
|
|
|
memcpy(&charX, pData + (i * 20) + 4, 2);
|
2019-11-23 20:27:39 +08:00
|
|
|
fontDef.rect.origin.x = charX;
|
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
uint16_t charY = 0;
|
|
|
|
memcpy(&charY, pData + (i * 20) + 6, 2);
|
2019-11-23 20:27:39 +08:00
|
|
|
fontDef.rect.origin.y = charY;
|
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
uint16_t charWidth = 0;
|
|
|
|
memcpy(&charWidth, pData + (i * 20) + 8, 2);
|
2019-11-23 20:27:39 +08:00
|
|
|
fontDef.rect.size.width = charWidth;
|
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
uint16_t charHeight = 0;
|
|
|
|
memcpy(&charHeight, pData + (i * 20) + 10, 2);
|
2019-11-23 20:27:39 +08:00
|
|
|
fontDef.rect.size.height = charHeight;
|
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
int16_t xoffset = 0;
|
|
|
|
memcpy(&xoffset, pData + (i * 20) + 12, 2);
|
2019-11-23 20:27:39 +08:00
|
|
|
fontDef.xOffset = xoffset;
|
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
int16_t yoffset = 0;
|
|
|
|
memcpy(&yoffset, pData + (i * 20) + 14, 2);
|
2019-11-23 20:27:39 +08:00
|
|
|
fontDef.yOffset = yoffset;
|
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
int16_t xadvance = 0;
|
|
|
|
memcpy(&xadvance, pData + (i * 20) + 16, 2);
|
2019-11-23 20:27:39 +08:00
|
|
|
fontDef.xAdvance = xadvance;
|
|
|
|
|
|
|
|
validCharsString->insert(fontDef.charID);
|
|
|
|
}
|
|
|
|
}
|
2021-12-25 10:04:45 +08:00
|
|
|
else if (blockId == 5)
|
|
|
|
{
|
2019-11-23 20:27:39 +08:00
|
|
|
/*
|
|
|
|
first 4 uint 0+c*10 These fields are repeated until all kerning pairs have been described
|
|
|
|
second 4 uint 4+c*10
|
|
|
|
amount 2 int 8+c*10
|
|
|
|
*/
|
|
|
|
|
2021-09-05 20:45:32 +08:00
|
|
|
uint32_t count = blockSize / 20;
|
2019-11-23 20:27:39 +08:00
|
|
|
|
2021-09-05 20:45:32 +08:00
|
|
|
for (uint32_t i = 0; i < count; i++)
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
2021-12-25 10:04:45 +08:00
|
|
|
uint32_t first = 0;
|
|
|
|
memcpy(&first, pData + (i * 10), 4);
|
|
|
|
uint32_t second = 0;
|
|
|
|
memcpy(&second, pData + (i * 10) + 4, 4);
|
|
|
|
int16_t amount = 0;
|
|
|
|
memcpy(&amount, pData + (i * 10) + 8, 2);
|
|
|
|
|
|
|
|
uint64_t key = ((uint64_t)first << 32) | ((uint64_t)second & 0xffffffffll);
|
2019-11-23 20:27:39 +08:00
|
|
|
_kerningDictionary[key] = amount;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
pData += blockSize;
|
|
|
|
remains -= blockSize;
|
2019-11-23 20:27:39 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
return validCharsString;
|
|
|
|
}
|
|
|
|
|
2021-12-26 23:26:34 +08:00
|
|
|
void BMFontConfiguration::parseImageFileName(const char* line, std::string_view fntFile)
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
// line to parse:
|
|
|
|
// page id=0 file="bitmapFontTest.png"
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
// page ID. Sanity check
|
|
|
|
int pageId;
|
|
|
|
sscanf(line, "page id=%d", &pageId);
|
|
|
|
CCASSERT(pageId == 0, "LabelBMFont file could not be found");
|
2020-01-06 09:35:52 +08:00
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
// file
|
2019-11-23 20:27:39 +08:00
|
|
|
char fileName[255];
|
2020-01-06 09:35:52 +08:00
|
|
|
sscanf(strstr(line, "file=\"") + 6, "%[^\"]", fileName);
|
2019-11-23 20:27:39 +08:00
|
|
|
_atlasName = FileUtils::getInstance()->fullPathFromRelativeFile(fileName, fntFile);
|
|
|
|
}
|
|
|
|
|
|
|
|
void BMFontConfiguration::parseInfoArguments(const char* line)
|
|
|
|
{
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
// possible lines to parse:
|
2021-12-25 10:04:45 +08:00
|
|
|
// info face="Script" size=32 bold=0 italic=0 charset="" unicode=1 stretchH=100 smooth=1 aa=1 padding=1,4,3,2
|
|
|
|
// spacing=0,0 outline=0 info face="Cracked" size=36 bold=0 italic=0 charset="" unicode=0 stretchH=100 smooth=1 aa=1
|
|
|
|
// padding=0,0,0,0 spacing=1,1
|
2019-11-23 20:27:39 +08:00
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
sscanf(strstr(line, "size=") + 5, "%d", &_fontSize);
|
|
|
|
// padding
|
2021-12-25 10:04:45 +08:00
|
|
|
sscanf(strstr(line, "padding=") + 8, "%d,%d,%d,%d", &_padding.top, &_padding.right, &_padding.bottom,
|
|
|
|
&_padding.left);
|
|
|
|
// CCLOG("cocos2d: padding: %d,%d,%d,%d", _padding.left, _padding.top, _padding.right, _padding.bottom);
|
2019-11-23 20:27:39 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void BMFontConfiguration::parseCommonArguments(const char* line)
|
|
|
|
{
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
// line to parse:
|
|
|
|
// common lineHeight=104 base=26 scaleW=1024 scaleH=512 pages=1 packed=0
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
2021-12-25 10:04:45 +08:00
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
// Height
|
|
|
|
auto tmp = strstr(line, "lineHeight=") + 11;
|
|
|
|
sscanf(tmp, "%d", &_commonHeight);
|
2021-12-25 10:04:45 +08:00
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
#if COCOS2D_DEBUG > 0
|
|
|
|
// scaleW. sanity check
|
|
|
|
int value;
|
|
|
|
tmp = strstr(tmp, "scaleW=") + 7;
|
|
|
|
sscanf(tmp, "%d", &value);
|
|
|
|
|
|
|
|
int maxTextureSize = Configuration::getInstance()->getMaxTextureSize();
|
|
|
|
CCASSERT(value <= maxTextureSize, "CCLabelBMFont: page can't be larger than supported");
|
|
|
|
|
|
|
|
// scaleH. sanity check
|
|
|
|
tmp = strstr(tmp, "scaleH=") + 7;
|
|
|
|
sscanf(tmp, "%d", &value);
|
|
|
|
CCASSERT(value <= maxTextureSize, "CCLabelBMFont: page can't be larger than supported");
|
|
|
|
|
|
|
|
// pages. sanity check
|
|
|
|
tmp = strstr(tmp, "pages=") + 6;
|
|
|
|
sscanf(tmp, "%d", &value);
|
|
|
|
CCASSERT(value == 1, "CCBitfontAtlas: only supports 1 page");
|
|
|
|
#endif
|
|
|
|
// packed (ignore) What does this mean ??
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned int BMFontConfiguration::parseCharacterDefinition(const char* line)
|
|
|
|
{
|
|
|
|
unsigned int charID = 0;
|
2021-12-25 10:04:45 +08:00
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
// line to parse:
|
2021-12-25 10:04:45 +08:00
|
|
|
// char id=32 x=0 y=0 width=0 height=0 xoffset=0 yoffset=44 xadvance=14 page=0 chnl=0
|
2019-11-23 20:27:39 +08:00
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
// Character ID
|
|
|
|
auto tmp = strstr(line, "id=") + 3;
|
|
|
|
sscanf(tmp, "%u", &charID);
|
2021-12-25 10:04:45 +08:00
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
BMFontDef& characterDefinition = _fontDefDictionary[charID];
|
2021-12-25 10:04:45 +08:00
|
|
|
characterDefinition.charID = charID;
|
2019-11-23 20:27:39 +08:00
|
|
|
|
|
|
|
// Character x
|
|
|
|
tmp = strstr(tmp, "x=") + 2;
|
|
|
|
sscanf(tmp, "%f", &characterDefinition.rect.origin.x);
|
|
|
|
// Character y
|
|
|
|
tmp = strstr(tmp, "y=") + 2;
|
|
|
|
sscanf(tmp, "%f", &characterDefinition.rect.origin.y);
|
|
|
|
// Character width
|
|
|
|
tmp = strstr(tmp, "width=") + 6;
|
|
|
|
sscanf(tmp, "%f", &characterDefinition.rect.size.width);
|
|
|
|
// Character height
|
|
|
|
tmp = strstr(tmp, "height=") + 7;
|
|
|
|
sscanf(tmp, "%f", &characterDefinition.rect.size.height);
|
|
|
|
// Character xoffset
|
|
|
|
tmp = strstr(tmp, "xoffset=") + 8;
|
|
|
|
sscanf(tmp, "%hd", &characterDefinition.xOffset);
|
|
|
|
// Character yoffset
|
|
|
|
tmp = strstr(tmp, "yoffset=") + 8;
|
|
|
|
sscanf(tmp, "%hd", &characterDefinition.yOffset);
|
|
|
|
// Character xadvance
|
|
|
|
tmp = strstr(tmp, "xadvance=") + 9;
|
|
|
|
sscanf(tmp, "%hd", &characterDefinition.xAdvance);
|
2021-12-25 10:04:45 +08:00
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
return charID;
|
|
|
|
}
|
|
|
|
|
|
|
|
void BMFontConfiguration::parseKerningEntry(const char* line)
|
2021-12-25 10:04:45 +08:00
|
|
|
{
|
2019-11-23 20:27:39 +08:00
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
// line to parse:
|
|
|
|
// kerning first=121 second=44 amount=-7
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
int first, second, amount;
|
|
|
|
auto tmp = strstr(line, "first=") + 6;
|
|
|
|
sscanf(tmp, "%d", &first);
|
|
|
|
|
|
|
|
tmp = strstr(tmp, "second=") + 7;
|
|
|
|
sscanf(tmp, "%d", &second);
|
|
|
|
|
|
|
|
tmp = strstr(tmp, "amount=") + 7;
|
|
|
|
sscanf(tmp, "%d", &amount);
|
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
uint64_t key = ((uint64_t)first << 32) | ((uint64_t)second & 0xffffffffll);
|
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
_kerningDictionary[key] = amount;
|
|
|
|
}
|
|
|
|
|
2021-12-26 23:26:34 +08:00
|
|
|
FontFNT* FontFNT::create(std::string_view fntFilePath, const Rect& imageRect, bool imageRotated)
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
2020-01-06 09:35:52 +08:00
|
|
|
const auto newConf = FNTConfigLoadFile(fntFilePath);
|
2019-11-23 20:27:39 +08:00
|
|
|
if (!newConf)
|
|
|
|
return nullptr;
|
2021-12-25 10:04:45 +08:00
|
|
|
|
2020-01-06 09:35:52 +08:00
|
|
|
const auto tempFont = new FontFNT(newConf, imageRect, imageRotated);
|
2021-12-25 10:04:45 +08:00
|
|
|
|
2020-01-06 09:35:52 +08:00
|
|
|
if (!tempFont)
|
|
|
|
{
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
tempFont->setFontSize((float)newConf->_fontSize);
|
|
|
|
tempFont->autorelease();
|
|
|
|
return tempFont;
|
|
|
|
}
|
|
|
|
|
2021-12-26 23:26:34 +08:00
|
|
|
FontFNT* FontFNT::create(std::string_view fntFilePath, std::string_view subTextureKey)
|
2020-01-06 09:35:52 +08:00
|
|
|
{
|
|
|
|
const auto newConf = FNTConfigLoadFile(fntFilePath);
|
|
|
|
if (!newConf)
|
|
|
|
return nullptr;
|
|
|
|
|
|
|
|
const auto frame = SpriteFrameCache::getInstance()->getSpriteFrameByName(subTextureKey);
|
|
|
|
if (!frame)
|
|
|
|
{
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
auto tempFont = new FontFNT(newConf, frame->getRectInPixels(), frame->isRotated());
|
|
|
|
|
|
|
|
if (!tempFont)
|
|
|
|
{
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
tempFont->setFontSize((float)newConf->_fontSize);
|
|
|
|
tempFont->autorelease();
|
|
|
|
return tempFont;
|
|
|
|
}
|
|
|
|
|
2021-12-26 23:26:34 +08:00
|
|
|
FontFNT* FontFNT::create(std::string_view fntFilePath)
|
2020-01-06 09:35:52 +08:00
|
|
|
{
|
|
|
|
const auto newConf = FNTConfigLoadFile(fntFilePath);
|
|
|
|
if (!newConf)
|
|
|
|
return nullptr;
|
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
// add the texture
|
2020-01-06 09:35:52 +08:00
|
|
|
const auto tempTexture = Director::getInstance()->getTextureCache()->addImage(newConf->getAtlasName());
|
2019-11-23 20:27:39 +08:00
|
|
|
if (!tempTexture)
|
|
|
|
{
|
|
|
|
return nullptr;
|
|
|
|
}
|
2020-01-06 09:35:52 +08:00
|
|
|
FontFNT* tempFont = new FontFNT(newConf);
|
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
tempFont->setFontSize((float)newConf->_fontSize);
|
|
|
|
if (!tempFont)
|
|
|
|
{
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
tempFont->autorelease();
|
|
|
|
return tempFont;
|
|
|
|
}
|
|
|
|
|
2021-12-26 23:26:34 +08:00
|
|
|
FontFNT* FontFNT::create(std::string_view fntFilePath, const Vec2& imageOffset)
|
2020-01-06 09:35:52 +08:00
|
|
|
{
|
|
|
|
return create(fntFilePath, Rect(imageOffset.x, imageOffset.y, 0, 0), false);
|
|
|
|
}
|
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
FontFNT::FontFNT(BMFontConfiguration* theContfig, const Rect& imageRect, bool imageRotated)
|
|
|
|
: _configuration(theContfig), _imageRectInPoints(CC_RECT_PIXELS_TO_POINTS(imageRect)), _imageRotated(imageRotated)
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
_configuration->retain();
|
|
|
|
}
|
|
|
|
|
2020-01-06 09:35:52 +08:00
|
|
|
FontFNT::FontFNT(BMFontConfiguration* theContfig)
|
2021-12-25 10:04:45 +08:00
|
|
|
: _configuration(theContfig), _imageRectInPoints(Rect::ZERO), _imageRotated(false)
|
|
|
|
{}
|
2020-01-06 09:35:52 +08:00
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
FontFNT::~FontFNT()
|
|
|
|
{
|
|
|
|
_configuration->release();
|
|
|
|
}
|
|
|
|
|
|
|
|
void FontFNT::purgeCachedData()
|
|
|
|
{
|
|
|
|
if (s_configurations)
|
|
|
|
{
|
|
|
|
s_configurations->clear();
|
|
|
|
CC_SAFE_DELETE(s_configurations);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
int* FontFNT::getHorizontalKerningForTextUTF32(const std::u32string& text, int& outNumLetters) const
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
outNumLetters = static_cast<int>(text.length());
|
2021-12-25 10:04:45 +08:00
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
if (!outNumLetters)
|
|
|
|
return nullptr;
|
2021-12-25 10:04:45 +08:00
|
|
|
|
|
|
|
int* sizes = new int[outNumLetters];
|
2019-11-23 20:27:39 +08:00
|
|
|
for (int c = 0; c < outNumLetters; ++c)
|
|
|
|
{
|
2021-12-25 10:04:45 +08:00
|
|
|
if (c < (outNumLetters - 1))
|
|
|
|
sizes[c] = getHorizontalKerningForChars(text[c], text[c + 1]);
|
2019-11-23 20:27:39 +08:00
|
|
|
else
|
|
|
|
sizes[c] = 0;
|
|
|
|
}
|
2021-12-25 10:04:45 +08:00
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
return sizes;
|
|
|
|
}
|
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
int FontFNT::getHorizontalKerningForChars(char32_t firstChar, char32_t secondChar) const
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
2021-12-25 10:04:45 +08:00
|
|
|
int ret = 0;
|
2019-11-23 20:27:39 +08:00
|
|
|
uint64_t key = ((uint64_t)firstChar << 32) | ((uint64_t)secondChar & 0xffffffffll);
|
2021-12-25 10:04:45 +08:00
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
auto iter = _configuration->_kerningDictionary.find(key);
|
2021-12-25 10:04:45 +08:00
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
if (iter != _configuration->_kerningDictionary.end())
|
|
|
|
{
|
|
|
|
ret = iter->second;
|
|
|
|
}
|
2021-12-25 10:04:45 +08:00
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
void FontFNT::setFontSize(float fontSize)
|
|
|
|
{
|
|
|
|
_fontSize = fontSize;
|
|
|
|
}
|
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
int FontFNT::getOriginalFontSize() const
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
return _configuration->_fontSize;
|
|
|
|
}
|
|
|
|
|
2021-11-11 17:20:21 +08:00
|
|
|
FontAtlas* FontFNT::newFontAtlas()
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
// check that everything is fine with the BMFontCofniguration
|
|
|
|
if (_configuration->_fontDefDictionary.empty())
|
|
|
|
return nullptr;
|
2021-12-25 10:04:45 +08:00
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
size_t numGlyphs = _configuration->_characterSet->size();
|
|
|
|
if (numGlyphs == 0)
|
|
|
|
return nullptr;
|
2021-12-25 10:04:45 +08:00
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
if (_configuration->_commonHeight == 0)
|
|
|
|
return nullptr;
|
2021-12-25 10:04:45 +08:00
|
|
|
|
|
|
|
FontAtlas* tempAtlas = new FontAtlas(this);
|
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
// common height
|
2021-12-25 10:04:45 +08:00
|
|
|
int originalFontSize = _configuration->_fontSize;
|
2019-11-23 20:27:39 +08:00
|
|
|
float originalLineHeight = (float)_configuration->_commonHeight;
|
2021-12-25 10:04:45 +08:00
|
|
|
float factor = 0.0f;
|
|
|
|
if (std::abs(_fontSize - originalFontSize) < FLT_EPSILON)
|
|
|
|
{
|
2019-11-23 20:27:39 +08:00
|
|
|
factor = 1.0f;
|
2021-12-25 10:04:45 +08:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2019-11-23 20:27:39 +08:00
|
|
|
factor = _fontSize / originalFontSize;
|
|
|
|
}
|
2021-12-25 10:04:45 +08:00
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
tempAtlas->setLineHeight(originalLineHeight * factor);
|
2020-01-06 09:35:52 +08:00
|
|
|
|
|
|
|
auto rw = _imageRectInPoints.size.width;
|
|
|
|
auto rh = _imageRectInPoints.size.height;
|
|
|
|
|
|
|
|
if (_imageRotated)
|
|
|
|
std::swap(rw, rh);
|
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
const auto left = _imageRectInPoints.origin.x;
|
2020-01-06 09:35:52 +08:00
|
|
|
const auto right = _imageRectInPoints.origin.x + rw;
|
2021-12-25 10:04:45 +08:00
|
|
|
const auto top = _imageRectInPoints.origin.y;
|
2020-01-06 09:35:52 +08:00
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
for (auto&& e : _configuration->_fontDefDictionary)
|
|
|
|
{
|
|
|
|
BMFontDef& fontDef = e.second;
|
2020-01-06 09:35:52 +08:00
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
FontLetterDefinition tempDefinition;
|
|
|
|
|
2020-01-06 09:35:52 +08:00
|
|
|
const auto tempRect = CC_RECT_PIXELS_TO_POINTS(fontDef.rect);
|
2021-12-25 10:04:45 +08:00
|
|
|
|
|
|
|
tempDefinition.offsetX = fontDef.xOffset;
|
|
|
|
tempDefinition.offsetY = fontDef.yOffset;
|
2020-01-06 09:35:52 +08:00
|
|
|
|
|
|
|
if (_imageRotated)
|
|
|
|
{
|
|
|
|
tempDefinition.U = right - tempRect.origin.y - tempRect.size.height;
|
|
|
|
tempDefinition.V = tempRect.origin.x + top;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
tempDefinition.U = tempRect.origin.x + left;
|
|
|
|
tempDefinition.V = tempRect.origin.y + top;
|
|
|
|
}
|
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
tempDefinition.width = tempRect.size.width;
|
|
|
|
tempDefinition.height = tempRect.size.height;
|
|
|
|
|
|
|
|
// carloX: only one texture supported FOR NOW
|
2019-11-23 20:27:39 +08:00
|
|
|
tempDefinition.textureID = 0;
|
2021-12-25 10:04:45 +08:00
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
tempDefinition.validDefinition = true;
|
2021-12-25 10:04:45 +08:00
|
|
|
tempDefinition.xAdvance = fontDef.xAdvance;
|
|
|
|
tempDefinition.rotated = _imageRotated;
|
2020-01-06 09:35:52 +08:00
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
// add the new definition
|
2021-12-25 10:04:45 +08:00
|
|
|
if (65535 < fontDef.charID)
|
|
|
|
{
|
2019-11-23 20:27:39 +08:00
|
|
|
CCLOGWARN("Warning: 65535 < fontDef.charID (%u), ignored", fontDef.charID);
|
2021-12-25 10:04:45 +08:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
tempAtlas->addLetterDefinition(fontDef.charID, tempDefinition);
|
2019-11-23 20:27:39 +08:00
|
|
|
}
|
|
|
|
}
|
2021-12-25 10:04:45 +08:00
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
// add the texture (only one texture for now)
|
2021-12-25 10:04:45 +08:00
|
|
|
|
|
|
|
Texture2D* tempTexture = Director::getInstance()->getTextureCache()->addImage(_configuration->getAtlasName());
|
|
|
|
if (!tempTexture)
|
|
|
|
{
|
2019-11-23 20:27:39 +08:00
|
|
|
CC_SAFE_RELEASE(tempAtlas);
|
|
|
|
return nullptr;
|
|
|
|
}
|
2021-12-25 10:04:45 +08:00
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
// add the texture
|
|
|
|
tempAtlas->addTexture(tempTexture, 0);
|
2021-12-25 10:04:45 +08:00
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
// done
|
|
|
|
return tempAtlas;
|
|
|
|
}
|
|
|
|
|
2021-12-26 23:26:34 +08:00
|
|
|
void FontFNT::reloadBMFontResource(std::string_view fntFilePath)
|
2019-11-23 20:27:39 +08:00
|
|
|
{
|
|
|
|
if (s_configurations == nullptr)
|
|
|
|
{
|
2021-12-26 23:26:34 +08:00
|
|
|
s_configurations = new StringMap<BMFontConfiguration*>();
|
2019-11-23 20:27:39 +08:00
|
|
|
}
|
|
|
|
|
2021-12-25 10:04:45 +08:00
|
|
|
BMFontConfiguration* ret = s_configurations->at(fntFilePath);
|
2019-11-23 20:27:39 +08:00
|
|
|
if (ret != nullptr)
|
|
|
|
{
|
|
|
|
s_configurations->erase(fntFilePath);
|
|
|
|
}
|
2020-01-06 09:35:52 +08:00
|
|
|
|
2019-11-23 20:27:39 +08:00
|
|
|
ret = BMFontConfiguration::create(fntFilePath);
|
|
|
|
if (ret)
|
|
|
|
{
|
|
|
|
s_configurations->insert(fntFilePath, ret);
|
|
|
|
Director::getInstance()->getTextureCache()->reloadTexture(ret->getAtlasName());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_CC_END
|