spine-runtimes/spine-godot/spine_godot/SpineSkeletonFileResource.cpp

217 lines
7.3 KiB
C++

/******************************************************************************
* Spine Runtimes License Agreement
* Last updated July 28, 2023. Replaces all prior versions.
*
* Copyright (c) 2013-2023, Esoteric Software LLC
*
* Integration of the Spine Runtimes into software or otherwise creating
* derivative works of the Spine Runtimes is permitted under the terms and
* conditions of Section 2 of the Spine Editor License Agreement:
* http://esotericsoftware.com/spine-editor-license
*
* Otherwise, it is permitted to integrate the Spine Runtimes into software or
* otherwise create derivative works of the Spine Runtimes (collectively,
* "Products"), provided that each user of the Products must obtain their own
* Spine Editor license and redistribution of the Products in any form must
* include this license and copyright notice.
*
* THE SPINE RUNTIMES ARE PROVIDED BY ESOTERIC SOFTWARE LLC "AS IS" AND ANY
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
* DISCLAIMED. IN NO EVENT SHALL ESOTERIC SOFTWARE LLC BE LIABLE FOR ANY
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES,
* BUSINESS INTERRUPTION, OR LOSS OF USE, DATA, OR PROFITS) HOWEVER CAUSED AND
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THE
* SPINE RUNTIMES, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*****************************************************************************/
#include "SpineSkeletonFileResource.h"
#include "core/error/error_list.h"
#include "core/error/error_macros.h"
#if VERSION_MAJOR > 3
#include "core/io/file_access.h"
#else
#include "core/os/file_access.h"
#endif
#include <spine/Json.h>
#include <spine/Version.h>
#include <spine/Extension.h>
struct BinaryInput {
const unsigned char *cursor;
const unsigned char *end;
};
static unsigned char readByte(BinaryInput *input) {
return *input->cursor++;
}
static int readVarint(BinaryInput *input, bool optimizePositive) {
unsigned char b = readByte(input);
int value = b & 0x7F;
if (b & 0x80) {
b = readByte(input);
value |= (b & 0x7F) << 7;
if (b & 0x80) {
b = readByte(input);
value |= (b & 0x7F) << 14;
if (b & 0x80) {
b = readByte(input);
value |= (b & 0x7F) << 21;
if (b & 0x80) value |= (readByte(input) & 0x7F) << 28;
}
}
}
if (!optimizePositive) {
value = (((unsigned int) value >> 1) ^ -(value & 1));
}
return value;
}
static char *readString(BinaryInput *input) {
int length = readVarint(input, true);
char *string;
if (length == 0) {
return NULL;
}
string = spine::SpineExtension::alloc<char>(length, __FILE__, __LINE__);
memcpy(string, input->cursor, length - 1);
input->cursor += length - 1;
string[length - 1] = '\0';
return string;
}
void SpineSkeletonFileResource::_bind_methods() {
ADD_SIGNAL(MethodInfo("skeleton_file_changed"));
}
static bool checkVersion(const char *version) {
if (!version) return false;
char *result = (char *) (strstr(version, SPINE_VERSION_STRING) - version);
return result == 0;
}
static bool checkJson(const char *jsonData) {
spine::Json json(jsonData);
spine::Json *skeleton = spine::Json::getItem(&json, "skeleton");
if (!skeleton) return false;
const char *version = spine::Json::getString(skeleton, "spine", 0);
if (!version) return false;
return checkVersion(version);
}
static bool checkBinary(const char *binaryData, int length) {
BinaryInput input;
input.cursor = (const unsigned char *) binaryData;
input.end = (const unsigned char *) binaryData + length;
// Skip hash
input.cursor += 8;
char *version = readString(&input);
bool result = checkVersion(version);
spine::SpineExtension::free(version, __FILE__, __LINE__);
return result;
}
Error SpineSkeletonFileResource::load_from_file(const String &path) {
Error error = OK;
if (path.ends_with(".spjson") || path.ends_with(".spine-json")) {
json = FileAccess::get_file_as_string(path, &error);
if (error != OK) return error;
if (!checkJson(json.utf8())) return ERR_INVALID_DATA;
} else {
#if VERSION_MAJOR > 3
binary = FileAccess::get_file_as_bytes(path, &error);
#else
binary = FileAccess::get_file_as_array(path, &error);
#endif
if (error != OK) return error;
if (!checkBinary((const char *) binary.ptr(), binary.size())) return ERR_INVALID_DATA;
}
return error;
}
Error SpineSkeletonFileResource::save_to_file(const String &path) {
Error error;
#if VERSION_MAJOR > 3
Ref<FileAccess> file = FileAccess::open(path, FileAccess::WRITE, &error);
if (error != OK) return error;
#else
FileAccess *file = FileAccess::open(path, FileAccess::WRITE, &error);
if (error != OK) {
if (file) file->close();
return error;
}
#endif
if (!is_binary())
file->store_string(json);
else
file->store_buffer(binary.ptr(), binary.size());
#if VERSION_MAJOR > 3
file->flush();
#else
file->close();
#endif
return OK;
}
Error SpineSkeletonFileResource::copy_from(const Ref<Resource> &p_resource) {
auto error = Resource::copy_from(p_resource);
if (error != OK) return error;
const Ref<SpineSkeletonFileResource> &spineFile = static_cast<const Ref<SpineSkeletonFileResource> &>(p_resource);
this->json = spineFile->json;
this->binary = spineFile->binary;
emit_signal(SNAME("skeleton_file_changed"));
return OK;
}
#if VERSION_MAJOR > 3
RES SpineSkeletonFileResourceFormatLoader::load(const String &path, const String &original_path, Error *error, bool use_sub_threads, float *progress, CacheMode cache_mode) {
#else
RES SpineSkeletonFileResourceFormatLoader::load(const String &path, const String &original_path, Error *error) {
#endif
Ref<SpineSkeletonFileResource> skeleton_file = memnew(SpineSkeletonFileResource);
skeleton_file->load_from_file(path);
if (error) *error = OK;
return skeleton_file;
}
void SpineSkeletonFileResourceFormatLoader::get_recognized_extensions(List<String> *extensions) const {
extensions->push_back("spjson");
extensions->push_back("spskel");
}
String SpineSkeletonFileResourceFormatLoader::get_resource_type(const String &path) const {
return path.ends_with(".spjson") || path.ends_with(".spskel") || path.ends_with(".spine-json") || path.ends_with(".skel") ? "SpineSkeletonFileResource" : "";
}
bool SpineSkeletonFileResourceFormatLoader::handles_type(const String &type) const {
return type == "SpineSkeletonFileResource" || ClassDB::is_parent_class(type, "SpineSkeletonFileResource");
}
#if VERSION_MAJOR > 3
Error SpineSkeletonFileResourceFormatSaver::save(const RES &resource, const String &path, uint32_t flags) {
#else
Error SpineSkeletonFileResourceFormatSaver::save(const String &path, const RES &resource, uint32_t flags) {
#endif
Ref<SpineSkeletonFileResource> res = resource;
Error error = res->save_to_file(path);
return error;
}
void SpineSkeletonFileResourceFormatSaver::get_recognized_extensions(const RES &resource, List<String> *p_extensions) const {
if (Object::cast_to<SpineSkeletonFileResource>(*resource)) {
p_extensions->push_back("spjson");
p_extensions->push_back("spskel");
}
}
bool SpineSkeletonFileResourceFormatSaver::recognize(const RES &p_resource) const {
return Object::cast_to<SpineSkeletonFileResource>(*p_resource) != nullptr;
}