2010-08-26 17:53:35 +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-08-27 18:07:37 +08:00
|
|
|
#include <libxml/parser.h>
|
|
|
|
#include <libxml/tree.h>
|
|
|
|
#include <libxml/xmlmemory.h>
|
|
|
|
#include "CCTMXXMLParser.h"
|
2010-08-31 16:48:42 +08:00
|
|
|
#include "CCTMXTiledMap.h"
|
2010-08-27 18:07:37 +08:00
|
|
|
#include "ccMacros.h"
|
|
|
|
#include "CCXFileUtils.h"
|
|
|
|
#include "CGPointExtension.h"
|
|
|
|
#include "support/base64.h"
|
|
|
|
#include "platform/platform.h"
|
|
|
|
|
|
|
|
namespace cocos2d {
|
|
|
|
|
2010-08-27 18:14:17 +08:00
|
|
|
void tmx_startElement(void *ctx, const xmlChar *name, const xmlChar **atts);
|
|
|
|
void tmx_endElement(void *ctx, const xmlChar *name);
|
|
|
|
void tmx_characters(void *ctx, const xmlChar *ch, int len);
|
2010-08-27 18:07:37 +08:00
|
|
|
|
|
|
|
// implementation CCTMXLayerInfo
|
|
|
|
CCTMXLayerInfo::CCTMXLayerInfo()
|
2010-08-31 18:11:31 +08:00
|
|
|
:m_bOwnTiles(true)
|
|
|
|
,m_uMinGID(100000)
|
|
|
|
,m_uMaxGID(0)
|
|
|
|
,m_sName("")
|
|
|
|
,m_pTiles(NULL)
|
|
|
|
,m_tOffset(CGPointZero)
|
2010-08-27 18:07:37 +08:00
|
|
|
{
|
|
|
|
m_pProperties= new StringToStringDictionary();;
|
|
|
|
}
|
|
|
|
CCTMXLayerInfo::~CCTMXLayerInfo()
|
|
|
|
{
|
|
|
|
CCLOGINFO("cocos2d: deallocing.");
|
2010-08-30 15:23:07 +08:00
|
|
|
if (m_pProperties)
|
|
|
|
{
|
|
|
|
m_pProperties->clear();
|
|
|
|
delete m_pProperties;
|
2010-09-04 14:51:16 +08:00
|
|
|
m_pProperties = NULL;
|
2010-08-30 15:23:07 +08:00
|
|
|
}
|
2010-08-27 18:07:37 +08:00
|
|
|
if( m_bOwnTiles && m_pTiles )
|
|
|
|
{
|
|
|
|
delete [] m_pTiles;
|
|
|
|
m_pTiles = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// implementation CCTMXTilesetInfo
|
2010-08-31 18:11:31 +08:00
|
|
|
CCTMXTilesetInfo::CCTMXTilesetInfo()
|
|
|
|
:m_uFirstGid(0)
|
|
|
|
,m_tTileSize(CGSizeZero)
|
|
|
|
,m_uSpacing(0)
|
|
|
|
,m_uMargin(0)
|
|
|
|
,m_tImageSize(CGSizeZero)
|
|
|
|
{
|
|
|
|
}
|
2010-08-27 18:07:37 +08:00
|
|
|
CCTMXTilesetInfo::~CCTMXTilesetInfo()
|
|
|
|
{
|
|
|
|
CCLOGINFO("cocos2d: deallocing.");
|
|
|
|
}
|
|
|
|
CGRect CCTMXTilesetInfo::rectForGID(unsigned int gid)
|
|
|
|
{
|
|
|
|
CGRect rect;
|
|
|
|
rect.size = m_tTileSize;
|
|
|
|
gid = gid - m_uFirstGid;
|
|
|
|
int max_x = (int)((m_tImageSize.width - m_uMargin*2 + m_uSpacing) / (m_tTileSize.width + m_uSpacing));
|
|
|
|
// int max_y = (imageSize.height - margin*2 + spacing) / (tileSize.height + spacing);
|
|
|
|
rect.origin.x = (gid % max_x) * (m_tTileSize.width + m_uSpacing) + m_uMargin;
|
|
|
|
rect.origin.y = (gid / max_x) * (m_tTileSize.height + m_uSpacing) + m_uMargin;
|
|
|
|
return rect;
|
|
|
|
}
|
|
|
|
|
|
|
|
// implementation CCTMXMapInfo
|
|
|
|
|
|
|
|
CCTMXMapInfo * CCTMXMapInfo::formatWithTMXFile(const char *tmxFile)
|
|
|
|
{
|
|
|
|
CCTMXMapInfo *pRet = new CCTMXMapInfo();
|
|
|
|
if(pRet->initWithTMXFile(tmxFile))
|
|
|
|
{
|
2010-08-30 11:45:46 +08:00
|
|
|
pRet->autorelease();
|
|
|
|
return pRet;
|
2010-08-27 18:07:37 +08:00
|
|
|
}
|
|
|
|
CCX_SAFE_DELETE(pRet);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
bool CCTMXMapInfo::initWithTMXFile(const char *tmxFile)
|
|
|
|
{
|
2010-08-30 11:45:46 +08:00
|
|
|
m_pTilesets = new NSMutableArray<CCTMXTilesetInfo*>();
|
|
|
|
m_pLayers = new NSMutableArray<CCTMXLayerInfo*>();
|
2010-08-27 18:07:37 +08:00
|
|
|
m_sFilename = CCFileUtils::fullPathFromRelativePath(tmxFile);
|
2010-08-30 11:45:46 +08:00
|
|
|
m_pObjectGroups = new NSMutableArray<CCTMXObjectGroup*>();
|
2010-08-27 18:07:37 +08:00
|
|
|
m_pProperties = new StringToStringDictionary();
|
|
|
|
m_pTileProperties = new std::map<int, StringToStringDictionary*>();
|
|
|
|
|
|
|
|
// tmp vars
|
|
|
|
m_sCurrentString = "";
|
|
|
|
m_bStoringCharacters = false;
|
|
|
|
m_nLayerAttribs = TMXLayerAttribNone;
|
|
|
|
m_nParentElement = TMXPropertyNone;
|
|
|
|
|
|
|
|
return parseXMLFile(m_sFilename.c_str());
|
|
|
|
}
|
2010-08-31 18:11:31 +08:00
|
|
|
CCTMXMapInfo::CCTMXMapInfo()
|
|
|
|
:m_bStoringCharacters(false)
|
|
|
|
,m_nLayerAttribs(0)
|
|
|
|
,m_tMapSize(CGSizeZero)
|
|
|
|
,m_tTileSize(CGSizeZero)
|
|
|
|
,m_pLayers(NULL)
|
|
|
|
,m_pTilesets(NULL)
|
|
|
|
{
|
|
|
|
}
|
2010-08-27 18:07:37 +08:00
|
|
|
CCTMXMapInfo::~CCTMXMapInfo()
|
|
|
|
{
|
|
|
|
CCLOGINFO("cocos2d: deallocing.");
|
|
|
|
m_pTilesets->release();
|
|
|
|
m_pLayers->release();
|
|
|
|
m_pObjectGroups->release();
|
2010-09-02 18:16:02 +08:00
|
|
|
if (m_pProperties)
|
|
|
|
{
|
|
|
|
m_pProperties->clear();
|
|
|
|
delete m_pProperties;
|
2010-09-04 14:51:16 +08:00
|
|
|
m_pProperties = NULL;
|
2010-09-02 18:16:02 +08:00
|
|
|
}
|
|
|
|
if (m_pTileProperties)
|
|
|
|
{
|
|
|
|
std::map<int, StringToStringDictionary*>::iterator it;
|
|
|
|
for (it = m_pTileProperties->begin(); it != m_pTileProperties->end(); ++it)
|
|
|
|
{
|
|
|
|
if (it->second)
|
|
|
|
{
|
|
|
|
it->second->clear();
|
|
|
|
delete it->second;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
m_pTileProperties->clear();
|
|
|
|
delete m_pTileProperties;
|
2010-09-04 14:51:16 +08:00
|
|
|
m_pTileProperties = NULL;
|
2010-09-02 18:16:02 +08:00
|
|
|
}
|
2010-08-27 18:07:37 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
bool CCTMXMapInfo::parseXMLFile(const char *xmlFilename)
|
|
|
|
{
|
|
|
|
FILE *fp = NULL;
|
|
|
|
if( !(fp = fopen(xmlFilename, "r")) )
|
|
|
|
{
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
fseek(fp,0,SEEK_END);
|
|
|
|
int size = ftell(fp);
|
|
|
|
fseek(fp,0,SEEK_SET);
|
|
|
|
char *buffer = new char[size+1];
|
|
|
|
fread(buffer,sizeof(char),size,fp);
|
|
|
|
fclose(fp);
|
|
|
|
/*
|
|
|
|
* this initialize the library and check potential ABI mismatches
|
|
|
|
* between the version it was compiled for and the actual shared
|
|
|
|
* library used.
|
|
|
|
*/
|
|
|
|
LIBXML_TEST_VERSION
|
|
|
|
xmlSAXHandler saxHandler;
|
|
|
|
memset( &saxHandler, 0, sizeof(saxHandler) );
|
|
|
|
// Using xmlSAXVersion( &saxHandler, 2 ) generate crash as it sets plenty of other pointers...
|
|
|
|
saxHandler.initialized = XML_SAX2_MAGIC; // so we do this to force parsing as SAX2.
|
2010-08-27 18:14:17 +08:00
|
|
|
saxHandler.startElement = &tmx_startElement;
|
|
|
|
saxHandler.endElement = &tmx_endElement;
|
|
|
|
saxHandler.characters = &tmx_characters;
|
2010-08-27 18:07:37 +08:00
|
|
|
|
|
|
|
int result = xmlSAXUserParseMemory( &saxHandler, this, buffer, size );
|
|
|
|
if ( result != 0 )
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
/*
|
|
|
|
* Cleanup function for the XML library.
|
|
|
|
*/
|
|
|
|
xmlCleanupParser();
|
|
|
|
/*
|
|
|
|
* this is to debug memory for regression tests
|
|
|
|
*/
|
|
|
|
xmlMemoryDump();
|
|
|
|
delete []buffer;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// the XML parser calls here with all the elements
|
2010-08-27 18:14:17 +08:00
|
|
|
void tmx_startElement(void *ctx, const xmlChar *name, const xmlChar **atts)
|
2010-08-27 18:07:37 +08:00
|
|
|
{
|
|
|
|
CCTMXMapInfo *pTMXMapInfo = (CCTMXMapInfo*)(ctx);
|
|
|
|
std::string elementName = (char*)name;
|
2010-09-02 18:16:02 +08:00
|
|
|
StringToStringDictionary *attributeDict = new StringToStringDictionary();
|
2010-08-27 18:07:37 +08:00
|
|
|
if(atts && atts[0])
|
|
|
|
{
|
|
|
|
for(int i = 0; atts[i]; i += 2)
|
|
|
|
{
|
|
|
|
std::string key = (char*)atts[i];
|
|
|
|
std::string value = (char*)atts[i+1];
|
2010-09-02 18:16:02 +08:00
|
|
|
attributeDict->insert(StringToStringPair(key, value));
|
2010-08-27 18:07:37 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if(elementName == "map")
|
|
|
|
{
|
2010-09-02 18:16:02 +08:00
|
|
|
std::string version = valueForKey("version", attributeDict);
|
2010-08-27 18:07:37 +08:00
|
|
|
if ( version != "1.0")
|
|
|
|
{
|
|
|
|
CCLOG("cocos2d: TMXFormat: Unsupported TMX version: %@", version);
|
|
|
|
}
|
2010-09-02 18:16:02 +08:00
|
|
|
std::string orientationStr = valueForKey("orientation", attributeDict);
|
2010-08-27 18:07:37 +08:00
|
|
|
if( orientationStr == "orthogonal")
|
|
|
|
pTMXMapInfo->setOrientation(CCTMXOrientationOrtho);
|
|
|
|
else if ( orientationStr == "isometric")
|
|
|
|
pTMXMapInfo->setOrientation(CCTMXOrientationIso);
|
|
|
|
else if( orientationStr == "hexagonal")
|
|
|
|
pTMXMapInfo->setOrientation(CCTMXOrientationHex);
|
|
|
|
else
|
|
|
|
CCLOG("cocos2d: TMXFomat: Unsupported orientation: %d", pTMXMapInfo->getOrientation());
|
|
|
|
|
|
|
|
CGSize s;
|
2010-09-02 18:16:02 +08:00
|
|
|
s.width = (float)atof(valueForKey("width", attributeDict));
|
|
|
|
s.height = (float)atof(valueForKey("height", attributeDict));
|
2010-08-27 18:07:37 +08:00
|
|
|
pTMXMapInfo->setMapSize(s);
|
|
|
|
|
2010-09-02 18:16:02 +08:00
|
|
|
s.width = (float)atof(valueForKey("tilewidth", attributeDict));
|
|
|
|
s.height = (float)atof(valueForKey("tileheight", attributeDict));
|
2010-08-27 18:07:37 +08:00
|
|
|
pTMXMapInfo->setTileSize(s);
|
|
|
|
|
|
|
|
// The parent element is now "map"
|
|
|
|
pTMXMapInfo->setParentElement(TMXPropertyMap);
|
|
|
|
}
|
|
|
|
else if(elementName == "tileset")
|
|
|
|
{
|
|
|
|
// If this is an external tileset then start parsing that
|
2010-09-02 18:16:02 +08:00
|
|
|
std::string externalTilesetFilename = valueForKey("source", attributeDict);
|
2010-08-27 18:07:37 +08:00
|
|
|
if (externalTilesetFilename != "")
|
|
|
|
{
|
|
|
|
externalTilesetFilename = CCFileUtils::fullPathFromRelativePath(externalTilesetFilename.c_str());
|
|
|
|
pTMXMapInfo->parseXMLFile(externalTilesetFilename.c_str());
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
CCTMXTilesetInfo *tileset = new CCTMXTilesetInfo();
|
2010-09-02 18:16:02 +08:00
|
|
|
tileset->m_sName = valueForKey("name", attributeDict);
|
|
|
|
tileset->m_uFirstGid = (unsigned int)atoi(valueForKey("firstgid", attributeDict));
|
|
|
|
tileset->m_uSpacing = (unsigned int)atoi(valueForKey("spacing", attributeDict));
|
|
|
|
tileset->m_uMargin = (unsigned int)atoi(valueForKey("margin", attributeDict));
|
2010-08-27 18:07:37 +08:00
|
|
|
CGSize s;
|
2010-09-02 18:16:02 +08:00
|
|
|
s.width = (float)atof(valueForKey("tilewidth", attributeDict));
|
|
|
|
s.height = (float)atof(valueForKey("tileheight", attributeDict));
|
2010-08-27 18:07:37 +08:00
|
|
|
tileset->m_tTileSize = s;
|
|
|
|
|
|
|
|
pTMXMapInfo->getTilesets()->addObject(tileset);
|
|
|
|
tileset->release();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if(elementName == "tile")
|
|
|
|
{
|
|
|
|
CCTMXTilesetInfo* info = pTMXMapInfo->getTilesets()->getLastObject();
|
|
|
|
StringToStringDictionary *dict = new StringToStringDictionary();
|
2010-09-02 18:16:02 +08:00
|
|
|
pTMXMapInfo->setParentGID(info->m_uFirstGid + atoi(valueForKey("id", attributeDict)));
|
2010-08-27 18:07:37 +08:00
|
|
|
pTMXMapInfo->getTileProperties()->insert(std::pair<int, StringToStringDictionary*>(pTMXMapInfo->getParentGID(), dict));
|
|
|
|
|
|
|
|
pTMXMapInfo->setParentElement(TMXPropertyTile);
|
|
|
|
|
|
|
|
}
|
|
|
|
else if(elementName == "layer")
|
|
|
|
{
|
|
|
|
CCTMXLayerInfo *layer = new CCTMXLayerInfo();
|
2010-09-02 18:16:02 +08:00
|
|
|
layer->m_sName = valueForKey("name", attributeDict);
|
2010-08-27 18:07:37 +08:00
|
|
|
|
|
|
|
CGSize s;
|
2010-09-02 18:16:02 +08:00
|
|
|
s.width = (float)atof(valueForKey("width", attributeDict));
|
|
|
|
s.height = (float)atof(valueForKey("height", attributeDict));
|
2010-08-27 18:07:37 +08:00
|
|
|
layer->m_tLayerSize = s;
|
|
|
|
|
2010-09-02 18:16:02 +08:00
|
|
|
std::string visible = valueForKey("visible", attributeDict);
|
2010-08-27 18:07:37 +08:00
|
|
|
layer->m_bVisible = !(visible == "0");
|
|
|
|
|
2010-09-02 18:16:02 +08:00
|
|
|
std::string opacity = valueForKey("opacity", attributeDict);
|
2010-08-27 18:07:37 +08:00
|
|
|
if( opacity != "" )
|
|
|
|
{
|
|
|
|
layer->m_cOpacity = (unsigned char)(255 * atof(opacity.c_str()));
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
layer->m_cOpacity = 255;
|
|
|
|
}
|
|
|
|
|
2010-09-02 18:16:02 +08:00
|
|
|
float x = (float)atof(valueForKey("x", attributeDict));
|
|
|
|
float y = (float)atof(valueForKey("y", attributeDict));
|
2010-08-27 18:07:37 +08:00
|
|
|
layer->m_tOffset = ccp(x,y);
|
|
|
|
|
|
|
|
pTMXMapInfo->getLayers()->addObject(layer);
|
|
|
|
layer->release();
|
|
|
|
|
|
|
|
// The parent element is now "layer"
|
|
|
|
pTMXMapInfo->setParentElement(TMXPropertyLayer);
|
|
|
|
|
|
|
|
}
|
|
|
|
else if(elementName == "objectgroup")
|
|
|
|
{
|
2010-08-30 11:45:46 +08:00
|
|
|
CCTMXObjectGroup *objectGroup = new CCTMXObjectGroup();
|
2010-09-02 18:16:02 +08:00
|
|
|
objectGroup->setGroupName(valueForKey("name", attributeDict));
|
2010-08-27 18:07:37 +08:00
|
|
|
CGPoint positionOffset;
|
2010-09-02 18:16:02 +08:00
|
|
|
positionOffset.x = (float)atof(valueForKey("x", attributeDict)) * pTMXMapInfo->getTileSize().width;
|
|
|
|
positionOffset.y = (float)atof(valueForKey("y", attributeDict)) * pTMXMapInfo->getTileSize().height;
|
2010-08-30 11:45:46 +08:00
|
|
|
objectGroup->setPositionOffset(positionOffset);
|
2010-08-27 18:07:37 +08:00
|
|
|
|
2010-08-30 11:45:46 +08:00
|
|
|
pTMXMapInfo->getObjectGroups()->addObject(objectGroup);
|
|
|
|
objectGroup->release();
|
2010-08-27 18:07:37 +08:00
|
|
|
|
|
|
|
// The parent element is now "objectgroup"
|
2010-08-30 11:45:46 +08:00
|
|
|
pTMXMapInfo->setParentElement(TMXPropertyObjectGroup);
|
2010-08-27 18:07:37 +08:00
|
|
|
|
|
|
|
}
|
|
|
|
else if(elementName == "image")
|
|
|
|
{
|
|
|
|
CCTMXTilesetInfo *tileset = pTMXMapInfo->getTilesets()->getLastObject();
|
|
|
|
|
|
|
|
// build full path
|
2010-09-02 18:16:02 +08:00
|
|
|
std::string imagename = valueForKey("source", attributeDict);
|
2010-08-27 18:07:37 +08:00
|
|
|
tileset->m_sSourceImage = CCFileUtils::fullPathFromRelativePath(imagename.c_str());
|
|
|
|
|
|
|
|
}
|
|
|
|
else if(elementName == "data")
|
|
|
|
{
|
2010-09-02 18:16:02 +08:00
|
|
|
std::string encoding = valueForKey("encoding", attributeDict);
|
|
|
|
std::string compression = valueForKey("compression", attributeDict);
|
2010-08-27 18:07:37 +08:00
|
|
|
|
|
|
|
if( encoding == "base64" )
|
|
|
|
{
|
|
|
|
int layerAttribs = pTMXMapInfo->getLayerAttribs();
|
|
|
|
pTMXMapInfo->setLayerAttribs(layerAttribs | TMXLayerAttribBase64);
|
|
|
|
pTMXMapInfo->setStoringCharacters(true);
|
|
|
|
|
|
|
|
if( compression == "gzip" )
|
|
|
|
{
|
|
|
|
layerAttribs = pTMXMapInfo->getLayerAttribs();
|
|
|
|
pTMXMapInfo->setLayerAttribs(layerAttribs | TMXLayerAttribGzip);
|
|
|
|
}
|
2010-09-01 17:26:16 +08:00
|
|
|
NSAssert( compression == "" || compression == "gzip", "TMX: unsupported compression method" );
|
2010-08-27 18:07:37 +08:00
|
|
|
}
|
|
|
|
NSAssert( pTMXMapInfo->getLayerAttribs() != TMXLayerAttribNone, "TMX tile map: Only base64 and/or gzip maps are supported" );
|
|
|
|
|
|
|
|
}
|
|
|
|
else if(elementName == "object")
|
|
|
|
{
|
2010-08-30 15:23:07 +08:00
|
|
|
char buffer[32];
|
2010-08-30 11:45:46 +08:00
|
|
|
CCTMXObjectGroup *objectGroup = pTMXMapInfo->getObjectGroups()->getLastObject();
|
2010-08-27 18:07:37 +08:00
|
|
|
|
|
|
|
// The value for "type" was blank or not a valid class name
|
|
|
|
// Create an instance of TMXObjectInfo to store the object and its properties
|
2010-08-30 11:45:46 +08:00
|
|
|
StringToStringDictionary *dict = new StringToStringDictionary();
|
2010-08-27 18:07:37 +08:00
|
|
|
|
|
|
|
// Set the name of the object to the value for "name"
|
2010-08-30 11:45:46 +08:00
|
|
|
std::string key = "name";
|
2010-09-02 18:16:02 +08:00
|
|
|
std::string value = valueForKey("name", attributeDict);
|
2010-08-30 11:45:46 +08:00
|
|
|
dict->insert(StringToStringPair(key, value));
|
2010-08-27 18:07:37 +08:00
|
|
|
|
|
|
|
// Assign all the attributes as key/name pairs in the properties dictionary
|
2010-08-30 11:45:46 +08:00
|
|
|
key = "type";
|
2010-09-02 18:16:02 +08:00
|
|
|
value = valueForKey("type", attributeDict);
|
2010-08-30 11:45:46 +08:00
|
|
|
dict->insert(StringToStringPair(key, value));
|
|
|
|
|
2010-09-02 18:16:02 +08:00
|
|
|
int x = atoi(valueForKey("x", attributeDict)) + (int)objectGroup->getPositionOffset().x;
|
2010-08-30 11:45:46 +08:00
|
|
|
key = "x";
|
2010-09-03 14:40:48 +08:00
|
|
|
/*value = itoa(x, buffer, 10);*/
|
|
|
|
sprintf(buffer, "%d", x);
|
|
|
|
value = buffer;
|
2010-08-30 11:45:46 +08:00
|
|
|
dict->insert(StringToStringPair(key, value));
|
|
|
|
|
2010-09-02 18:16:02 +08:00
|
|
|
int y = atoi(valueForKey("y", attributeDict)) + (int)objectGroup->getPositionOffset().y;
|
2010-08-27 18:07:37 +08:00
|
|
|
// Correct y position. (Tiled uses Flipped, cocos2d uses Standard)
|
2010-09-02 18:16:02 +08:00
|
|
|
y = (int)(pTMXMapInfo->getMapSize().height * pTMXMapInfo->getTileSize().height) - y - atoi(valueForKey("height", attributeDict));
|
2010-08-30 11:45:46 +08:00
|
|
|
key = "y";
|
2010-09-03 14:40:48 +08:00
|
|
|
/*value = itoa(y, buffer, 10);*/
|
|
|
|
sprintf(buffer, "%d", y);
|
|
|
|
value = buffer;
|
2010-08-30 11:45:46 +08:00
|
|
|
dict->insert(StringToStringPair(key, value));
|
|
|
|
|
|
|
|
key = "width";
|
2010-09-02 18:16:02 +08:00
|
|
|
value = valueForKey("width", attributeDict);
|
2010-08-30 11:45:46 +08:00
|
|
|
dict->insert(StringToStringPair(key, value));
|
|
|
|
|
|
|
|
key = "height";
|
2010-09-02 18:16:02 +08:00
|
|
|
value = valueForKey("height", attributeDict);
|
2010-08-30 11:45:46 +08:00
|
|
|
dict->insert(StringToStringPair(key, value));
|
2010-08-27 18:07:37 +08:00
|
|
|
|
|
|
|
// Add the object to the objectGroup
|
2010-08-30 11:45:46 +08:00
|
|
|
objectGroup->getObjects()->insert(objectGroup->getObjects()->begin(), dict);
|
2010-08-27 18:07:37 +08:00
|
|
|
|
|
|
|
// The parent element is now "object"
|
2010-08-30 11:45:46 +08:00
|
|
|
pTMXMapInfo->setParentElement(TMXPropertyObject);
|
2010-08-27 18:07:37 +08:00
|
|
|
|
|
|
|
}
|
|
|
|
else if(elementName == "property")
|
|
|
|
{
|
|
|
|
if ( pTMXMapInfo->getParentElement() == TMXPropertyNone )
|
|
|
|
{
|
|
|
|
CCLOG( "TMX tile map: Parent element is unsupported. Cannot add property named '%s' with value '%s'",
|
2010-09-02 18:16:02 +08:00
|
|
|
valueForKey("name", attributeDict), valueForKey("value",attributeDict) );
|
2010-08-27 18:07:37 +08:00
|
|
|
}
|
|
|
|
else if ( pTMXMapInfo->getParentElement() == TMXPropertyMap )
|
|
|
|
{
|
|
|
|
// The parent element is the map
|
2010-09-02 18:16:02 +08:00
|
|
|
std::string value = valueForKey("value", attributeDict);
|
|
|
|
std::string key = valueForKey("name", attributeDict);
|
2010-08-27 18:07:37 +08:00
|
|
|
pTMXMapInfo->getProperties()->insert(StringToStringPair(key, value));
|
|
|
|
|
|
|
|
}
|
|
|
|
else if ( pTMXMapInfo->getParentElement() == TMXPropertyLayer )
|
|
|
|
{
|
|
|
|
// The parent element is the last layer
|
|
|
|
CCTMXLayerInfo *layer = pTMXMapInfo->getLayers()->getLastObject();
|
2010-09-02 18:16:02 +08:00
|
|
|
std::string value = valueForKey("value", attributeDict);
|
|
|
|
std::string key = valueForKey("name", attributeDict);
|
2010-08-27 18:07:37 +08:00
|
|
|
// Add the property to the layer
|
|
|
|
layer->m_pProperties->insert(StringToStringPair(key, value));
|
|
|
|
|
|
|
|
}
|
|
|
|
else if ( pTMXMapInfo->getParentElement() == TMXPropertyObjectGroup )
|
|
|
|
{
|
|
|
|
// The parent element is the last object group
|
2010-08-30 11:45:46 +08:00
|
|
|
CCTMXObjectGroup *objectGroup = pTMXMapInfo->getObjectGroups()->getLastObject();
|
2010-09-02 18:16:02 +08:00
|
|
|
std::string key = valueForKey("name", attributeDict);
|
|
|
|
std::string value = valueForKey("value", attributeDict);
|
2010-08-30 11:45:46 +08:00
|
|
|
objectGroup->getProperties()->insert(StringToStringPair(key, value));
|
2010-08-27 18:07:37 +08:00
|
|
|
|
|
|
|
}
|
|
|
|
else if ( pTMXMapInfo->getParentElement() == TMXPropertyObject )
|
|
|
|
{
|
|
|
|
// The parent element is the last object
|
2010-08-30 11:45:46 +08:00
|
|
|
CCTMXObjectGroup *objectGroup = pTMXMapInfo->getObjectGroups()->getLastObject();
|
|
|
|
StringToStringDictionary *dict = *objectGroup->getObjects()->begin();
|
2010-08-27 18:07:37 +08:00
|
|
|
|
2010-09-02 18:16:02 +08:00
|
|
|
std::string propertyName = valueForKey("name", attributeDict);
|
|
|
|
std::string propertyValue = valueForKey("value", attributeDict);
|
2010-08-27 18:07:37 +08:00
|
|
|
|
2010-08-30 11:45:46 +08:00
|
|
|
dict->insert(StringToStringPair(propertyName, propertyValue));
|
2010-08-27 18:07:37 +08:00
|
|
|
}
|
|
|
|
else if ( pTMXMapInfo->getParentElement() == TMXPropertyTile )
|
|
|
|
{
|
|
|
|
StringToStringDictionary *dict;
|
|
|
|
dict = pTMXMapInfo->getTileProperties()->find(pTMXMapInfo->getParentGID())->second;
|
2010-09-02 18:16:02 +08:00
|
|
|
std::string propertyName = valueForKey("name", attributeDict);
|
|
|
|
std::string propertyValue = valueForKey("value", attributeDict);
|
2010-08-27 18:07:37 +08:00
|
|
|
dict->insert(StringToStringPair(propertyName, propertyValue));
|
|
|
|
}
|
|
|
|
}
|
2010-09-02 18:16:02 +08:00
|
|
|
if (attributeDict)
|
|
|
|
{
|
|
|
|
attributeDict->clear();
|
|
|
|
delete attributeDict;
|
|
|
|
}
|
2010-08-27 18:07:37 +08:00
|
|
|
}
|
|
|
|
|
2010-08-27 18:14:17 +08:00
|
|
|
void tmx_endElement(void *ctx, const xmlChar *name)
|
2010-08-27 18:07:37 +08:00
|
|
|
{
|
|
|
|
CCTMXMapInfo *pTMXMapInfo = (CCTMXMapInfo*)(ctx);
|
|
|
|
std::string elementName = (char*)name;
|
|
|
|
|
|
|
|
int len = 0;
|
|
|
|
|
|
|
|
if(elementName == "data" && pTMXMapInfo->getLayerAttribs()&TMXLayerAttribBase64)
|
|
|
|
{
|
|
|
|
pTMXMapInfo->setStoringCharacters(false);
|
|
|
|
|
|
|
|
CCTMXLayerInfo *layer = pTMXMapInfo->getLayers()->getLastObject();
|
|
|
|
|
|
|
|
std::string currentString = pTMXMapInfo->getCurrentString();
|
|
|
|
unsigned char *buffer;
|
|
|
|
len = base64Decode((unsigned char*)currentString.c_str(), currentString.length(), &buffer);
|
|
|
|
if( ! buffer )
|
|
|
|
{
|
|
|
|
CCLOG("cocos2d: TiledMap: decode data error");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if( pTMXMapInfo->getLayerAttribs() & TMXLayerAttribGzip )
|
|
|
|
{
|
|
|
|
unsigned char *deflated;
|
|
|
|
ZipUtils::inflateMemory(buffer, len, &deflated);
|
2010-09-02 15:31:37 +08:00
|
|
|
delete [] buffer;
|
|
|
|
buffer = NULL;
|
2010-08-27 18:07:37 +08:00
|
|
|
|
|
|
|
if( ! deflated )
|
|
|
|
{
|
|
|
|
CCLOG("cocos2d: TiledMap: inflate data error");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
layer->m_pTiles = (unsigned int*) deflated;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
layer->m_pTiles = (unsigned int*) buffer;
|
|
|
|
}
|
|
|
|
|
|
|
|
pTMXMapInfo->setCurrentString("");
|
|
|
|
|
|
|
|
}
|
|
|
|
else if (elementName == "map")
|
|
|
|
{
|
|
|
|
// The map element has ended
|
|
|
|
pTMXMapInfo->setParentElement(TMXPropertyNone);
|
|
|
|
}
|
|
|
|
else if (elementName == "layer")
|
|
|
|
{
|
|
|
|
// The layer element has ended
|
|
|
|
pTMXMapInfo->setParentElement(TMXPropertyNone);
|
|
|
|
}
|
|
|
|
else if (elementName == "objectgroup")
|
|
|
|
{
|
|
|
|
// The objectgroup element has ended
|
|
|
|
pTMXMapInfo->setParentElement(TMXPropertyNone);
|
|
|
|
}
|
|
|
|
else if (elementName == "object")
|
|
|
|
{
|
|
|
|
// The object element has ended
|
|
|
|
pTMXMapInfo->setParentElement(TMXPropertyNone);
|
|
|
|
}
|
|
|
|
}
|
2010-08-27 18:14:17 +08:00
|
|
|
void tmx_characters(void *ctx, const xmlChar *ch, int len)
|
2010-08-27 18:07:37 +08:00
|
|
|
{
|
|
|
|
CCTMXMapInfo *pTMXMapInfo = (CCTMXMapInfo*)(ctx);
|
|
|
|
std::string pText((char*)ch,0,len);
|
|
|
|
|
|
|
|
if (pTMXMapInfo->getStoringCharacters())
|
|
|
|
{
|
|
|
|
std::string currentString = pTMXMapInfo->getCurrentString();
|
|
|
|
currentString += pText;
|
|
|
|
pTMXMapInfo->setCurrentString(currentString);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}//namespace cocos2d
|