add ResourceLoader
This commit is contained in:
parent
f6b1bca46a
commit
696de0326d
|
|
@ -102,6 +102,7 @@
|
||||||
<ClInclude Include="..\..\src\kiwano\utils\Json.h" />
|
<ClInclude Include="..\..\src\kiwano\utils\Json.h" />
|
||||||
<ClInclude Include="..\..\src\kiwano\utils\Logger.h" />
|
<ClInclude Include="..\..\src\kiwano\utils\Logger.h" />
|
||||||
<ClInclude Include="..\..\src\kiwano\utils\ResourceCache.h" />
|
<ClInclude Include="..\..\src\kiwano\utils\ResourceCache.h" />
|
||||||
|
<ClInclude Include="..\..\src\kiwano\utils\ResourceLoader.h" />
|
||||||
<ClInclude Include="..\..\src\kiwano\utils\Task.h" />
|
<ClInclude Include="..\..\src\kiwano\utils\Task.h" />
|
||||||
<ClInclude Include="..\..\src\kiwano\utils\TaskScheduler.h" />
|
<ClInclude Include="..\..\src\kiwano\utils\TaskScheduler.h" />
|
||||||
<ClInclude Include="..\..\src\kiwano\utils\Ticker.h" />
|
<ClInclude Include="..\..\src\kiwano\utils\Ticker.h" />
|
||||||
|
|
@ -179,6 +180,7 @@
|
||||||
<ClCompile Include="..\..\src\kiwano\utils\EventTicker.cpp" />
|
<ClCompile Include="..\..\src\kiwano\utils\EventTicker.cpp" />
|
||||||
<ClCompile Include="..\..\src\kiwano\utils\Logger.cpp" />
|
<ClCompile Include="..\..\src\kiwano\utils\Logger.cpp" />
|
||||||
<ClCompile Include="..\..\src\kiwano\utils\ResourceCache.cpp" />
|
<ClCompile Include="..\..\src\kiwano\utils\ResourceCache.cpp" />
|
||||||
|
<ClCompile Include="..\..\src\kiwano\utils\ResourceLoader.cpp" />
|
||||||
<ClCompile Include="..\..\src\kiwano\utils\Task.cpp" />
|
<ClCompile Include="..\..\src\kiwano\utils\Task.cpp" />
|
||||||
<ClCompile Include="..\..\src\kiwano\utils\TaskScheduler.cpp" />
|
<ClCompile Include="..\..\src\kiwano\utils\TaskScheduler.cpp" />
|
||||||
<ClCompile Include="..\..\src\kiwano\utils\Ticker.cpp" />
|
<ClCompile Include="..\..\src\kiwano\utils\Ticker.cpp" />
|
||||||
|
|
|
||||||
|
|
@ -354,6 +354,9 @@
|
||||||
<ClInclude Include="..\..\src\kiwano\base\RefPtr.h">
|
<ClInclude Include="..\..\src\kiwano\base\RefPtr.h">
|
||||||
<Filter>base</Filter>
|
<Filter>base</Filter>
|
||||||
</ClInclude>
|
</ClInclude>
|
||||||
|
<ClInclude Include="..\..\src\kiwano\utils\ResourceLoader.h">
|
||||||
|
<Filter>utils</Filter>
|
||||||
|
</ClInclude>
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
<ClCompile Include="..\..\src\kiwano\2d\Canvas.cpp">
|
<ClCompile Include="..\..\src\kiwano\2d\Canvas.cpp">
|
||||||
|
|
@ -578,6 +581,9 @@
|
||||||
<ClCompile Include="..\..\src\kiwano\base\RefObject.cpp">
|
<ClCompile Include="..\..\src\kiwano\base\RefObject.cpp">
|
||||||
<Filter>base</Filter>
|
<Filter>base</Filter>
|
||||||
</ClCompile>
|
</ClCompile>
|
||||||
|
<ClCompile Include="..\..\src\kiwano\utils\ResourceLoader.cpp">
|
||||||
|
<Filter>utils</Filter>
|
||||||
|
</ClCompile>
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
<None Include="suppress_warning.ruleset" />
|
<None Include="suppress_warning.ruleset" />
|
||||||
|
|
|
||||||
|
|
@ -183,7 +183,7 @@ void ObjectBase::SetStatus(const ObjectStatus& status)
|
||||||
|
|
||||||
void ObjectBase::Fail(const String& msg, int code)
|
void ObjectBase::Fail(const String& msg, int code)
|
||||||
{
|
{
|
||||||
SetStatus(ObjectStatus{ code, msg });
|
SetStatus(ObjectStatus(code, msg));
|
||||||
}
|
}
|
||||||
|
|
||||||
void ObjectBase::ClearStatus()
|
void ObjectBase::ClearStatus()
|
||||||
|
|
|
||||||
|
|
@ -39,6 +39,14 @@ struct ObjectStatus
|
||||||
int code = 0; ///< 状态码,等于 0 时为成功状态,否则为失败状态
|
int code = 0; ///< 状态码,等于 0 时为成功状态,否则为失败状态
|
||||||
String msg; ///< 状态信息
|
String msg; ///< 状态信息
|
||||||
|
|
||||||
|
ObjectStatus() = default;
|
||||||
|
|
||||||
|
ObjectStatus(int code, const String& msg)
|
||||||
|
: code(code)
|
||||||
|
, msg(msg)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
/// \~chinese
|
/// \~chinese
|
||||||
/// @brief 对象状态是否成功
|
/// @brief 对象状态是否成功
|
||||||
inline bool Success() const
|
inline bool Success() const
|
||||||
|
|
|
||||||
|
|
@ -128,6 +128,7 @@
|
||||||
|
|
||||||
#include <kiwano/utils/Logger.h>
|
#include <kiwano/utils/Logger.h>
|
||||||
#include <kiwano/utils/ResourceCache.h>
|
#include <kiwano/utils/ResourceCache.h>
|
||||||
|
#include <kiwano/utils/ResourceLoader.h>
|
||||||
#include <kiwano/utils/UserData.h>
|
#include <kiwano/utils/UserData.h>
|
||||||
#include <kiwano/utils/Timer.h>
|
#include <kiwano/utils/Timer.h>
|
||||||
#include <kiwano/utils/Ticker.h>
|
#include <kiwano/utils/Ticker.h>
|
||||||
|
|
|
||||||
|
|
@ -23,7 +23,6 @@
|
||||||
#include <kiwano/base/Director.h>
|
#include <kiwano/base/Director.h>
|
||||||
#include <kiwano/render/Renderer.h>
|
#include <kiwano/render/Renderer.h>
|
||||||
#include <kiwano/render/TextureCache.h>
|
#include <kiwano/render/TextureCache.h>
|
||||||
#include <kiwano/utils/ResourceCache.h>
|
|
||||||
#include <kiwano/utils/Logger.h>
|
#include <kiwano/utils/Logger.h>
|
||||||
|
|
||||||
namespace kiwano
|
namespace kiwano
|
||||||
|
|
@ -122,7 +121,6 @@ void Application::Destroy()
|
||||||
|
|
||||||
// Clear user resources
|
// Clear user resources
|
||||||
Director::GetInstance().ClearStages();
|
Director::GetInstance().ClearStages();
|
||||||
ResourceCache::GetInstance().Clear();
|
|
||||||
|
|
||||||
for (auto iter = modules_.rbegin(); iter != modules_.rend(); ++iter)
|
for (auto iter = modules_.rbegin(); iter != modules_.rend(); ++iter)
|
||||||
{
|
{
|
||||||
|
|
|
||||||
|
|
@ -168,8 +168,10 @@ void RendererImpl::CreateTexture(Texture& texture, const String& file_path)
|
||||||
|
|
||||||
if (!FileSystem::GetInstance().IsFileExists(file_path))
|
if (!FileSystem::GetInstance().IsFileExists(file_path))
|
||||||
{
|
{
|
||||||
KGE_WARNF("Texture file '%s' not found!", file_path.c_str());
|
|
||||||
hr = HRESULT_FROM_WIN32(ERROR_FILE_NOT_FOUND);
|
hr = HRESULT_FROM_WIN32(ERROR_FILE_NOT_FOUND);
|
||||||
|
KGE_SET_STATUS_IF_FAILED(hr, texture,
|
||||||
|
strings::Format("Texture file '%s' not found!", file_path.c_str()).c_str());
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (SUCCEEDED(hr))
|
if (SUCCEEDED(hr))
|
||||||
|
|
@ -273,8 +275,10 @@ void RendererImpl::CreateGifImage(GifImage& gif, const String& file_path)
|
||||||
|
|
||||||
if (!FileSystem::GetInstance().IsFileExists(file_path))
|
if (!FileSystem::GetInstance().IsFileExists(file_path))
|
||||||
{
|
{
|
||||||
KGE_WARNF("Gif texture file '%s' not found!", file_path.c_str());
|
|
||||||
hr = HRESULT_FROM_WIN32(ERROR_FILE_NOT_FOUND);
|
hr = HRESULT_FROM_WIN32(ERROR_FILE_NOT_FOUND);
|
||||||
|
KGE_SET_STATUS_IF_FAILED(hr, gif,
|
||||||
|
strings::Format("Gif texture file '%s' not found!", file_path.c_str()).c_str());
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (SUCCEEDED(hr))
|
if (SUCCEEDED(hr))
|
||||||
|
|
@ -495,8 +499,9 @@ void RendererImpl::CreateFontCollection(Font& font, const String& file_path)
|
||||||
{
|
{
|
||||||
if (!FileSystem::GetInstance().IsFileExists(file_path))
|
if (!FileSystem::GetInstance().IsFileExists(file_path))
|
||||||
{
|
{
|
||||||
KGE_WARNF("Font file '%s' not found!", file_path.c_str());
|
|
||||||
hr = HRESULT_FROM_WIN32(ERROR_FILE_NOT_FOUND);
|
hr = HRESULT_FROM_WIN32(ERROR_FILE_NOT_FOUND);
|
||||||
|
KGE_SET_STATUS_IF_FAILED(hr, font, strings::Format("Font file '%s' not found!", file_path.c_str()).c_str());
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -18,35 +18,10 @@
|
||||||
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
||||||
// THE SOFTWARE.
|
// THE SOFTWARE.
|
||||||
|
|
||||||
#include <fstream>
|
|
||||||
#include <kiwano/platform/FileSystem.h>
|
|
||||||
#include <kiwano/utils/Logger.h>
|
|
||||||
#include <kiwano/utils/ResourceCache.h>
|
#include <kiwano/utils/ResourceCache.h>
|
||||||
|
|
||||||
namespace kiwano
|
namespace kiwano
|
||||||
{
|
{
|
||||||
namespace resource_cache_01
|
|
||||||
{
|
|
||||||
|
|
||||||
bool LoadJsonData(ResourceCache* loader, const Json& json_data);
|
|
||||||
bool LoadXmlData(ResourceCache* loader, const XmlNode& elem);
|
|
||||||
|
|
||||||
} // namespace resource_cache_01
|
|
||||||
|
|
||||||
namespace
|
|
||||||
{
|
|
||||||
|
|
||||||
Map<String, Function<bool(ResourceCache*, const Json&)>> load_json_funcs = {
|
|
||||||
{ "latest", resource_cache_01::LoadJsonData },
|
|
||||||
{ "0.1", resource_cache_01::LoadJsonData },
|
|
||||||
};
|
|
||||||
|
|
||||||
Map<String, Function<bool(ResourceCache*, const XmlNode&)>> load_xml_funcs = {
|
|
||||||
{ "latest", resource_cache_01::LoadXmlData },
|
|
||||||
{ "0.1", resource_cache_01::LoadXmlData },
|
|
||||||
};
|
|
||||||
|
|
||||||
} // namespace
|
|
||||||
|
|
||||||
ResourceCache::ResourceCache() {}
|
ResourceCache::ResourceCache() {}
|
||||||
|
|
||||||
|
|
@ -55,124 +30,9 @@ ResourceCache::~ResourceCache()
|
||||||
Clear();
|
Clear();
|
||||||
}
|
}
|
||||||
|
|
||||||
bool ResourceCache::LoadFromJsonFile(const String& file_path)
|
void ResourceCache::AddObject(const String& id, ObjectBasePtr obj)
|
||||||
{
|
{
|
||||||
if (!FileSystem::GetInstance().IsFileExists(file_path))
|
object_cache_.insert(std::make_pair(id, obj));
|
||||||
{
|
|
||||||
KGE_ERRORF("%s failed: File not found.", __FUNCTION__);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
Json json_data;
|
|
||||||
|
|
||||||
std::ifstream ifs;
|
|
||||||
ifs.exceptions(std::ifstream::failbit | std::ifstream::badbit);
|
|
||||||
|
|
||||||
try
|
|
||||||
{
|
|
||||||
String full_path = FileSystem::GetInstance().GetFullPathForFile(file_path);
|
|
||||||
ifs.open(full_path.c_str());
|
|
||||||
ifs >> json_data;
|
|
||||||
ifs.close();
|
|
||||||
}
|
|
||||||
catch (std::ios_base::failure& e)
|
|
||||||
{
|
|
||||||
KGE_ERRORF("%s failed: cannot open file. (%s)", __FUNCTION__, e.what());
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
catch (Json::exception& e)
|
|
||||||
{
|
|
||||||
KGE_ERRORF("%s failed: cannot parse JSON. (%s)", __FUNCTION__, e.what());
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
return LoadFromJson(json_data);
|
|
||||||
}
|
|
||||||
|
|
||||||
bool ResourceCache::LoadFromJson(const Json& json_data)
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
String version = json_data["version"];
|
|
||||||
|
|
||||||
auto load = load_json_funcs.find(version);
|
|
||||||
if (load != load_json_funcs.end())
|
|
||||||
{
|
|
||||||
return load->second(this, json_data);
|
|
||||||
}
|
|
||||||
else if (version.empty())
|
|
||||||
{
|
|
||||||
return load_json_funcs["latest"](this, json_data);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
KGE_ERRORF("%s failed: unknown resource data version", __FUNCTION__);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
catch (Json::exception& e)
|
|
||||||
{
|
|
||||||
KGE_ERRORF("%s failed: JSON data is invalid. (%s)", __FUNCTION__, e.what());
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool ResourceCache::LoadFromXmlFile(const String& file_path)
|
|
||||||
{
|
|
||||||
if (!FileSystem::GetInstance().IsFileExists(file_path))
|
|
||||||
{
|
|
||||||
KGE_ERRORF("%s failed: File not found.", __FUNCTION__);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
String full_path = FileSystem::GetInstance().GetFullPathForFile(file_path);
|
|
||||||
|
|
||||||
XmlDocument doc;
|
|
||||||
|
|
||||||
auto result = doc.load_file(full_path.c_str());
|
|
||||||
if (result)
|
|
||||||
{
|
|
||||||
return LoadFromXml(doc);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
KGE_ERRORF("%s failed: XML [%s] parsed with errors: %s", __FUNCTION__, full_path.c_str(), result.description());
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
bool ResourceCache::LoadFromXml(const XmlDocument& doc)
|
|
||||||
{
|
|
||||||
if (XmlNode root = doc.child("resources"))
|
|
||||||
{
|
|
||||||
String version;
|
|
||||||
if (auto version_node = root.child("version"))
|
|
||||||
version = version_node.child_value();
|
|
||||||
|
|
||||||
auto load = load_xml_funcs.find(version);
|
|
||||||
if (load != load_xml_funcs.end())
|
|
||||||
{
|
|
||||||
return load->second(this, root);
|
|
||||||
}
|
|
||||||
else if (version.empty())
|
|
||||||
{
|
|
||||||
return load_xml_funcs["latest"](this, root);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
KGE_ERRORF("%s failed: unknown resource data version", __FUNCTION__);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool ResourceCache::AddObject(const String& id, ObjectBasePtr obj)
|
|
||||||
{
|
|
||||||
if (obj)
|
|
||||||
{
|
|
||||||
object_cache_.insert(std::make_pair(id, obj));
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void ResourceCache::Remove(const String& id)
|
void ResourceCache::Remove(const String& id)
|
||||||
|
|
@ -194,272 +54,3 @@ ObjectBasePtr ResourceCache::Get(const String& id) const
|
||||||
}
|
}
|
||||||
|
|
||||||
} // namespace kiwano
|
} // namespace kiwano
|
||||||
|
|
||||||
namespace kiwano
|
|
||||||
{
|
|
||||||
namespace resource_cache_01
|
|
||||||
{
|
|
||||||
struct GlobalData
|
|
||||||
{
|
|
||||||
String path;
|
|
||||||
};
|
|
||||||
|
|
||||||
bool LoadTexturesFromData(ResourceCache* loader, GlobalData* gdata, const String& id, const String& type,
|
|
||||||
const String& file)
|
|
||||||
{
|
|
||||||
if (!gdata)
|
|
||||||
return false;
|
|
||||||
|
|
||||||
if (type == "gif")
|
|
||||||
{
|
|
||||||
// GIF image
|
|
||||||
GifImagePtr gif = MakePtr<GifImage>();
|
|
||||||
if (gif && gif->Load(gdata->path + file))
|
|
||||||
{
|
|
||||||
return loader->AddObject(id, gif);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!file.empty())
|
|
||||||
{
|
|
||||||
// Simple image
|
|
||||||
FramePtr frame = MakePtr<Frame>();
|
|
||||||
if (frame && frame->Load(gdata->path + file))
|
|
||||||
{
|
|
||||||
return loader->AddObject(id, frame);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool LoadTexturesFromData(ResourceCache* loader, GlobalData* gdata, const String& id, const Vector<String>& files)
|
|
||||||
{
|
|
||||||
if (!gdata)
|
|
||||||
return false;
|
|
||||||
|
|
||||||
if (files.empty())
|
|
||||||
return true;
|
|
||||||
|
|
||||||
// Frames
|
|
||||||
Vector<FramePtr> frames;
|
|
||||||
frames.reserve(files.size());
|
|
||||||
for (const auto& file : files)
|
|
||||||
{
|
|
||||||
FramePtr frame = MakePtr<Frame>();
|
|
||||||
if (frame->Load(gdata->path + file))
|
|
||||||
{
|
|
||||||
frames.push_back(frame);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
FrameSequencePtr frame_seq = MakePtr<FrameSequence>(frames);
|
|
||||||
if (frame_seq)
|
|
||||||
{
|
|
||||||
return !!loader->AddObject(id, frame_seq);
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool LoadTexturesFromData(ResourceCache* loader, GlobalData* gdata, const String& id, const String& file, int rows,
|
|
||||||
int cols, int max_num, float padding_x, float padding_y)
|
|
||||||
{
|
|
||||||
if (!gdata)
|
|
||||||
return false;
|
|
||||||
|
|
||||||
if (!file.empty())
|
|
||||||
{
|
|
||||||
if (rows || cols)
|
|
||||||
{
|
|
||||||
// Frame slices
|
|
||||||
FramePtr frame = MakePtr<Frame>();
|
|
||||||
if (frame && frame->Load(gdata->path + file))
|
|
||||||
{
|
|
||||||
FrameSequencePtr frame_seq = MakePtr<FrameSequence>();
|
|
||||||
if (frame_seq)
|
|
||||||
{
|
|
||||||
frame_seq->AddFrames(frame, cols, rows, max_num, padding_x, padding_y);
|
|
||||||
return loader->AddObject(id, frame_seq);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
// Simple image
|
|
||||||
FramePtr frame = MakePtr<Frame>();
|
|
||||||
if (frame && frame->Load(gdata->path + file))
|
|
||||||
{
|
|
||||||
return loader->AddObject(id, frame);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool LoadFontsFromData(ResourceCache* loader, GlobalData* gdata, const String& id, const String& file)
|
|
||||||
{
|
|
||||||
if (!gdata)
|
|
||||||
return false;
|
|
||||||
|
|
||||||
FontPtr font = MakePtr<Font>();
|
|
||||||
if (font && font->Load(gdata->path + file))
|
|
||||||
{
|
|
||||||
return loader->AddObject(id, font);
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool LoadJsonData(ResourceCache* loader, const Json& json_data)
|
|
||||||
{
|
|
||||||
GlobalData global_data;
|
|
||||||
if (json_data.count("path"))
|
|
||||||
{
|
|
||||||
global_data.path = json_data["path"].get<String>();
|
|
||||||
}
|
|
||||||
|
|
||||||
if (json_data.count("images"))
|
|
||||||
{
|
|
||||||
for (const auto& image : json_data["images"])
|
|
||||||
{
|
|
||||||
String id, type, file;
|
|
||||||
int rows = 0, cols = 0, max_num = -1;
|
|
||||||
|
|
||||||
if (image.count("id"))
|
|
||||||
id = image["id"].get<String>();
|
|
||||||
if (image.count("type"))
|
|
||||||
type = image["type"].get<String>();
|
|
||||||
if (image.count("file"))
|
|
||||||
file = image["file"].get<String>();
|
|
||||||
if (image.count("rows"))
|
|
||||||
rows = image["rows"].get<int>();
|
|
||||||
if (image.count("cols"))
|
|
||||||
cols = image["cols"].get<int>();
|
|
||||||
if (image.count("max_num"))
|
|
||||||
max_num = image["max_num"].get<int>();
|
|
||||||
|
|
||||||
if (rows || cols)
|
|
||||||
{
|
|
||||||
float padding_x = 0, padding_y = 0;
|
|
||||||
if (image.count("padding-x"))
|
|
||||||
padding_x = image["padding-x"].get<float>();
|
|
||||||
if (image.count("padding-y"))
|
|
||||||
padding_y = image["padding-y"].get<float>();
|
|
||||||
|
|
||||||
if (!LoadTexturesFromData(loader, &global_data, id, file, rows, cols, max_num, padding_x, padding_y))
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (image.count("files"))
|
|
||||||
{
|
|
||||||
Vector<String> files;
|
|
||||||
files.reserve(image["files"].size());
|
|
||||||
for (const auto& file : image["files"])
|
|
||||||
{
|
|
||||||
files.push_back(file.get<String>());
|
|
||||||
}
|
|
||||||
if (!LoadTexturesFromData(loader, &global_data, id, files))
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
if (!LoadTexturesFromData(loader, &global_data, id, type, file))
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (json_data.count("fonts"))
|
|
||||||
{
|
|
||||||
for (const auto& font : json_data["fonts"])
|
|
||||||
{
|
|
||||||
String id, file;
|
|
||||||
|
|
||||||
if (font.count("id"))
|
|
||||||
id = font["id"].get<String>();
|
|
||||||
if (font.count("file"))
|
|
||||||
file = font["file"].get<String>();
|
|
||||||
|
|
||||||
if (!LoadFontsFromData(loader, &global_data, id, file))
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool LoadXmlData(ResourceCache* loader, const XmlNode& elem)
|
|
||||||
{
|
|
||||||
GlobalData global_data;
|
|
||||||
if (auto path = elem.child("path"))
|
|
||||||
{
|
|
||||||
global_data.path = path.child_value();
|
|
||||||
}
|
|
||||||
|
|
||||||
if (auto images = elem.child("images"))
|
|
||||||
{
|
|
||||||
for (auto image : images.children())
|
|
||||||
{
|
|
||||||
String id, type, file;
|
|
||||||
int rows = 0, cols = 0, max_num = -1;
|
|
||||||
|
|
||||||
if (auto attr = image.attribute("id"))
|
|
||||||
id = attr.value();
|
|
||||||
if (auto attr = image.attribute("type"))
|
|
||||||
type = attr.value();
|
|
||||||
if (auto attr = image.attribute("file"))
|
|
||||||
file = attr.value();
|
|
||||||
if (auto attr = image.attribute("rows"))
|
|
||||||
rows = attr.as_int(0);
|
|
||||||
if (auto attr = image.attribute("cols"))
|
|
||||||
cols = attr.as_int(0);
|
|
||||||
if (auto attr = image.attribute("max_num"))
|
|
||||||
max_num = attr.as_int(-1);
|
|
||||||
|
|
||||||
if (rows || cols)
|
|
||||||
{
|
|
||||||
float padding_x = 0, padding_y = 0;
|
|
||||||
if (auto attr = image.attribute("padding-x"))
|
|
||||||
padding_x = attr.as_float(0.0f);
|
|
||||||
if (auto attr = image.attribute("padding-y"))
|
|
||||||
padding_y = attr.as_float(0.0f);
|
|
||||||
|
|
||||||
if (!LoadTexturesFromData(loader, &global_data, id, file, rows, cols, max_num, padding_x, padding_y))
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (file.empty() && !image.empty())
|
|
||||||
{
|
|
||||||
Vector<String> files_arr;
|
|
||||||
for (auto file : image.children())
|
|
||||||
{
|
|
||||||
if (auto path = file.attribute("path"))
|
|
||||||
{
|
|
||||||
files_arr.push_back(path.value());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (!LoadTexturesFromData(loader, &global_data, id, files_arr))
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
if (!LoadTexturesFromData(loader, &global_data, id, type, file))
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (auto fonts = elem.child("fonts"))
|
|
||||||
{
|
|
||||||
for (auto font : fonts.children())
|
|
||||||
{
|
|
||||||
String id, file;
|
|
||||||
if (auto attr = font.attribute("id"))
|
|
||||||
id = attr.value();
|
|
||||||
if (auto attr = font.attribute("file"))
|
|
||||||
file = attr.value();
|
|
||||||
|
|
||||||
if (!LoadFontsFromData(loader, &global_data, id, file))
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
} // namespace resource_cache_01
|
|
||||||
} // namespace kiwano
|
|
||||||
|
|
|
||||||
|
|
@ -20,8 +20,6 @@
|
||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
#include <kiwano/core/Resource.h>
|
#include <kiwano/core/Resource.h>
|
||||||
#include <kiwano/utils/Json.h>
|
|
||||||
#include <kiwano/utils/Xml.h>
|
|
||||||
#include <kiwano/render/Frame.h>
|
#include <kiwano/render/Frame.h>
|
||||||
#include <kiwano/render/FrameSequence.h>
|
#include <kiwano/render/FrameSequence.h>
|
||||||
#include <kiwano/render/Font.h>
|
#include <kiwano/render/Font.h>
|
||||||
|
|
@ -29,33 +27,15 @@
|
||||||
|
|
||||||
namespace kiwano
|
namespace kiwano
|
||||||
{
|
{
|
||||||
|
|
||||||
|
KGE_DECLARE_SMART_PTR(ResourceCache);
|
||||||
|
|
||||||
/// \~chinese
|
/// \~chinese
|
||||||
/// @brief 资源缓存
|
/// @brief 资源缓存
|
||||||
/// @details 资源缓存
|
class KGE_API ResourceCache final : public ObjectBase
|
||||||
class KGE_API ResourceCache final : public Singleton<ResourceCache>
|
|
||||||
{
|
{
|
||||||
friend Singleton<ResourceCache>;
|
|
||||||
|
|
||||||
public:
|
public:
|
||||||
/// \~chinese
|
ResourceCache();
|
||||||
/// @brief 从 JSON 文件加载资源信息
|
|
||||||
/// @param file_path JSON文件路径
|
|
||||||
bool LoadFromJsonFile(const String& file_path);
|
|
||||||
|
|
||||||
/// \~chinese
|
|
||||||
/// @brief 从 JSON 加载资源信息
|
|
||||||
/// @param json_data JSON对象
|
|
||||||
bool LoadFromJson(const Json& json_data);
|
|
||||||
|
|
||||||
/// \~chinese
|
|
||||||
/// @brief 从 XML 文件加载资源信息
|
|
||||||
/// @param file_path XML文件路径
|
|
||||||
bool LoadFromXmlFile(const String& file_path);
|
|
||||||
|
|
||||||
/// \~chinese
|
|
||||||
/// @brief 从 XML 文档对象加载资源信息
|
|
||||||
/// @param doc XML文档对象
|
|
||||||
bool LoadFromXml(const XmlDocument& doc);
|
|
||||||
|
|
||||||
/// \~chinese
|
/// \~chinese
|
||||||
/// @brief 获取资源
|
/// @brief 获取资源
|
||||||
|
|
@ -77,7 +57,7 @@ public:
|
||||||
/// @brief 将对象放入缓存
|
/// @brief 将对象放入缓存
|
||||||
/// @param id 对象ID
|
/// @param id 对象ID
|
||||||
/// @param obj 对象
|
/// @param obj 对象
|
||||||
bool AddObject(const String& id, ObjectBasePtr obj);
|
void AddObject(const String& id, ObjectBasePtr obj);
|
||||||
|
|
||||||
/// \~chinese
|
/// \~chinese
|
||||||
/// @brief 删除指定资源
|
/// @brief 删除指定资源
|
||||||
|
|
@ -90,10 +70,8 @@ public:
|
||||||
|
|
||||||
virtual ~ResourceCache();
|
virtual ~ResourceCache();
|
||||||
|
|
||||||
private:
|
|
||||||
ResourceCache();
|
|
||||||
|
|
||||||
private:
|
private:
|
||||||
UnorderedMap<String, ObjectBasePtr> object_cache_;
|
UnorderedMap<String, ObjectBasePtr> object_cache_;
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace kiwano
|
} // namespace kiwano
|
||||||
|
|
|
||||||
|
|
@ -0,0 +1,452 @@
|
||||||
|
// Copyright (c) 2016-2018 Kiwano - Nomango
|
||||||
|
//
|
||||||
|
// 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 <fstream>
|
||||||
|
#include <kiwano/platform/FileSystem.h>
|
||||||
|
#include <kiwano/utils/Logger.h>
|
||||||
|
#include <kiwano/utils/ResourceLoader.h>
|
||||||
|
|
||||||
|
namespace kiwano
|
||||||
|
{
|
||||||
|
namespace resource_cache_01
|
||||||
|
{
|
||||||
|
|
||||||
|
void LoadJsonData(ResourceCache* cache, const Json& json_data);
|
||||||
|
void LoadXmlData(ResourceCache* cache, const XmlNode& elem);
|
||||||
|
|
||||||
|
} // namespace resource_cache_01
|
||||||
|
|
||||||
|
namespace
|
||||||
|
{
|
||||||
|
|
||||||
|
Map<String, Function<void(ResourceCache*, const Json&)>> load_json_funcs = {
|
||||||
|
{ "latest", resource_cache_01::LoadJsonData },
|
||||||
|
{ "0.1", resource_cache_01::LoadJsonData },
|
||||||
|
};
|
||||||
|
|
||||||
|
Map<String, Function<void(ResourceCache*, const XmlNode&)>> load_xml_funcs = {
|
||||||
|
{ "latest", resource_cache_01::LoadXmlData },
|
||||||
|
{ "0.1", resource_cache_01::LoadXmlData },
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace
|
||||||
|
|
||||||
|
bool ResourceLoader::LoadFromJsonFile(ResourceCachePtr cache, const String& file_path)
|
||||||
|
{
|
||||||
|
if (!cache)
|
||||||
|
{
|
||||||
|
KGE_ERROR("ResourceLoader::LoadFromJsonFile failed, cache is nullptr");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!FileSystem::GetInstance().IsFileExists(file_path))
|
||||||
|
{
|
||||||
|
cache->Fail(
|
||||||
|
strings::Format("ResourceLoader::LoadFromJsonFile failed: [%s] file not found.", file_path.c_str()));
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
Json json_data;
|
||||||
|
|
||||||
|
std::ifstream ifs;
|
||||||
|
ifs.exceptions(std::ifstream::failbit | std::ifstream::badbit);
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
String full_path = FileSystem::GetInstance().GetFullPathForFile(file_path);
|
||||||
|
ifs.open(full_path.c_str());
|
||||||
|
ifs >> json_data;
|
||||||
|
ifs.close();
|
||||||
|
}
|
||||||
|
catch (std::ios_base::failure& e)
|
||||||
|
{
|
||||||
|
cache->Fail(strings::Format("ResourceLoader::LoadFromJsonFile failed: cannot open file [%s]. %s",
|
||||||
|
file_path.c_str(), e.what()));
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
catch (Json::exception& e)
|
||||||
|
{
|
||||||
|
cache->Fail(strings::Format("ResourceLoader::LoadFromJsonFile failed: Json file [%s] parsed with errors: %s",
|
||||||
|
file_path.c_str(), e.what()));
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return LoadFromJson(cache, json_data);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool ResourceLoader::LoadFromJson(ResourceCachePtr cache, const Json& json_data)
|
||||||
|
{
|
||||||
|
if (!cache)
|
||||||
|
{
|
||||||
|
KGE_ERROR("ResourceLoader::LoadFromJson failed, cache is nullptr");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
String version = json_data["version"];
|
||||||
|
|
||||||
|
auto load = load_json_funcs.find(version);
|
||||||
|
if (load != load_json_funcs.end())
|
||||||
|
{
|
||||||
|
load->second(cache.Get(), json_data);
|
||||||
|
}
|
||||||
|
else if (version.empty())
|
||||||
|
{
|
||||||
|
load_json_funcs["latest"](cache.Get(), json_data);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
cache->Fail("ResourceLoader::LoadFromJson failed: unknown resource data version");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
catch (Json::exception& e)
|
||||||
|
{
|
||||||
|
cache->Fail(String("ResourceLoader::LoadFromJson failed: ") + e.what());
|
||||||
|
}
|
||||||
|
return cache->IsValid();
|
||||||
|
}
|
||||||
|
|
||||||
|
bool ResourceLoader::LoadFromXmlFile(ResourceCachePtr cache, const String& file_path)
|
||||||
|
{
|
||||||
|
if (!cache)
|
||||||
|
{
|
||||||
|
KGE_ERROR("ResourceLoader::LoadFromXmlFile failed, cache is nullptr");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!FileSystem::GetInstance().IsFileExists(file_path))
|
||||||
|
{
|
||||||
|
cache->Fail(strings::Format("ResourceLoader::LoadFromXmlFile failed: [%s] file not found.", file_path.c_str()));
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
String full_path = FileSystem::GetInstance().GetFullPathForFile(file_path);
|
||||||
|
|
||||||
|
XmlDocument doc;
|
||||||
|
|
||||||
|
auto result = doc.load_file(full_path.c_str());
|
||||||
|
if (result)
|
||||||
|
{
|
||||||
|
return LoadFromXml(cache, doc);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
cache->Fail(strings::Format("ResourceLoader::LoadFromXmlFile failed: XML file [%s] parsed with errors: %s",
|
||||||
|
file_path.c_str(), result.description()));
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
bool ResourceLoader::LoadFromXml(ResourceCachePtr cache, const XmlDocument& doc)
|
||||||
|
{
|
||||||
|
if (!cache)
|
||||||
|
{
|
||||||
|
KGE_ERROR("ResourceLoader::LoadFromXml failed, cache is nullptr");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (XmlNode root = doc.child("resources"))
|
||||||
|
{
|
||||||
|
String version;
|
||||||
|
if (auto version_node = root.child("version"))
|
||||||
|
version = version_node.child_value();
|
||||||
|
|
||||||
|
auto load = load_xml_funcs.find(version);
|
||||||
|
if (load != load_xml_funcs.end())
|
||||||
|
{
|
||||||
|
load->second(cache.Get(), root);
|
||||||
|
}
|
||||||
|
else if (version.empty())
|
||||||
|
{
|
||||||
|
load_xml_funcs["latest"](cache.Get(), root);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
cache->Fail("ResourceLoader::LoadFromXml failed: unknown resource data version");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
cache->Fail("ResourceLoader::LoadFromXml failed: unknown file format");
|
||||||
|
}
|
||||||
|
return cache->IsValid();
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace kiwano
|
||||||
|
|
||||||
|
namespace kiwano
|
||||||
|
{
|
||||||
|
namespace resource_cache_01
|
||||||
|
{
|
||||||
|
struct GlobalData
|
||||||
|
{
|
||||||
|
String path;
|
||||||
|
};
|
||||||
|
|
||||||
|
void LoadTexturesFromData(ResourceCache* cache, GlobalData* gdata, const String& id, const String& type,
|
||||||
|
const String& file)
|
||||||
|
{
|
||||||
|
if (type == "gif")
|
||||||
|
{
|
||||||
|
// GIF image
|
||||||
|
GifImagePtr gif = MakePtr<GifImage>();
|
||||||
|
if (gif && gif->Load(gdata->path + file))
|
||||||
|
{
|
||||||
|
cache->AddObject(id, gif);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if (!file.empty())
|
||||||
|
{
|
||||||
|
// Simple image
|
||||||
|
FramePtr frame = MakePtr<Frame>();
|
||||||
|
if (frame && frame->Load(gdata->path + file))
|
||||||
|
{
|
||||||
|
cache->AddObject(id, frame);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
cache->Fail(strings::Format("%s failed", __FUNCTION__));
|
||||||
|
}
|
||||||
|
|
||||||
|
void LoadTexturesFromData(ResourceCache* cache, GlobalData* gdata, const String& id, const Vector<String>& files)
|
||||||
|
{
|
||||||
|
if (files.empty())
|
||||||
|
return;
|
||||||
|
|
||||||
|
// Frames
|
||||||
|
Vector<FramePtr> frames;
|
||||||
|
frames.reserve(files.size());
|
||||||
|
for (const auto& file : files)
|
||||||
|
{
|
||||||
|
FramePtr frame = MakePtr<Frame>();
|
||||||
|
if (frame->Load(gdata->path + file))
|
||||||
|
{
|
||||||
|
frames.push_back(frame);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!frames.empty())
|
||||||
|
{
|
||||||
|
FrameSequencePtr frame_seq = MakePtr<FrameSequence>(frames);
|
||||||
|
if (frame_seq)
|
||||||
|
{
|
||||||
|
cache->AddObject(id, frame_seq);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
cache->Fail(strings::Format("%s failed", __FUNCTION__));
|
||||||
|
}
|
||||||
|
|
||||||
|
void LoadTexturesFromData(ResourceCache* cache, GlobalData* gdata, const String& id, const String& file, int rows,
|
||||||
|
int cols, int max_num, float padding_x, float padding_y)
|
||||||
|
{
|
||||||
|
if (!file.empty())
|
||||||
|
{
|
||||||
|
if (rows || cols)
|
||||||
|
{
|
||||||
|
// Frame slices
|
||||||
|
FramePtr frame = MakePtr<Frame>();
|
||||||
|
if (frame && frame->Load(gdata->path + file))
|
||||||
|
{
|
||||||
|
FrameSequencePtr frame_seq = MakePtr<FrameSequence>();
|
||||||
|
if (frame_seq)
|
||||||
|
{
|
||||||
|
frame_seq->AddFrames(frame, cols, rows, max_num, padding_x, padding_y);
|
||||||
|
cache->AddObject(id, frame_seq);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
// Simple image
|
||||||
|
FramePtr frame = MakePtr<Frame>();
|
||||||
|
if (frame && frame->Load(gdata->path + file))
|
||||||
|
{
|
||||||
|
cache->AddObject(id, frame);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
cache->Fail(strings::Format("%s failed", __FUNCTION__));
|
||||||
|
}
|
||||||
|
|
||||||
|
void LoadFontsFromData(ResourceCache* cache, GlobalData* gdata, const String& id, const String& file)
|
||||||
|
{
|
||||||
|
FontPtr font = MakePtr<Font>();
|
||||||
|
if (font && font->Load(gdata->path + file))
|
||||||
|
{
|
||||||
|
cache->AddObject(id, font);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
cache->Fail(strings::Format("%s failed", __FUNCTION__));
|
||||||
|
}
|
||||||
|
|
||||||
|
void LoadJsonData(ResourceCache* cache, const Json& json_data)
|
||||||
|
{
|
||||||
|
GlobalData global_data;
|
||||||
|
if (json_data.count("path"))
|
||||||
|
{
|
||||||
|
global_data.path = json_data["path"].get<String>();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (json_data.count("images"))
|
||||||
|
{
|
||||||
|
for (const auto& image : json_data["images"])
|
||||||
|
{
|
||||||
|
String id, type, file;
|
||||||
|
int rows = 0, cols = 0, max_num = -1;
|
||||||
|
|
||||||
|
if (image.count("id"))
|
||||||
|
id = image["id"].get<String>();
|
||||||
|
if (image.count("type"))
|
||||||
|
type = image["type"].get<String>();
|
||||||
|
if (image.count("file"))
|
||||||
|
file = image["file"].get<String>();
|
||||||
|
if (image.count("rows"))
|
||||||
|
rows = image["rows"].get<int>();
|
||||||
|
if (image.count("cols"))
|
||||||
|
cols = image["cols"].get<int>();
|
||||||
|
if (image.count("max_num"))
|
||||||
|
max_num = image["max_num"].get<int>();
|
||||||
|
|
||||||
|
if (rows || cols)
|
||||||
|
{
|
||||||
|
float padding_x = 0, padding_y = 0;
|
||||||
|
if (image.count("padding-x"))
|
||||||
|
padding_x = image["padding-x"].get<float>();
|
||||||
|
if (image.count("padding-y"))
|
||||||
|
padding_y = image["padding-y"].get<float>();
|
||||||
|
|
||||||
|
LoadTexturesFromData(cache, &global_data, id, file, rows, cols, max_num, padding_x, padding_y);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (image.count("files"))
|
||||||
|
{
|
||||||
|
Vector<String> files;
|
||||||
|
files.reserve(image["files"].size());
|
||||||
|
for (const auto& file : image["files"])
|
||||||
|
{
|
||||||
|
files.push_back(file.get<String>());
|
||||||
|
}
|
||||||
|
LoadTexturesFromData(cache, &global_data, id, files);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
LoadTexturesFromData(cache, &global_data, id, type, file);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (json_data.count("fonts"))
|
||||||
|
{
|
||||||
|
for (const auto& font : json_data["fonts"])
|
||||||
|
{
|
||||||
|
String id, file;
|
||||||
|
|
||||||
|
if (font.count("id"))
|
||||||
|
id = font["id"].get<String>();
|
||||||
|
if (font.count("file"))
|
||||||
|
file = font["file"].get<String>();
|
||||||
|
|
||||||
|
LoadFontsFromData(cache, &global_data, id, file);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void LoadXmlData(ResourceCache* cache, const XmlNode& elem)
|
||||||
|
{
|
||||||
|
GlobalData global_data;
|
||||||
|
if (auto path = elem.child("path"))
|
||||||
|
{
|
||||||
|
global_data.path = path.child_value();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (auto images = elem.child("images"))
|
||||||
|
{
|
||||||
|
for (auto image : images.children())
|
||||||
|
{
|
||||||
|
String id, type, file;
|
||||||
|
int rows = 0, cols = 0, max_num = -1;
|
||||||
|
|
||||||
|
if (auto attr = image.attribute("id"))
|
||||||
|
id = attr.value();
|
||||||
|
if (auto attr = image.attribute("type"))
|
||||||
|
type = attr.value();
|
||||||
|
if (auto attr = image.attribute("file"))
|
||||||
|
file = attr.value();
|
||||||
|
if (auto attr = image.attribute("rows"))
|
||||||
|
rows = attr.as_int(0);
|
||||||
|
if (auto attr = image.attribute("cols"))
|
||||||
|
cols = attr.as_int(0);
|
||||||
|
if (auto attr = image.attribute("max_num"))
|
||||||
|
max_num = attr.as_int(-1);
|
||||||
|
|
||||||
|
if (rows || cols)
|
||||||
|
{
|
||||||
|
float padding_x = 0, padding_y = 0;
|
||||||
|
if (auto attr = image.attribute("padding-x"))
|
||||||
|
padding_x = attr.as_float(0.0f);
|
||||||
|
if (auto attr = image.attribute("padding-y"))
|
||||||
|
padding_y = attr.as_float(0.0f);
|
||||||
|
|
||||||
|
LoadTexturesFromData(cache, &global_data, id, file, rows, cols, max_num, padding_x, padding_y);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (file.empty() && !image.empty())
|
||||||
|
{
|
||||||
|
Vector<String> files_arr;
|
||||||
|
for (auto file : image.children())
|
||||||
|
{
|
||||||
|
if (auto path = file.attribute("path"))
|
||||||
|
{
|
||||||
|
files_arr.push_back(path.value());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
LoadTexturesFromData(cache, &global_data, id, files_arr);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
LoadTexturesFromData(cache, &global_data, id, type, file);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (auto fonts = elem.child("fonts"))
|
||||||
|
{
|
||||||
|
for (auto font : fonts.children())
|
||||||
|
{
|
||||||
|
String id, file;
|
||||||
|
if (auto attr = font.attribute("id"))
|
||||||
|
id = attr.value();
|
||||||
|
if (auto attr = font.attribute("file"))
|
||||||
|
file = attr.value();
|
||||||
|
|
||||||
|
LoadFontsFromData(cache, &global_data, id, file);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace resource_cache_01
|
||||||
|
} // namespace kiwano
|
||||||
|
|
@ -0,0 +1,55 @@
|
||||||
|
// Copyright (c) 2016-2018 Kiwano - Nomango
|
||||||
|
//
|
||||||
|
// 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.
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
#include <kiwano/utils/ResourceCache.h>
|
||||||
|
#include <kiwano/utils/Json.h>
|
||||||
|
#include <kiwano/utils/Xml.h>
|
||||||
|
|
||||||
|
namespace kiwano
|
||||||
|
{
|
||||||
|
|
||||||
|
/// \~chinese
|
||||||
|
/// @brief 资源加载器
|
||||||
|
class KGE_API ResourceLoader final : Noncopyable
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
/// \~chinese
|
||||||
|
/// @brief 从 JSON 文件加载资源信息
|
||||||
|
/// @param file_path JSON文件路径
|
||||||
|
static bool LoadFromJsonFile(ResourceCachePtr cache, const String& file_path);
|
||||||
|
|
||||||
|
/// \~chinese
|
||||||
|
/// @brief 从 JSON 加载资源信息
|
||||||
|
/// @param json_data JSON对象
|
||||||
|
static bool LoadFromJson(ResourceCachePtr cache, const Json& json_data);
|
||||||
|
|
||||||
|
/// \~chinese
|
||||||
|
/// @brief 从 XML 文件加载资源信息
|
||||||
|
/// @param file_path XML文件路径
|
||||||
|
static bool LoadFromXmlFile(ResourceCachePtr cache, const String& file_path);
|
||||||
|
|
||||||
|
/// \~chinese
|
||||||
|
/// @brief 从 XML 文档对象加载资源信息
|
||||||
|
/// @param doc XML文档对象
|
||||||
|
static bool LoadFromXml(ResourceCachePtr cache, const XmlDocument& doc);
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace kiwano
|
||||||
Loading…
Reference in New Issue