2012-04-19 14:35:52 +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.
|
|
|
|
****************************************************************************/
|
|
|
|
|
2012-04-25 16:18:04 +08:00
|
|
|
#define __CC_PLATFORM_FILEUTILS_CPP__
|
2013-01-28 23:19:57 +08:00
|
|
|
#include <string>
|
2012-06-19 16:20:46 +08:00
|
|
|
#include "platform/CCFileUtilsCommon_cpp.h"
|
2012-11-08 19:46:53 +08:00
|
|
|
#include "support/zip_support/ZipUtils.h"
|
2013-01-28 17:44:53 +08:00
|
|
|
#include "platform/CCCommon.h"
|
|
|
|
#include "jni/Java_org_cocos2dx_lib_Cocos2dxHelper.h"
|
2012-04-25 16:18:04 +08:00
|
|
|
|
2012-06-11 10:59:57 +08:00
|
|
|
using namespace std;
|
|
|
|
|
2012-04-19 14:35:52 +08:00
|
|
|
NS_CC_BEGIN
|
|
|
|
|
2012-06-11 10:59:57 +08:00
|
|
|
static CCFileUtils* s_pFileUtils = NULL;
|
2012-11-09 15:53:40 +08:00
|
|
|
// record the zip on the resource path
|
2012-11-08 19:46:53 +08:00
|
|
|
static ZipFile *s_pZipFile = NULL;
|
2013-01-28 17:44:53 +08:00
|
|
|
// The full path cache, key: relative path, value: full path.
|
2013-01-25 21:52:35 +08:00
|
|
|
static std::map<std::string, std::string> s_fullPathCache;
|
|
|
|
|
2012-06-11 10:59:57 +08:00
|
|
|
CCFileUtils* CCFileUtils::sharedFileUtils()
|
|
|
|
{
|
|
|
|
if (s_pFileUtils == NULL)
|
|
|
|
{
|
|
|
|
s_pFileUtils = new CCFileUtils();
|
2013-01-24 00:32:22 +08:00
|
|
|
s_pFileUtils->init();
|
2012-11-09 15:53:40 +08:00
|
|
|
std::string resourcePath = getApkPath();
|
|
|
|
s_pZipFile = new ZipFile(resourcePath, "assets/");
|
2012-06-11 10:59:57 +08:00
|
|
|
}
|
|
|
|
return s_pFileUtils;
|
|
|
|
}
|
|
|
|
|
2013-01-24 00:32:22 +08:00
|
|
|
bool CCFileUtils::init()
|
|
|
|
{
|
2013-01-28 17:44:53 +08:00
|
|
|
m_strDefaultResRootPath = "assets/";
|
|
|
|
m_searchPathArray.push_back(m_strDefaultResRootPath);
|
2013-01-26 22:31:57 +08:00
|
|
|
m_searchResolutionsOrderArray.push_back("");
|
2013-01-28 17:44:53 +08:00
|
|
|
|
2013-01-24 00:32:22 +08:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2012-06-11 10:59:57 +08:00
|
|
|
void CCFileUtils::purgeFileUtils()
|
|
|
|
{
|
|
|
|
if (s_pFileUtils != NULL)
|
|
|
|
{
|
|
|
|
s_pFileUtils->purgeCachedEntries();
|
2013-01-25 20:51:52 +08:00
|
|
|
CC_SAFE_RELEASE(s_pFileUtils->m_pFilenameLookupDict);
|
2012-06-11 10:59:57 +08:00
|
|
|
}
|
2013-01-25 20:51:52 +08:00
|
|
|
|
2012-11-09 15:53:40 +08:00
|
|
|
CC_SAFE_DELETE(s_pZipFile);
|
2012-06-11 10:59:57 +08:00
|
|
|
CC_SAFE_DELETE(s_pFileUtils);
|
|
|
|
}
|
|
|
|
|
|
|
|
void CCFileUtils::purgeCachedEntries()
|
|
|
|
{
|
2013-01-25 21:52:35 +08:00
|
|
|
s_fullPathCache.clear();
|
2012-06-11 10:59:57 +08:00
|
|
|
}
|
|
|
|
|
2012-08-08 18:39:33 +08:00
|
|
|
const char* CCFileUtils::fullPathFromRelativePath(const char *pszRelativePath)
|
2012-04-19 14:35:52 +08:00
|
|
|
{
|
2013-01-25 21:52:35 +08:00
|
|
|
return CCString::create(fullPathForFilename(pszRelativePath))->getCString();
|
2013-01-18 18:05:32 +08:00
|
|
|
}
|
|
|
|
|
2013-01-25 20:51:52 +08:00
|
|
|
std::string CCFileUtils::fullPathForFilename(const char* pszFileName)
|
2013-01-18 18:05:32 +08:00
|
|
|
{
|
2013-01-26 14:19:14 +08:00
|
|
|
CCAssert(pszFileName != NULL, "CCFileUtils: Invalid path");
|
|
|
|
|
|
|
|
// Return directly if it's an absolute path.
|
2013-01-28 10:23:19 +08:00
|
|
|
// On Android, there are two situations for full path.
|
|
|
|
// 1) Files in APK, e.g. assets/path/path/file.png
|
|
|
|
// 2) Files not in APK, e.g. /data/data/org.cocos2dx.hellocpp/cache/path/path/file.png, or /sdcard/path/path/file.png.
|
|
|
|
// So these two situations need to be checked on Android.
|
|
|
|
std::string strFileName = pszFileName;
|
2013-01-29 00:42:21 +08:00
|
|
|
if (pszFileName[0] == '/' || strFileName.find(m_strDefaultResRootPath) == 0)
|
2013-01-26 14:19:14 +08:00
|
|
|
{
|
2013-01-28 10:23:19 +08:00
|
|
|
CCLOG("Return absolute path( %s ) directly.", pszFileName);
|
2013-01-18 18:05:32 +08:00
|
|
|
return pszFileName;
|
|
|
|
}
|
2013-01-25 21:52:35 +08:00
|
|
|
// Already Cached ?
|
|
|
|
std::map<std::string, std::string>::iterator cacheIter = s_fullPathCache.find(pszFileName);
|
|
|
|
if (cacheIter != s_fullPathCache.end())
|
|
|
|
{
|
|
|
|
CCLOG("Return full path from cache: %s", cacheIter->second.c_str());
|
|
|
|
return cacheIter->second;
|
|
|
|
}
|
|
|
|
|
2013-01-25 20:51:52 +08:00
|
|
|
// Get the new file name.
|
|
|
|
std::string newFilename = getNewFilename(pszFileName);
|
2013-01-18 18:05:32 +08:00
|
|
|
|
2013-01-25 20:51:52 +08:00
|
|
|
string fullpath = "";
|
2012-04-19 14:35:52 +08:00
|
|
|
|
2013-01-25 20:51:52 +08:00
|
|
|
bool bFound = false;
|
2013-01-26 22:31:57 +08:00
|
|
|
|
|
|
|
for (std::vector<std::string>::iterator searchPathsIter = m_searchPathArray.begin();
|
|
|
|
searchPathsIter != m_searchPathArray.end(); ++searchPathsIter) {
|
|
|
|
for (std::vector<std::string>::iterator resOrderIter = m_searchResolutionsOrderArray.begin();
|
|
|
|
resOrderIter != m_searchResolutionsOrderArray.end(); ++resOrderIter) {
|
2013-01-25 20:51:52 +08:00
|
|
|
|
2013-01-26 22:31:57 +08:00
|
|
|
CCLOG("\n\nSEARCHING: %s, %s, %s", newFilename.c_str(), resOrderIter->c_str(), searchPathsIter->c_str());
|
|
|
|
fullpath = this->getPathForFilename(newFilename, *resOrderIter, *searchPathsIter);
|
2013-01-25 20:51:52 +08:00
|
|
|
|
|
|
|
// Check whether file exists in apk.
|
|
|
|
if (s_pZipFile->fileExists(fullpath))
|
2013-01-18 21:54:48 +08:00
|
|
|
{
|
2013-01-25 20:51:52 +08:00
|
|
|
bFound = true;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
FILE *fp = fopen(fullpath.c_str(), "r");
|
|
|
|
if(fp)
|
|
|
|
{
|
|
|
|
bFound = true;
|
|
|
|
fclose(fp);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (bFound)
|
|
|
|
{
|
2013-01-26 14:19:14 +08:00
|
|
|
// Using the filename passed in as key.
|
|
|
|
s_fullPathCache.insert(std::pair<std::string, std::string>(pszFileName, fullpath));
|
2013-01-25 20:51:52 +08:00
|
|
|
CCLOG("Returning path: %s", fullpath.c_str());
|
|
|
|
return fullpath;
|
2013-01-18 21:54:48 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2013-01-25 20:51:52 +08:00
|
|
|
|
2013-01-26 14:19:14 +08:00
|
|
|
// The file wasn't found, return the file name passed in.
|
|
|
|
return pszFileName;
|
2013-01-25 20:51:52 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
const char* CCFileUtils::fullPathFromRelativeFile(const char *pszFilename, const char *pszRelativeFile)
|
|
|
|
{
|
|
|
|
std::string relativeFile = pszRelativeFile;
|
|
|
|
CCString *pRet = CCString::create("");
|
|
|
|
pRet->m_sString = relativeFile.substr(0, relativeFile.rfind('/')+1);
|
|
|
|
pRet->m_sString += getNewFilename(pszFilename);
|
|
|
|
return pRet->getCString();
|
2013-01-18 21:54:48 +08:00
|
|
|
}
|
|
|
|
|
2013-01-24 00:32:22 +08:00
|
|
|
std::string CCFileUtils::getPathForFilename(const std::string& filename, const std::string& resourceDirectory, const std::string& searchPath)
|
|
|
|
{
|
|
|
|
std::string file = filename;
|
|
|
|
std::string file_path = "";
|
|
|
|
size_t pos = filename.find_last_of("/");
|
|
|
|
if (pos != std::string::npos)
|
|
|
|
{
|
|
|
|
file_path = filename.substr(0, pos+1);
|
|
|
|
file = filename.substr(pos+1);
|
|
|
|
}
|
|
|
|
|
|
|
|
// searchPath + file_path + resourceDirectory
|
|
|
|
std::string path = searchPath;
|
|
|
|
if (path.size() > 0 && path[path.length()-1] != '/')
|
|
|
|
{
|
|
|
|
path += "/";
|
|
|
|
}
|
|
|
|
path += file_path;
|
|
|
|
path += resourceDirectory;
|
|
|
|
|
|
|
|
if (path.size() > 0 && path[path.length()-1] != '/')
|
|
|
|
{
|
|
|
|
path += "/";
|
|
|
|
}
|
|
|
|
path += file;
|
|
|
|
|
2013-01-25 21:52:35 +08:00
|
|
|
CCLOG("getPathForFilename, fullPath = %s", path.c_str());
|
|
|
|
return path;
|
2013-01-24 00:32:22 +08:00
|
|
|
}
|
2013-01-18 21:54:48 +08:00
|
|
|
|
2013-01-24 17:23:53 +08:00
|
|
|
unsigned char* CCFileUtils::getFileData(const char* pszFileName, const char* pszMode, unsigned long * pSize)
|
|
|
|
{
|
|
|
|
unsigned char * pData = 0;
|
|
|
|
|
|
|
|
if ((! pszFileName) || (! pszMode) || 0 == strlen(pszFileName))
|
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2013-01-25 20:51:52 +08:00
|
|
|
if (pszFileName[0] != '/')
|
|
|
|
{
|
|
|
|
CCLOG("GETTING FILE RELATIVE DATA: %s", pszFileName);
|
2013-01-28 23:19:57 +08:00
|
|
|
string fullPath = fullPathForFilename(pszFileName);
|
|
|
|
pData = s_pZipFile->getFileData(fullPath.c_str(), pSize);
|
2012-04-19 14:35:52 +08:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
do
|
|
|
|
{
|
|
|
|
// read rrom other path than user set it
|
2013-01-25 20:51:52 +08:00
|
|
|
CCLOG("GETTING FILE ABSOLUTE DATA: %s", pszFileName);
|
|
|
|
FILE *fp = fopen(pszFileName, pszMode);
|
2012-04-19 14:35:52 +08:00
|
|
|
CC_BREAK_IF(!fp);
|
|
|
|
|
|
|
|
unsigned long size;
|
|
|
|
fseek(fp,0,SEEK_END);
|
|
|
|
size = ftell(fp);
|
|
|
|
fseek(fp,0,SEEK_SET);
|
|
|
|
pData = new unsigned char[size];
|
|
|
|
size = fread(pData,sizeof(unsigned char), size,fp);
|
|
|
|
fclose(fp);
|
|
|
|
|
|
|
|
if (pSize)
|
|
|
|
{
|
|
|
|
*pSize = size;
|
|
|
|
}
|
|
|
|
} while (0);
|
|
|
|
}
|
|
|
|
|
2012-06-15 17:39:13 +08:00
|
|
|
if (! pData && isPopupNotify())
|
2012-04-19 14:35:52 +08:00
|
|
|
{
|
|
|
|
std::string title = "Notification";
|
|
|
|
std::string msg = "Get data from file(";
|
2013-01-25 20:51:52 +08:00
|
|
|
msg.append(pszFileName).append(") failed!");
|
2012-04-19 14:35:52 +08:00
|
|
|
CCMessageBox(msg.c_str(), title.c_str());
|
|
|
|
}
|
|
|
|
|
|
|
|
return pData;
|
|
|
|
}
|
|
|
|
|
2013-01-25 22:55:20 +08:00
|
|
|
void CCFileUtils::setResourceDirectory(const char* pszResourceDirectory)
|
|
|
|
{
|
2013-01-25 23:01:47 +08:00
|
|
|
if (pszResourceDirectory == NULL) return;
|
2013-01-25 22:55:20 +08:00
|
|
|
m_obDirectory = pszResourceDirectory;
|
2013-01-28 17:44:53 +08:00
|
|
|
std::vector<std::string> searchPaths = this->getSearchPath();;
|
|
|
|
searchPaths.insert(searchPaths.begin(), pszResourceDirectory);
|
|
|
|
this->setSearchPath(searchPaths);
|
|
|
|
}
|
|
|
|
|
|
|
|
void CCFileUtils::setSearchPath(const std::vector<std::string>& searchPaths)
|
|
|
|
{
|
|
|
|
bool bExistDefaultRootPath = false;
|
|
|
|
|
|
|
|
m_searchPathArray.clear();
|
|
|
|
for (std::vector<std::string>::const_iterator iter = searchPaths.begin(); iter != searchPaths.end(); ++iter)
|
2013-01-25 22:55:20 +08:00
|
|
|
{
|
2013-01-28 17:44:53 +08:00
|
|
|
std::string strPrefix;
|
|
|
|
std::string path;
|
|
|
|
if ((*iter)[0] != '/')
|
|
|
|
{ // Not an absolute path
|
|
|
|
if (iter->find(m_strDefaultResRootPath) != 0)
|
|
|
|
{ // The path contains no default resource root path, insert the root path.
|
|
|
|
strPrefix = m_strDefaultResRootPath;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
path = strPrefix+(*iter);
|
|
|
|
if (path.length() > 0 && path[path.length()-1] != '/')
|
|
|
|
{
|
|
|
|
path += "/";
|
|
|
|
}
|
|
|
|
if (!bExistDefaultRootPath && path == m_strDefaultResRootPath)
|
|
|
|
{
|
|
|
|
bExistDefaultRootPath = true;
|
|
|
|
}
|
|
|
|
m_searchPathArray.push_back(path);
|
2013-01-25 22:55:20 +08:00
|
|
|
}
|
2013-01-28 17:44:53 +08:00
|
|
|
|
|
|
|
if (!bExistDefaultRootPath)
|
2013-01-25 23:01:47 +08:00
|
|
|
{
|
2013-01-28 17:44:53 +08:00
|
|
|
CCLOG("Default root path doesn't exist, adding it.");
|
|
|
|
m_searchPathArray.push_back(m_strDefaultResRootPath);
|
2013-01-25 23:01:47 +08:00
|
|
|
}
|
2013-01-25 22:55:20 +08:00
|
|
|
}
|
|
|
|
|
2012-04-19 14:35:52 +08:00
|
|
|
string CCFileUtils::getWriteablePath()
|
|
|
|
{
|
2012-12-10 02:40:41 +08:00
|
|
|
// Fix for Nexus 10 (Android 4.2 multi-user environment)
|
|
|
|
// the path is retrieved through Java Context.getCacheDir() method
|
|
|
|
string dir("");
|
|
|
|
const char *tmp = getCacheDirectoryJNI();
|
2012-04-19 14:35:52 +08:00
|
|
|
|
|
|
|
if (tmp)
|
|
|
|
{
|
|
|
|
dir.append(tmp).append("/");
|
|
|
|
|
|
|
|
return dir;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_CC_END
|