2016-06-18 12:46:12 +00:00
|
|
|
/**************************************************************************/
|
|
|
|
/* variant_parser.cpp */
|
|
|
|
/**************************************************************************/
|
|
|
|
/* This file is part of: */
|
|
|
|
/* GODOT ENGINE */
|
|
|
|
/* https://godotengine.org */
|
|
|
|
/**************************************************************************/
|
|
|
|
/* Copyright (c) 2014-present Godot Engine contributors (see AUTHORS.md). */
|
|
|
|
/* Copyright (c) 2007-2014 Juan Linietsky, Ariel Manzur. */
|
|
|
|
/* */
|
|
|
|
/* 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. */
|
|
|
|
/**************************************************************************/
|
2018-01-04 23:50:27 +00:00
|
|
|
|
2015-11-24 13:42:05 +00:00
|
|
|
#include "variant_parser.h"
|
2017-01-16 07:04:19 +00:00
|
|
|
|
2024-03-05 16:53:57 +00:00
|
|
|
#include "core/crypto/crypto_core.h"
|
2020-03-01 22:14:37 +00:00
|
|
|
#include "core/input/input_event.h"
|
2018-09-11 16:13:45 +00:00
|
|
|
#include "core/io/resource_loader.h"
|
2023-09-06 19:02:52 +00:00
|
|
|
#include "core/object/script_language.h"
|
2018-09-11 16:13:45 +00:00
|
|
|
#include "core/os/keyboard.h"
|
2020-11-07 22:33:38 +00:00
|
|
|
#include "core/string/string_buffer.h"
|
2015-11-24 13:42:05 +00:00
|
|
|
|
2022-11-24 15:54:16 +00:00
|
|
|
char32_t VariantParser::Stream::get_char() {
|
|
|
|
// is within buffer?
|
|
|
|
if (readahead_pointer < readahead_filled) {
|
|
|
|
return readahead_buffer[readahead_pointer++];
|
|
|
|
}
|
|
|
|
|
|
|
|
// attempt to readahead
|
2022-12-12 15:14:39 +00:00
|
|
|
readahead_filled = _read_buffer(readahead_buffer, readahead_enabled ? READAHEAD_SIZE : 1);
|
2022-11-24 15:54:16 +00:00
|
|
|
if (readahead_filled) {
|
|
|
|
readahead_pointer = 0;
|
|
|
|
} else {
|
|
|
|
// EOF
|
|
|
|
readahead_pointer = 1;
|
|
|
|
eof = true;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
return get_char();
|
2015-11-24 13:42:05 +00:00
|
|
|
}
|
|
|
|
|
2022-12-12 15:14:39 +00:00
|
|
|
bool VariantParser::Stream::is_eof() const {
|
|
|
|
if (readahead_enabled) {
|
|
|
|
return eof;
|
|
|
|
}
|
|
|
|
return _is_eof();
|
|
|
|
}
|
|
|
|
|
2015-11-24 13:42:05 +00:00
|
|
|
bool VariantParser::StreamFile::is_utf8() const {
|
|
|
|
return true;
|
|
|
|
}
|
2020-05-14 12:29:06 +00:00
|
|
|
|
2022-12-12 15:14:39 +00:00
|
|
|
bool VariantParser::StreamFile::_is_eof() const {
|
|
|
|
return f->eof_reached();
|
|
|
|
}
|
|
|
|
|
2022-11-24 15:54:16 +00:00
|
|
|
uint32_t VariantParser::StreamFile::_read_buffer(char32_t *p_buffer, uint32_t p_num_chars) {
|
|
|
|
// The buffer is assumed to include at least one character (for null terminator)
|
|
|
|
ERR_FAIL_COND_V(!p_num_chars, 0);
|
2015-11-24 13:42:05 +00:00
|
|
|
|
2022-11-24 15:54:16 +00:00
|
|
|
uint8_t *temp = (uint8_t *)alloca(p_num_chars);
|
|
|
|
uint64_t num_read = f->get_buffer(temp, p_num_chars);
|
|
|
|
ERR_FAIL_COND_V(num_read == UINT64_MAX, 0);
|
|
|
|
|
|
|
|
// translate to wchar
|
|
|
|
for (uint32_t n = 0; n < num_read; n++) {
|
|
|
|
p_buffer[n] = temp[n];
|
2020-02-10 08:18:58 +00:00
|
|
|
}
|
2022-11-24 15:54:16 +00:00
|
|
|
|
|
|
|
// could be less than p_num_chars, or zero
|
|
|
|
return num_read;
|
2015-12-31 03:54:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool VariantParser::StreamString::is_utf8() const {
|
|
|
|
return false;
|
|
|
|
}
|
2020-05-14 12:29:06 +00:00
|
|
|
|
2022-12-12 15:14:39 +00:00
|
|
|
bool VariantParser::StreamString::_is_eof() const {
|
|
|
|
return pos > s.length();
|
|
|
|
}
|
|
|
|
|
2022-11-24 15:54:16 +00:00
|
|
|
uint32_t VariantParser::StreamString::_read_buffer(char32_t *p_buffer, uint32_t p_num_chars) {
|
|
|
|
// The buffer is assumed to include at least one character (for null terminator)
|
|
|
|
ERR_FAIL_COND_V(!p_num_chars, 0);
|
|
|
|
|
|
|
|
int available = MAX(s.length() - pos, 0);
|
|
|
|
if (available >= (int)p_num_chars) {
|
|
|
|
const char32_t *src = s.ptr();
|
|
|
|
src += pos;
|
|
|
|
memcpy(p_buffer, src, p_num_chars * sizeof(char32_t));
|
|
|
|
pos += p_num_chars;
|
|
|
|
|
|
|
|
return p_num_chars;
|
|
|
|
}
|
|
|
|
|
|
|
|
// going to reach EOF
|
|
|
|
if (available) {
|
|
|
|
const char32_t *src = s.ptr();
|
|
|
|
src += pos;
|
|
|
|
memcpy(p_buffer, src, available * sizeof(char32_t));
|
|
|
|
pos += available;
|
|
|
|
}
|
|
|
|
|
|
|
|
// add a zero
|
|
|
|
p_buffer[available] = 0;
|
|
|
|
|
|
|
|
return available;
|
2015-12-31 03:54:00 +00:00
|
|
|
}
|
2015-11-24 13:42:05 +00:00
|
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
const char *VariantParser::tk_name[TK_MAX] = {
|
|
|
|
"'{'",
|
|
|
|
"'}'",
|
|
|
|
"'['",
|
|
|
|
"']'",
|
|
|
|
"'('",
|
|
|
|
"')'",
|
|
|
|
"identifier",
|
|
|
|
"string",
|
2020-02-20 21:58:05 +00:00
|
|
|
"string_name",
|
2015-11-24 13:42:05 +00:00
|
|
|
"number",
|
2015-12-31 17:30:50 +00:00
|
|
|
"color",
|
2015-11-24 13:42:05 +00:00
|
|
|
"':'",
|
|
|
|
"','",
|
2016-01-07 12:04:44 +00:00
|
|
|
"'.'",
|
2015-11-24 13:42:05 +00:00
|
|
|
"'='",
|
|
|
|
"EOF",
|
|
|
|
"ERROR"
|
|
|
|
};
|
|
|
|
|
2020-07-23 04:42:19 +00:00
|
|
|
static double stor_fix(const String &p_str) {
|
|
|
|
if (p_str == "inf") {
|
|
|
|
return INFINITY;
|
|
|
|
} else if (p_str == "inf_neg") {
|
|
|
|
return -INFINITY;
|
|
|
|
} else if (p_str == "nan") {
|
|
|
|
return NAN;
|
|
|
|
}
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2015-11-24 13:42:05 +00:00
|
|
|
Error VariantParser::get_token(Stream *p_stream, Token &r_token, int &line, String &r_err_str) {
|
2020-02-20 21:58:05 +00:00
|
|
|
bool string_name = false;
|
|
|
|
|
2015-11-24 13:42:05 +00:00
|
|
|
while (true) {
|
2020-07-27 10:43:20 +00:00
|
|
|
char32_t cchar;
|
2015-11-24 13:42:05 +00:00
|
|
|
if (p_stream->saved) {
|
|
|
|
cchar = p_stream->saved;
|
|
|
|
p_stream->saved = 0;
|
|
|
|
} else {
|
|
|
|
cchar = p_stream->get_char();
|
2015-11-28 23:56:14 +00:00
|
|
|
if (p_stream->is_eof()) {
|
|
|
|
r_token.type = TK_EOF;
|
|
|
|
return OK;
|
|
|
|
}
|
2015-11-24 13:42:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
switch (cchar) {
|
|
|
|
case '\n': {
|
|
|
|
line++;
|
|
|
|
break;
|
2020-05-19 13:46:49 +00:00
|
|
|
}
|
2015-11-24 13:42:05 +00:00
|
|
|
case 0: {
|
|
|
|
r_token.type = TK_EOF;
|
|
|
|
return OK;
|
|
|
|
} break;
|
|
|
|
case '{': {
|
|
|
|
r_token.type = TK_CURLY_BRACKET_OPEN;
|
|
|
|
return OK;
|
2020-05-19 13:46:49 +00:00
|
|
|
}
|
2015-11-24 13:42:05 +00:00
|
|
|
case '}': {
|
|
|
|
r_token.type = TK_CURLY_BRACKET_CLOSE;
|
|
|
|
return OK;
|
2020-05-19 13:46:49 +00:00
|
|
|
}
|
2015-11-24 13:42:05 +00:00
|
|
|
case '[': {
|
|
|
|
r_token.type = TK_BRACKET_OPEN;
|
|
|
|
return OK;
|
2020-05-19 13:46:49 +00:00
|
|
|
}
|
2015-11-24 13:42:05 +00:00
|
|
|
case ']': {
|
|
|
|
r_token.type = TK_BRACKET_CLOSE;
|
|
|
|
return OK;
|
2020-05-19 13:46:49 +00:00
|
|
|
}
|
2015-11-24 13:42:05 +00:00
|
|
|
case '(': {
|
|
|
|
r_token.type = TK_PARENTHESIS_OPEN;
|
|
|
|
return OK;
|
2020-05-19 13:46:49 +00:00
|
|
|
}
|
2015-11-24 13:42:05 +00:00
|
|
|
case ')': {
|
|
|
|
r_token.type = TK_PARENTHESIS_CLOSE;
|
|
|
|
return OK;
|
2020-05-19 13:46:49 +00:00
|
|
|
}
|
2015-11-24 13:42:05 +00:00
|
|
|
case ':': {
|
|
|
|
r_token.type = TK_COLON;
|
|
|
|
return OK;
|
2020-05-19 13:46:49 +00:00
|
|
|
}
|
2015-12-31 13:25:21 +00:00
|
|
|
case ';': {
|
|
|
|
while (true) {
|
2020-07-27 10:43:20 +00:00
|
|
|
char32_t ch = p_stream->get_char();
|
2015-12-31 13:25:21 +00:00
|
|
|
if (p_stream->is_eof()) {
|
|
|
|
r_token.type = TK_EOF;
|
|
|
|
return OK;
|
|
|
|
}
|
2020-05-14 14:41:43 +00:00
|
|
|
if (ch == '\n') {
|
2021-05-23 07:37:53 +00:00
|
|
|
line++;
|
2015-12-31 13:25:21 +00:00
|
|
|
break;
|
2020-05-14 14:41:43 +00:00
|
|
|
}
|
2015-12-31 13:25:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
2020-05-19 13:46:49 +00:00
|
|
|
}
|
2015-11-24 13:42:05 +00:00
|
|
|
case ',': {
|
|
|
|
r_token.type = TK_COMMA;
|
|
|
|
return OK;
|
2020-05-19 13:46:49 +00:00
|
|
|
}
|
2016-01-07 12:04:44 +00:00
|
|
|
case '.': {
|
|
|
|
r_token.type = TK_PERIOD;
|
|
|
|
return OK;
|
2020-05-19 13:46:49 +00:00
|
|
|
}
|
2015-11-24 13:42:05 +00:00
|
|
|
case '=': {
|
|
|
|
r_token.type = TK_EQUAL;
|
|
|
|
return OK;
|
2020-05-19 13:46:49 +00:00
|
|
|
}
|
2015-12-31 17:30:50 +00:00
|
|
|
case '#': {
|
2018-01-20 19:03:17 +00:00
|
|
|
StringBuffer<> color_str;
|
2017-09-05 04:04:33 +00:00
|
|
|
color_str += '#';
|
2015-12-31 17:30:50 +00:00
|
|
|
while (true) {
|
2020-07-27 10:43:20 +00:00
|
|
|
char32_t ch = p_stream->get_char();
|
2015-12-31 17:30:50 +00:00
|
|
|
if (p_stream->is_eof()) {
|
|
|
|
r_token.type = TK_EOF;
|
|
|
|
return OK;
|
2022-02-04 08:32:20 +00:00
|
|
|
} else if (is_hex_digit(ch)) {
|
2017-09-05 04:04:33 +00:00
|
|
|
color_str += ch;
|
2015-12-31 17:30:50 +00:00
|
|
|
|
|
|
|
} else {
|
|
|
|
p_stream->saved = ch;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-09-05 04:04:33 +00:00
|
|
|
r_token.value = Color::html(color_str.as_string());
|
2015-12-31 17:30:50 +00:00
|
|
|
r_token.type = TK_COLOR;
|
|
|
|
return OK;
|
2020-05-19 13:46:49 +00:00
|
|
|
}
|
2021-06-12 19:52:15 +00:00
|
|
|
#ifndef DISABLE_DEPRECATED
|
|
|
|
case '@': // Compatibility with 3.x StringNames.
|
|
|
|
#endif
|
|
|
|
case '&': { // StringName.
|
2020-02-20 21:58:05 +00:00
|
|
|
cchar = p_stream->get_char();
|
|
|
|
if (cchar != '"') {
|
2021-06-12 19:52:15 +00:00
|
|
|
r_err_str = "Expected '\"' after '&'";
|
2020-02-20 21:58:05 +00:00
|
|
|
r_token.type = TK_ERROR;
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
string_name = true;
|
2020-02-22 19:47:50 +00:00
|
|
|
[[fallthrough]];
|
2020-02-20 21:58:05 +00:00
|
|
|
}
|
2015-11-24 13:42:05 +00:00
|
|
|
case '"': {
|
|
|
|
String str;
|
2022-01-30 13:44:07 +00:00
|
|
|
char32_t prev = 0;
|
2015-11-24 13:42:05 +00:00
|
|
|
while (true) {
|
2020-07-27 10:43:20 +00:00
|
|
|
char32_t ch = p_stream->get_char();
|
2015-11-24 13:42:05 +00:00
|
|
|
|
|
|
|
if (ch == 0) {
|
|
|
|
r_err_str = "Unterminated String";
|
|
|
|
r_token.type = TK_ERROR;
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
} else if (ch == '"') {
|
|
|
|
break;
|
|
|
|
} else if (ch == '\\') {
|
|
|
|
//escaped characters...
|
2020-07-27 10:43:20 +00:00
|
|
|
char32_t next = p_stream->get_char();
|
2015-11-24 13:42:05 +00:00
|
|
|
if (next == 0) {
|
|
|
|
r_err_str = "Unterminated String";
|
|
|
|
r_token.type = TK_ERROR;
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
2020-07-27 10:43:20 +00:00
|
|
|
char32_t res = 0;
|
2015-11-24 13:42:05 +00:00
|
|
|
|
|
|
|
switch (next) {
|
2020-05-10 11:00:47 +00:00
|
|
|
case 'b':
|
|
|
|
res = 8;
|
|
|
|
break;
|
|
|
|
case 't':
|
|
|
|
res = 9;
|
|
|
|
break;
|
|
|
|
case 'n':
|
|
|
|
res = 10;
|
|
|
|
break;
|
|
|
|
case 'f':
|
|
|
|
res = 12;
|
|
|
|
break;
|
|
|
|
case 'r':
|
|
|
|
res = 13;
|
|
|
|
break;
|
2022-01-30 13:44:07 +00:00
|
|
|
case 'U':
|
2015-11-24 13:42:05 +00:00
|
|
|
case 'u': {
|
2022-01-30 13:44:07 +00:00
|
|
|
// Hexadecimal sequence.
|
|
|
|
int hex_len = (next == 'U') ? 6 : 4;
|
|
|
|
for (int j = 0; j < hex_len; j++) {
|
2020-07-27 10:43:20 +00:00
|
|
|
char32_t c = p_stream->get_char();
|
2022-01-30 13:44:07 +00:00
|
|
|
|
2015-11-24 13:42:05 +00:00
|
|
|
if (c == 0) {
|
|
|
|
r_err_str = "Unterminated String";
|
|
|
|
r_token.type = TK_ERROR;
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
2022-02-04 08:32:20 +00:00
|
|
|
if (!is_hex_digit(c)) {
|
2015-11-24 13:42:05 +00:00
|
|
|
r_err_str = "Malformed hex constant in string";
|
|
|
|
r_token.type = TK_ERROR;
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
2020-07-27 10:43:20 +00:00
|
|
|
char32_t v;
|
2022-02-04 08:32:20 +00:00
|
|
|
if (is_digit(c)) {
|
2015-11-24 13:42:05 +00:00
|
|
|
v = c - '0';
|
|
|
|
} else if (c >= 'a' && c <= 'f') {
|
|
|
|
v = c - 'a';
|
|
|
|
v += 10;
|
|
|
|
} else if (c >= 'A' && c <= 'F') {
|
|
|
|
v = c - 'A';
|
|
|
|
v += 10;
|
|
|
|
} else {
|
2020-02-13 10:37:37 +00:00
|
|
|
ERR_PRINT("Bug parsing hex constant.");
|
2015-11-24 13:42:05 +00:00
|
|
|
v = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
res <<= 4;
|
|
|
|
res |= v;
|
|
|
|
}
|
|
|
|
|
|
|
|
} break;
|
|
|
|
default: {
|
|
|
|
res = next;
|
|
|
|
} break;
|
|
|
|
}
|
|
|
|
|
2022-01-30 13:44:07 +00:00
|
|
|
// Parse UTF-16 pair.
|
|
|
|
if ((res & 0xfffffc00) == 0xd800) {
|
|
|
|
if (prev == 0) {
|
|
|
|
prev = res;
|
|
|
|
continue;
|
|
|
|
} else {
|
|
|
|
r_err_str = "Invalid UTF-16 sequence in string, unpaired lead surrogate";
|
|
|
|
r_token.type = TK_ERROR;
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
|
|
|
} else if ((res & 0xfffffc00) == 0xdc00) {
|
|
|
|
if (prev == 0) {
|
|
|
|
r_err_str = "Invalid UTF-16 sequence in string, unpaired trail surrogate";
|
|
|
|
r_token.type = TK_ERROR;
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
} else {
|
|
|
|
res = (prev << 10UL) + res - ((0xd800 << 10UL) + 0xdc00 - 0x10000);
|
|
|
|
prev = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (prev != 0) {
|
|
|
|
r_err_str = "Invalid UTF-16 sequence in string, unpaired lead surrogate";
|
|
|
|
r_token.type = TK_ERROR;
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
2015-11-24 13:42:05 +00:00
|
|
|
str += res;
|
|
|
|
} else {
|
2022-01-30 13:44:07 +00:00
|
|
|
if (prev != 0) {
|
|
|
|
r_err_str = "Invalid UTF-16 sequence in string, unpaired lead surrogate";
|
|
|
|
r_token.type = TK_ERROR;
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
2020-05-14 14:41:43 +00:00
|
|
|
if (ch == '\n') {
|
2015-11-24 13:42:05 +00:00
|
|
|
line++;
|
2020-05-14 14:41:43 +00:00
|
|
|
}
|
2015-11-24 13:42:05 +00:00
|
|
|
str += ch;
|
|
|
|
}
|
|
|
|
}
|
2022-01-30 13:44:07 +00:00
|
|
|
if (prev != 0) {
|
|
|
|
r_err_str = "Invalid UTF-16 sequence in string, unpaired lead surrogate";
|
|
|
|
r_token.type = TK_ERROR;
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
2015-11-24 13:42:05 +00:00
|
|
|
|
2015-11-28 23:56:14 +00:00
|
|
|
if (p_stream->is_utf8()) {
|
|
|
|
str.parse_utf8(str.ascii(true).get_data());
|
|
|
|
}
|
2020-02-20 21:58:05 +00:00
|
|
|
if (string_name) {
|
|
|
|
r_token.type = TK_STRING_NAME;
|
|
|
|
r_token.value = StringName(str);
|
|
|
|
} else {
|
|
|
|
r_token.type = TK_STRING;
|
|
|
|
r_token.value = str;
|
|
|
|
}
|
2015-11-24 13:42:05 +00:00
|
|
|
return OK;
|
|
|
|
|
|
|
|
} break;
|
|
|
|
default: {
|
|
|
|
if (cchar <= 32) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (cchar == '-' || (cchar >= '0' && cchar <= '9')) {
|
|
|
|
//a number
|
2015-11-28 23:56:14 +00:00
|
|
|
|
2018-01-20 19:03:17 +00:00
|
|
|
StringBuffer<> num;
|
2015-11-24 13:42:05 +00:00
|
|
|
#define READING_SIGN 0
|
|
|
|
#define READING_INT 1
|
|
|
|
#define READING_DEC 2
|
|
|
|
#define READING_EXP 3
|
|
|
|
#define READING_DONE 4
|
|
|
|
int reading = READING_INT;
|
|
|
|
|
|
|
|
if (cchar == '-') {
|
2017-03-05 15:44:50 +00:00
|
|
|
num += '-';
|
2015-11-24 13:42:05 +00:00
|
|
|
cchar = p_stream->get_char();
|
|
|
|
}
|
|
|
|
|
2020-07-27 10:43:20 +00:00
|
|
|
char32_t c = cchar;
|
2015-11-24 13:42:05 +00:00
|
|
|
bool exp_sign = false;
|
|
|
|
bool exp_beg = false;
|
|
|
|
bool is_float = false;
|
|
|
|
|
|
|
|
while (true) {
|
|
|
|
switch (reading) {
|
|
|
|
case READING_INT: {
|
2022-02-04 08:32:20 +00:00
|
|
|
if (is_digit(c)) {
|
2015-11-24 13:42:05 +00:00
|
|
|
//pass
|
|
|
|
} else if (c == '.') {
|
|
|
|
reading = READING_DEC;
|
|
|
|
is_float = true;
|
|
|
|
} else if (c == 'e') {
|
|
|
|
reading = READING_EXP;
|
2017-08-21 21:51:46 +00:00
|
|
|
is_float = true;
|
2015-11-24 13:42:05 +00:00
|
|
|
} else {
|
|
|
|
reading = READING_DONE;
|
|
|
|
}
|
|
|
|
|
|
|
|
} break;
|
|
|
|
case READING_DEC: {
|
2022-02-04 08:32:20 +00:00
|
|
|
if (is_digit(c)) {
|
2015-11-24 13:42:05 +00:00
|
|
|
} else if (c == 'e') {
|
|
|
|
reading = READING_EXP;
|
|
|
|
} else {
|
|
|
|
reading = READING_DONE;
|
|
|
|
}
|
|
|
|
|
|
|
|
} break;
|
|
|
|
case READING_EXP: {
|
2022-02-04 08:32:20 +00:00
|
|
|
if (is_digit(c)) {
|
2015-11-24 13:42:05 +00:00
|
|
|
exp_beg = true;
|
2015-11-28 23:56:14 +00:00
|
|
|
|
2015-11-24 13:42:05 +00:00
|
|
|
} else if ((c == '-' || c == '+') && !exp_sign && !exp_beg) {
|
|
|
|
exp_sign = true;
|
2015-11-28 23:56:14 +00:00
|
|
|
|
2015-11-24 13:42:05 +00:00
|
|
|
} else {
|
|
|
|
reading = READING_DONE;
|
|
|
|
}
|
|
|
|
} break;
|
|
|
|
}
|
|
|
|
|
2020-05-14 14:41:43 +00:00
|
|
|
if (reading == READING_DONE) {
|
2015-11-24 13:42:05 +00:00
|
|
|
break;
|
2020-05-14 14:41:43 +00:00
|
|
|
}
|
2017-09-05 04:04:33 +00:00
|
|
|
num += c;
|
2015-11-24 13:42:05 +00:00
|
|
|
c = p_stream->get_char();
|
|
|
|
}
|
|
|
|
|
|
|
|
p_stream->saved = c;
|
|
|
|
|
|
|
|
r_token.type = TK_NUMBER;
|
2016-02-01 22:17:05 +00:00
|
|
|
|
2020-05-14 14:41:43 +00:00
|
|
|
if (is_float) {
|
2017-09-05 04:04:33 +00:00
|
|
|
r_token.value = num.as_double();
|
2020-05-14 14:41:43 +00:00
|
|
|
} else {
|
2017-09-05 04:04:33 +00:00
|
|
|
r_token.value = num.as_int();
|
2020-05-14 14:41:43 +00:00
|
|
|
}
|
2015-11-24 13:42:05 +00:00
|
|
|
return OK;
|
2024-04-20 09:36:41 +00:00
|
|
|
} else if (is_ascii_alphabet_char(cchar) || is_underscore(cchar)) {
|
2018-01-20 19:03:17 +00:00
|
|
|
StringBuffer<> id;
|
2015-11-28 23:56:14 +00:00
|
|
|
bool first = true;
|
2015-11-24 13:42:05 +00:00
|
|
|
|
2024-04-20 09:36:41 +00:00
|
|
|
while (is_ascii_alphabet_char(cchar) || is_underscore(cchar) || (!first && is_digit(cchar))) {
|
2017-09-05 04:04:33 +00:00
|
|
|
id += cchar;
|
2015-11-24 13:42:05 +00:00
|
|
|
cchar = p_stream->get_char();
|
2015-11-28 23:56:14 +00:00
|
|
|
first = false;
|
2015-11-24 13:42:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
p_stream->saved = cchar;
|
|
|
|
|
|
|
|
r_token.type = TK_IDENTIFIER;
|
2017-09-05 04:04:33 +00:00
|
|
|
r_token.value = id.as_string();
|
2015-11-24 13:42:05 +00:00
|
|
|
return OK;
|
|
|
|
} else {
|
|
|
|
r_err_str = "Unexpected character.";
|
|
|
|
r_token.type = TK_ERROR;
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
r_token.type = TK_ERROR;
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
|
|
|
|
2015-12-31 03:31:00 +00:00
|
|
|
Error VariantParser::_parse_enginecfg(Stream *p_stream, Vector<String> &strings, int &line, String &r_err_str) {
|
|
|
|
Token token;
|
|
|
|
get_token(p_stream, token, line, r_err_str);
|
|
|
|
if (token.type != TK_PARENTHESIS_OPEN) {
|
2017-05-01 15:44:52 +00:00
|
|
|
r_err_str = "Expected '(' in old-style project.godot construct";
|
2015-12-31 03:31:00 +00:00
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
String accum;
|
|
|
|
|
|
|
|
while (true) {
|
2020-07-27 10:43:20 +00:00
|
|
|
char32_t c = p_stream->get_char();
|
2015-12-31 03:31:00 +00:00
|
|
|
|
|
|
|
if (p_stream->is_eof()) {
|
2017-05-01 15:44:52 +00:00
|
|
|
r_err_str = "Unexpected EOF while parsing old-style project.godot construct";
|
2015-12-31 03:31:00 +00:00
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (c == ',') {
|
|
|
|
strings.push_back(accum.strip_edges());
|
|
|
|
accum = String();
|
|
|
|
} else if (c == ')') {
|
|
|
|
strings.push_back(accum.strip_edges());
|
|
|
|
return OK;
|
|
|
|
} else if (c == '\n') {
|
|
|
|
line++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-11-28 23:56:14 +00:00
|
|
|
template <typename T>
|
|
|
|
Error VariantParser::_parse_construct(Stream *p_stream, Vector<T> &r_construct, int &line, String &r_err_str) {
|
2015-11-24 13:42:05 +00:00
|
|
|
Token token;
|
|
|
|
get_token(p_stream, token, line, r_err_str);
|
|
|
|
if (token.type != TK_PARENTHESIS_OPEN) {
|
2015-11-28 23:56:14 +00:00
|
|
|
r_err_str = "Expected '(' in constructor";
|
2015-11-24 13:42:05 +00:00
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool first = true;
|
|
|
|
while (true) {
|
|
|
|
if (!first) {
|
|
|
|
get_token(p_stream, token, line, r_err_str);
|
|
|
|
if (token.type == TK_COMMA) {
|
2016-03-08 23:00:52 +00:00
|
|
|
//do none
|
2015-11-28 23:56:14 +00:00
|
|
|
} else if (token.type == TK_PARENTHESIS_CLOSE) {
|
2015-11-24 13:42:05 +00:00
|
|
|
break;
|
|
|
|
} else {
|
2015-11-28 23:56:14 +00:00
|
|
|
r_err_str = "Expected ',' or ')' in constructor";
|
2015-11-24 13:42:05 +00:00
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
get_token(p_stream, token, line, r_err_str);
|
2016-01-11 02:22:48 +00:00
|
|
|
|
|
|
|
if (first && token.type == TK_PARENTHESIS_CLOSE) {
|
|
|
|
break;
|
|
|
|
} else if (token.type != TK_NUMBER) {
|
2020-07-23 04:42:19 +00:00
|
|
|
bool valid = false;
|
|
|
|
if (token.type == TK_IDENTIFIER) {
|
|
|
|
double real = stor_fix(token.value);
|
|
|
|
if (real != -1) {
|
|
|
|
token.type = TK_NUMBER;
|
|
|
|
token.value = real;
|
|
|
|
valid = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!valid) {
|
|
|
|
r_err_str = "Expected float in constructor";
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
2015-11-24 13:42:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
r_construct.push_back(token.value);
|
2015-11-28 23:56:14 +00:00
|
|
|
first = false;
|
2015-11-24 13:42:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return OK;
|
|
|
|
}
|
|
|
|
|
2024-03-05 16:53:57 +00:00
|
|
|
Error VariantParser::_parse_byte_array(Stream *p_stream, Vector<uint8_t> &r_construct, int &line, String &r_err_str) {
|
|
|
|
Token token;
|
|
|
|
get_token(p_stream, token, line, r_err_str);
|
|
|
|
if (token.type != TK_PARENTHESIS_OPEN) {
|
|
|
|
r_err_str = "Expected '(' in constructor";
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
get_token(p_stream, token, line, r_err_str);
|
|
|
|
if (token.type == TK_STRING) {
|
|
|
|
// Base64 encoded array.
|
|
|
|
String base64_encoded_string = token.value;
|
|
|
|
int strlen = base64_encoded_string.length();
|
|
|
|
CharString cstr = base64_encoded_string.ascii();
|
|
|
|
|
|
|
|
size_t arr_len = 0;
|
|
|
|
r_construct.resize(strlen / 4 * 3 + 1);
|
|
|
|
uint8_t *w = r_construct.ptrw();
|
|
|
|
Error err = CryptoCore::b64_decode(&w[0], r_construct.size(), &arr_len, (unsigned char *)cstr.get_data(), strlen);
|
|
|
|
if (err) {
|
|
|
|
r_err_str = "Invalid base64-encoded string";
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
|
|
|
r_construct.resize(arr_len);
|
|
|
|
|
|
|
|
get_token(p_stream, token, line, r_err_str);
|
|
|
|
if (token.type != TK_PARENTHESIS_CLOSE) {
|
|
|
|
r_err_str = "Expected ')' in constructor";
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
} else if (token.type == TK_NUMBER || token.type == TK_IDENTIFIER) {
|
|
|
|
// Individual elements.
|
|
|
|
while (true) {
|
|
|
|
if (token.type != TK_NUMBER) {
|
|
|
|
bool valid = false;
|
|
|
|
if (token.type == TK_IDENTIFIER) {
|
|
|
|
double real = stor_fix(token.value);
|
|
|
|
if (real != -1) {
|
|
|
|
token.type = TK_NUMBER;
|
|
|
|
token.value = real;
|
|
|
|
valid = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!valid) {
|
|
|
|
r_err_str = "Expected number in constructor";
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
r_construct.push_back(token.value);
|
|
|
|
|
|
|
|
get_token(p_stream, token, line, r_err_str);
|
|
|
|
|
|
|
|
if (token.type == TK_COMMA) {
|
|
|
|
//do none
|
|
|
|
} else if (token.type == TK_PARENTHESIS_CLOSE) {
|
|
|
|
break;
|
|
|
|
} else {
|
|
|
|
r_err_str = "Expected ',' or ')' in constructor";
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
get_token(p_stream, token, line, r_err_str);
|
|
|
|
}
|
|
|
|
} else if (token.type == TK_PARENTHESIS_CLOSE) {
|
|
|
|
// Empty array.
|
|
|
|
return OK;
|
|
|
|
} else {
|
|
|
|
r_err_str = "Expected base64 string, or list of numbers in constructor";
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
return OK;
|
|
|
|
}
|
|
|
|
|
2015-11-24 13:42:05 +00:00
|
|
|
Error VariantParser::parse_value(Token &token, Variant &value, Stream *p_stream, int &line, String &r_err_str, ResourceParser *p_res_parser) {
|
|
|
|
if (token.type == TK_CURLY_BRACKET_OPEN) {
|
|
|
|
Dictionary d;
|
|
|
|
Error err = _parse_dictionary(d, p_stream, line, r_err_str, p_res_parser);
|
2020-05-14 14:41:43 +00:00
|
|
|
if (err) {
|
2015-11-24 13:42:05 +00:00
|
|
|
return err;
|
2020-05-14 14:41:43 +00:00
|
|
|
}
|
2015-11-24 13:42:05 +00:00
|
|
|
value = d;
|
|
|
|
return OK;
|
|
|
|
} else if (token.type == TK_BRACKET_OPEN) {
|
|
|
|
Array a;
|
|
|
|
Error err = _parse_array(a, p_stream, line, r_err_str, p_res_parser);
|
2020-05-14 14:41:43 +00:00
|
|
|
if (err) {
|
2015-11-24 13:42:05 +00:00
|
|
|
return err;
|
2020-05-14 14:41:43 +00:00
|
|
|
}
|
2015-11-24 13:42:05 +00:00
|
|
|
value = a;
|
|
|
|
return OK;
|
|
|
|
} else if (token.type == TK_IDENTIFIER) {
|
|
|
|
String id = token.value;
|
2020-05-14 14:41:43 +00:00
|
|
|
if (id == "true") {
|
2015-11-24 13:42:05 +00:00
|
|
|
value = true;
|
2020-05-14 14:41:43 +00:00
|
|
|
} else if (id == "false") {
|
2015-11-24 13:42:05 +00:00
|
|
|
value = false;
|
2020-05-14 14:41:43 +00:00
|
|
|
} else if (id == "null" || id == "nil") {
|
2015-11-24 13:42:05 +00:00
|
|
|
value = Variant();
|
2020-05-14 14:41:43 +00:00
|
|
|
} else if (id == "inf") {
|
2021-07-21 08:40:31 +00:00
|
|
|
value = INFINITY;
|
2020-07-23 04:42:19 +00:00
|
|
|
} else if (id == "inf_neg") {
|
|
|
|
value = -INFINITY;
|
2020-05-14 14:41:43 +00:00
|
|
|
} else if (id == "nan") {
|
2021-07-21 08:40:31 +00:00
|
|
|
value = NAN;
|
2020-05-14 14:41:43 +00:00
|
|
|
} else if (id == "Vector2") {
|
2021-01-29 19:27:20 +00:00
|
|
|
Vector<real_t> args;
|
|
|
|
Error err = _parse_construct<real_t>(p_stream, args, line, r_err_str);
|
2020-05-14 14:41:43 +00:00
|
|
|
if (err) {
|
2015-11-24 13:42:05 +00:00
|
|
|
return err;
|
2020-05-14 14:41:43 +00:00
|
|
|
}
|
2015-11-24 13:42:05 +00:00
|
|
|
|
|
|
|
if (args.size() != 2) {
|
|
|
|
r_err_str = "Expected 2 arguments for constructor";
|
2020-06-08 09:36:41 +00:00
|
|
|
return ERR_PARSE_ERROR;
|
2015-11-24 13:42:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
value = Vector2(args[0], args[1]);
|
2020-02-22 03:26:41 +00:00
|
|
|
} else if (id == "Vector2i") {
|
|
|
|
Vector<int32_t> args;
|
|
|
|
Error err = _parse_construct<int32_t>(p_stream, args, line, r_err_str);
|
2020-05-14 14:41:43 +00:00
|
|
|
if (err) {
|
2020-02-22 03:26:41 +00:00
|
|
|
return err;
|
2020-05-14 14:41:43 +00:00
|
|
|
}
|
2020-02-22 03:26:41 +00:00
|
|
|
|
|
|
|
if (args.size() != 2) {
|
|
|
|
r_err_str = "Expected 2 arguments for constructor";
|
2020-06-08 09:36:41 +00:00
|
|
|
return ERR_PARSE_ERROR;
|
2020-02-22 03:26:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
value = Vector2i(args[0], args[1]);
|
2015-12-12 20:10:43 +00:00
|
|
|
} else if (id == "Rect2") {
|
2021-01-29 19:27:20 +00:00
|
|
|
Vector<real_t> args;
|
|
|
|
Error err = _parse_construct<real_t>(p_stream, args, line, r_err_str);
|
2020-05-14 14:41:43 +00:00
|
|
|
if (err) {
|
2015-12-12 20:10:43 +00:00
|
|
|
return err;
|
2020-05-14 14:41:43 +00:00
|
|
|
}
|
2015-12-12 20:10:43 +00:00
|
|
|
|
|
|
|
if (args.size() != 4) {
|
2015-12-12 22:21:43 +00:00
|
|
|
r_err_str = "Expected 4 arguments for constructor";
|
2020-06-08 09:36:41 +00:00
|
|
|
return ERR_PARSE_ERROR;
|
2015-12-12 20:10:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
value = Rect2(args[0], args[1], args[2], args[3]);
|
2020-02-22 03:26:41 +00:00
|
|
|
} else if (id == "Rect2i") {
|
|
|
|
Vector<int32_t> args;
|
|
|
|
Error err = _parse_construct<int32_t>(p_stream, args, line, r_err_str);
|
2020-05-14 14:41:43 +00:00
|
|
|
if (err) {
|
2020-02-22 03:26:41 +00:00
|
|
|
return err;
|
2020-05-14 14:41:43 +00:00
|
|
|
}
|
2020-02-22 03:26:41 +00:00
|
|
|
|
|
|
|
if (args.size() != 4) {
|
|
|
|
r_err_str = "Expected 4 arguments for constructor";
|
2020-06-08 09:36:41 +00:00
|
|
|
return ERR_PARSE_ERROR;
|
2020-02-22 03:26:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
value = Rect2i(args[0], args[1], args[2], args[3]);
|
2015-11-24 13:42:05 +00:00
|
|
|
} else if (id == "Vector3") {
|
2021-01-29 19:27:20 +00:00
|
|
|
Vector<real_t> args;
|
|
|
|
Error err = _parse_construct<real_t>(p_stream, args, line, r_err_str);
|
2020-05-14 14:41:43 +00:00
|
|
|
if (err) {
|
2015-11-24 13:42:05 +00:00
|
|
|
return err;
|
2020-05-14 14:41:43 +00:00
|
|
|
}
|
2015-11-24 13:42:05 +00:00
|
|
|
|
|
|
|
if (args.size() != 3) {
|
|
|
|
r_err_str = "Expected 3 arguments for constructor";
|
2020-06-08 09:36:41 +00:00
|
|
|
return ERR_PARSE_ERROR;
|
2015-11-24 13:42:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
value = Vector3(args[0], args[1], args[2]);
|
2020-02-22 03:26:41 +00:00
|
|
|
} else if (id == "Vector3i") {
|
|
|
|
Vector<int32_t> args;
|
|
|
|
Error err = _parse_construct<int32_t>(p_stream, args, line, r_err_str);
|
2020-05-14 14:41:43 +00:00
|
|
|
if (err) {
|
2020-02-22 03:26:41 +00:00
|
|
|
return err;
|
2020-05-14 14:41:43 +00:00
|
|
|
}
|
2020-02-22 03:26:41 +00:00
|
|
|
|
|
|
|
if (args.size() != 3) {
|
|
|
|
r_err_str = "Expected 3 arguments for constructor";
|
2020-06-08 09:36:41 +00:00
|
|
|
return ERR_PARSE_ERROR;
|
2020-02-22 03:26:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
value = Vector3i(args[0], args[1], args[2]);
|
Implement Vector4, Vector4i, Projection
Implement built-in classes Vector4, Vector4i and Projection.
* Two versions of Vector4 (float and integer).
* A Projection class, which is a 4x4 matrix specialized in projection types.
These types have been requested for a long time, but given they were very corner case they were not added before.
Because in Godot 4, reimplementing parts of the rendering engine is now possible, access to these types (heavily used by the rendering code) becomes a necessity.
**Q**: Why Projection and not Matrix4?
**A**: Godot does not use Matrix2, Matrix3, Matrix4x3, etc. naming convention because, within the engine, these types always have a *purpose*. As such, Godot names them: Transform2D, Transform3D or Basis. In this case, this 4x4 matrix is _always_ used as a _Projection_, hence the naming.
2022-07-19 23:11:13 +00:00
|
|
|
} else if (id == "Vector4") {
|
|
|
|
Vector<real_t> args;
|
|
|
|
Error err = _parse_construct<real_t>(p_stream, args, line, r_err_str);
|
|
|
|
if (err) {
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (args.size() != 4) {
|
|
|
|
r_err_str = "Expected 4 arguments for constructor";
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
value = Vector4(args[0], args[1], args[2], args[3]);
|
|
|
|
} else if (id == "Vector4i") {
|
|
|
|
Vector<int32_t> args;
|
|
|
|
Error err = _parse_construct<int32_t>(p_stream, args, line, r_err_str);
|
|
|
|
if (err) {
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (args.size() != 4) {
|
|
|
|
r_err_str = "Expected 4 arguments for constructor";
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
value = Vector4i(args[0], args[1], args[2], args[3]);
|
2017-01-11 03:52:51 +00:00
|
|
|
} else if (id == "Transform2D" || id == "Matrix32") { //compatibility
|
2021-01-29 19:27:20 +00:00
|
|
|
Vector<real_t> args;
|
|
|
|
Error err = _parse_construct<real_t>(p_stream, args, line, r_err_str);
|
2020-05-14 14:41:43 +00:00
|
|
|
if (err) {
|
2015-11-24 13:42:05 +00:00
|
|
|
return err;
|
2020-05-14 14:41:43 +00:00
|
|
|
}
|
2015-11-24 13:42:05 +00:00
|
|
|
|
|
|
|
if (args.size() != 6) {
|
|
|
|
r_err_str = "Expected 6 arguments for constructor";
|
2020-06-08 09:36:41 +00:00
|
|
|
return ERR_PARSE_ERROR;
|
2015-11-24 13:42:05 +00:00
|
|
|
}
|
2020-06-08 09:36:41 +00:00
|
|
|
|
2017-01-11 03:52:51 +00:00
|
|
|
Transform2D m;
|
2015-11-24 13:42:05 +00:00
|
|
|
m[0] = Vector2(args[0], args[1]);
|
|
|
|
m[1] = Vector2(args[2], args[3]);
|
|
|
|
m[2] = Vector2(args[4], args[5]);
|
|
|
|
value = m;
|
|
|
|
} else if (id == "Plane") {
|
2021-01-29 19:27:20 +00:00
|
|
|
Vector<real_t> args;
|
|
|
|
Error err = _parse_construct<real_t>(p_stream, args, line, r_err_str);
|
2020-05-14 14:41:43 +00:00
|
|
|
if (err) {
|
2015-11-24 13:42:05 +00:00
|
|
|
return err;
|
2020-05-14 14:41:43 +00:00
|
|
|
}
|
2015-11-24 13:42:05 +00:00
|
|
|
|
|
|
|
if (args.size() != 4) {
|
|
|
|
r_err_str = "Expected 4 arguments for constructor";
|
2020-06-08 09:36:41 +00:00
|
|
|
return ERR_PARSE_ERROR;
|
2015-11-24 13:42:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
value = Plane(args[0], args[1], args[2], args[3]);
|
2021-01-20 07:02:02 +00:00
|
|
|
} else if (id == "Quaternion" || id == "Quat") { // "Quat" kept for compatibility
|
2021-01-29 19:27:20 +00:00
|
|
|
Vector<real_t> args;
|
|
|
|
Error err = _parse_construct<real_t>(p_stream, args, line, r_err_str);
|
2020-05-14 14:41:43 +00:00
|
|
|
if (err) {
|
2015-11-24 13:42:05 +00:00
|
|
|
return err;
|
2020-05-14 14:41:43 +00:00
|
|
|
}
|
2015-11-24 13:42:05 +00:00
|
|
|
|
|
|
|
if (args.size() != 4) {
|
|
|
|
r_err_str = "Expected 4 arguments for constructor";
|
2020-06-08 09:36:41 +00:00
|
|
|
return ERR_PARSE_ERROR;
|
2015-11-24 13:42:05 +00:00
|
|
|
}
|
|
|
|
|
2021-01-20 07:02:02 +00:00
|
|
|
value = Quaternion(args[0], args[1], args[2], args[3]);
|
2017-11-20 11:25:43 +00:00
|
|
|
} else if (id == "AABB" || id == "Rect3") {
|
2021-01-29 19:27:20 +00:00
|
|
|
Vector<real_t> args;
|
|
|
|
Error err = _parse_construct<real_t>(p_stream, args, line, r_err_str);
|
2020-05-14 14:41:43 +00:00
|
|
|
if (err) {
|
2015-11-24 13:42:05 +00:00
|
|
|
return err;
|
2020-05-14 14:41:43 +00:00
|
|
|
}
|
2015-11-24 13:42:05 +00:00
|
|
|
|
|
|
|
if (args.size() != 6) {
|
|
|
|
r_err_str = "Expected 6 arguments for constructor";
|
2020-06-08 09:36:41 +00:00
|
|
|
return ERR_PARSE_ERROR;
|
2015-11-24 13:42:05 +00:00
|
|
|
}
|
|
|
|
|
2017-11-17 02:09:00 +00:00
|
|
|
value = AABB(Vector3(args[0], args[1], args[2]), Vector3(args[3], args[4], args[5]));
|
2017-01-11 03:52:51 +00:00
|
|
|
} else if (id == "Basis" || id == "Matrix3") { //compatibility
|
2021-01-29 19:27:20 +00:00
|
|
|
Vector<real_t> args;
|
|
|
|
Error err = _parse_construct<real_t>(p_stream, args, line, r_err_str);
|
2020-05-14 14:41:43 +00:00
|
|
|
if (err) {
|
2015-11-24 13:42:05 +00:00
|
|
|
return err;
|
2020-05-14 14:41:43 +00:00
|
|
|
}
|
2015-11-24 13:42:05 +00:00
|
|
|
|
|
|
|
if (args.size() != 9) {
|
|
|
|
r_err_str = "Expected 9 arguments for constructor";
|
2020-06-08 09:36:41 +00:00
|
|
|
return ERR_PARSE_ERROR;
|
2015-11-24 13:42:05 +00:00
|
|
|
}
|
|
|
|
|
2017-01-11 03:52:51 +00:00
|
|
|
value = Basis(args[0], args[1], args[2], args[3], args[4], args[5], args[6], args[7], args[8]);
|
2021-06-04 12:06:46 +00:00
|
|
|
} else if (id == "Transform3D" || id == "Transform") { // "Transform" kept for compatibility with Godot <4.
|
2021-01-29 19:27:20 +00:00
|
|
|
Vector<real_t> args;
|
|
|
|
Error err = _parse_construct<real_t>(p_stream, args, line, r_err_str);
|
2020-05-14 14:41:43 +00:00
|
|
|
if (err) {
|
2015-11-24 13:42:05 +00:00
|
|
|
return err;
|
2020-05-14 14:41:43 +00:00
|
|
|
}
|
2015-11-24 13:42:05 +00:00
|
|
|
|
|
|
|
if (args.size() != 12) {
|
|
|
|
r_err_str = "Expected 12 arguments for constructor";
|
2020-06-08 09:36:41 +00:00
|
|
|
return ERR_PARSE_ERROR;
|
2015-11-24 13:42:05 +00:00
|
|
|
}
|
|
|
|
|
2020-10-17 05:08:21 +00:00
|
|
|
value = Transform3D(Basis(args[0], args[1], args[2], args[3], args[4], args[5], args[6], args[7], args[8]), Vector3(args[9], args[10], args[11]));
|
Implement Vector4, Vector4i, Projection
Implement built-in classes Vector4, Vector4i and Projection.
* Two versions of Vector4 (float and integer).
* A Projection class, which is a 4x4 matrix specialized in projection types.
These types have been requested for a long time, but given they were very corner case they were not added before.
Because in Godot 4, reimplementing parts of the rendering engine is now possible, access to these types (heavily used by the rendering code) becomes a necessity.
**Q**: Why Projection and not Matrix4?
**A**: Godot does not use Matrix2, Matrix3, Matrix4x3, etc. naming convention because, within the engine, these types always have a *purpose*. As such, Godot names them: Transform2D, Transform3D or Basis. In this case, this 4x4 matrix is _always_ used as a _Projection_, hence the naming.
2022-07-19 23:11:13 +00:00
|
|
|
} else if (id == "Projection") { // "Transform" kept for compatibility with Godot <4.
|
|
|
|
Vector<real_t> args;
|
|
|
|
Error err = _parse_construct<real_t>(p_stream, args, line, r_err_str);
|
|
|
|
if (err) {
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (args.size() != 16) {
|
|
|
|
r_err_str = "Expected 16 arguments for constructor";
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
value = Projection(Vector4(args[0], args[1], args[2], args[3]), Vector4(args[4], args[5], args[6], args[7]), Vector4(args[8], args[9], args[10], args[11]), Vector4(args[12], args[13], args[14], args[15]));
|
2015-11-24 13:42:05 +00:00
|
|
|
} else if (id == "Color") {
|
|
|
|
Vector<float> args;
|
2015-11-28 23:56:14 +00:00
|
|
|
Error err = _parse_construct<float>(p_stream, args, line, r_err_str);
|
2020-05-14 14:41:43 +00:00
|
|
|
if (err) {
|
2015-11-24 13:42:05 +00:00
|
|
|
return err;
|
2020-05-14 14:41:43 +00:00
|
|
|
}
|
2015-11-24 13:42:05 +00:00
|
|
|
|
|
|
|
if (args.size() != 4) {
|
|
|
|
r_err_str = "Expected 4 arguments for constructor";
|
2020-06-08 09:36:41 +00:00
|
|
|
return ERR_PARSE_ERROR;
|
2015-11-24 13:42:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
value = Color(args[0], args[1], args[2], args[3]);
|
|
|
|
} else if (id == "NodePath") {
|
|
|
|
get_token(p_stream, token, line, r_err_str);
|
|
|
|
if (token.type != TK_PARENTHESIS_OPEN) {
|
|
|
|
r_err_str = "Expected '('";
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
get_token(p_stream, token, line, r_err_str);
|
|
|
|
if (token.type != TK_STRING) {
|
2015-11-28 23:56:14 +00:00
|
|
|
r_err_str = "Expected string as argument for NodePath()";
|
2015-11-24 13:42:05 +00:00
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
value = NodePath(String(token.value));
|
|
|
|
|
|
|
|
get_token(p_stream, token, line, r_err_str);
|
|
|
|
if (token.type != TK_PARENTHESIS_CLOSE) {
|
|
|
|
r_err_str = "Expected ')'";
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
|
|
|
} else if (id == "RID") {
|
|
|
|
get_token(p_stream, token, line, r_err_str);
|
|
|
|
if (token.type != TK_PARENTHESIS_OPEN) {
|
|
|
|
r_err_str = "Expected '('";
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
get_token(p_stream, token, line, r_err_str);
|
2023-06-21 09:06:32 +00:00
|
|
|
// Permit empty RID.
|
|
|
|
if (token.type == TK_PARENTHESIS_CLOSE) {
|
|
|
|
value = RID();
|
|
|
|
return OK;
|
|
|
|
} else if (token.type != TK_NUMBER) {
|
|
|
|
r_err_str = "Expected number as argument or ')'";
|
2015-11-24 13:42:05 +00:00
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
|
|
|
|
2023-06-21 09:06:32 +00:00
|
|
|
value = RID::from_uint64(token.value);
|
|
|
|
|
|
|
|
get_token(p_stream, token, line, r_err_str);
|
|
|
|
if (token.type != TK_PARENTHESIS_CLOSE) {
|
|
|
|
r_err_str = "Expected ')'";
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
|
|
|
} else if (id == "Signal") {
|
|
|
|
get_token(p_stream, token, line, r_err_str);
|
|
|
|
if (token.type != TK_PARENTHESIS_OPEN) {
|
|
|
|
r_err_str = "Expected '('";
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Load as empty.
|
|
|
|
value = Signal();
|
|
|
|
|
|
|
|
get_token(p_stream, token, line, r_err_str);
|
|
|
|
if (token.type != TK_PARENTHESIS_CLOSE) {
|
|
|
|
r_err_str = "Expected ')'";
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
|
|
|
} else if (id == "Callable") {
|
|
|
|
get_token(p_stream, token, line, r_err_str);
|
|
|
|
if (token.type != TK_PARENTHESIS_OPEN) {
|
|
|
|
r_err_str = "Expected '('";
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Load as empty.
|
|
|
|
value = Callable();
|
2015-11-24 13:42:05 +00:00
|
|
|
|
|
|
|
get_token(p_stream, token, line, r_err_str);
|
|
|
|
if (token.type != TK_PARENTHESIS_CLOSE) {
|
|
|
|
r_err_str = "Expected ')'";
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
2017-05-29 00:46:48 +00:00
|
|
|
} else if (id == "Object") {
|
|
|
|
get_token(p_stream, token, line, r_err_str);
|
|
|
|
if (token.type != TK_PARENTHESIS_OPEN) {
|
|
|
|
r_err_str = "Expected '('";
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
get_token(p_stream, token, line, r_err_str);
|
|
|
|
|
|
|
|
if (token.type != TK_IDENTIFIER) {
|
|
|
|
r_err_str = "Expected identifier with type of object";
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
String type = token.value;
|
|
|
|
|
2021-06-17 22:03:09 +00:00
|
|
|
Object *obj = ClassDB::instantiate(type);
|
2017-05-29 00:46:48 +00:00
|
|
|
|
|
|
|
if (!obj) {
|
2021-06-17 22:03:09 +00:00
|
|
|
r_err_str = "Can't instantiate Object() of type: " + type;
|
2017-05-29 00:46:48 +00:00
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
|
|
|
|
2022-05-02 23:43:50 +00:00
|
|
|
Ref<RefCounted> ref = Ref<RefCounted>(Object::cast_to<RefCounted>(obj));
|
2020-12-17 11:24:57 +00:00
|
|
|
|
2017-05-29 00:46:48 +00:00
|
|
|
get_token(p_stream, token, line, r_err_str);
|
|
|
|
if (token.type != TK_COMMA) {
|
|
|
|
r_err_str = "Expected ',' after object type";
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool at_key = true;
|
|
|
|
String key;
|
|
|
|
bool need_comma = false;
|
|
|
|
|
|
|
|
while (true) {
|
|
|
|
if (p_stream->is_eof()) {
|
|
|
|
r_err_str = "Unexpected End of File while parsing Object()";
|
|
|
|
return ERR_FILE_CORRUPT;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (at_key) {
|
2022-11-27 07:56:53 +00:00
|
|
|
Error err = get_token(p_stream, token, line, r_err_str);
|
2020-05-14 14:41:43 +00:00
|
|
|
if (err != OK) {
|
2017-05-29 00:46:48 +00:00
|
|
|
return err;
|
2020-05-14 14:41:43 +00:00
|
|
|
}
|
2017-05-29 00:46:48 +00:00
|
|
|
|
2022-11-27 07:56:53 +00:00
|
|
|
if (token.type == TK_PARENTHESIS_CLOSE) {
|
2020-12-17 11:24:57 +00:00
|
|
|
value = ref.is_valid() ? Variant(ref) : Variant(obj);
|
2017-05-29 00:46:48 +00:00
|
|
|
return OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (need_comma) {
|
2022-11-27 07:56:53 +00:00
|
|
|
if (token.type != TK_COMMA) {
|
2017-05-29 00:46:48 +00:00
|
|
|
r_err_str = "Expected '}' or ','";
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
} else {
|
|
|
|
need_comma = false;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-11-27 07:56:53 +00:00
|
|
|
if (token.type != TK_STRING) {
|
2017-05-29 00:46:48 +00:00
|
|
|
r_err_str = "Expected property name as string";
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
|
|
|
|
2022-11-27 07:56:53 +00:00
|
|
|
key = token.value;
|
2017-05-29 00:46:48 +00:00
|
|
|
|
2022-11-27 07:56:53 +00:00
|
|
|
err = get_token(p_stream, token, line, r_err_str);
|
2017-05-29 00:46:48 +00:00
|
|
|
|
2020-05-14 14:41:43 +00:00
|
|
|
if (err != OK) {
|
2017-05-29 00:46:48 +00:00
|
|
|
return err;
|
2020-05-14 14:41:43 +00:00
|
|
|
}
|
2022-11-27 07:56:53 +00:00
|
|
|
if (token.type != TK_COLON) {
|
2017-05-29 00:46:48 +00:00
|
|
|
r_err_str = "Expected ':'";
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
|
|
|
at_key = false;
|
|
|
|
} else {
|
2022-11-27 07:56:53 +00:00
|
|
|
Error err = get_token(p_stream, token, line, r_err_str);
|
2020-05-14 14:41:43 +00:00
|
|
|
if (err != OK) {
|
2017-05-29 00:46:48 +00:00
|
|
|
return err;
|
2020-05-14 14:41:43 +00:00
|
|
|
}
|
2017-05-29 00:46:48 +00:00
|
|
|
|
|
|
|
Variant v;
|
2022-11-27 07:56:53 +00:00
|
|
|
err = parse_value(token, v, p_stream, line, r_err_str, p_res_parser);
|
2020-05-14 14:41:43 +00:00
|
|
|
if (err) {
|
2017-05-29 00:46:48 +00:00
|
|
|
return err;
|
2020-05-14 14:41:43 +00:00
|
|
|
}
|
2017-05-29 00:46:48 +00:00
|
|
|
obj->set(key, v);
|
|
|
|
need_comma = true;
|
|
|
|
at_key = true;
|
|
|
|
}
|
|
|
|
}
|
2015-11-28 23:56:14 +00:00
|
|
|
} else if (id == "Resource" || id == "SubResource" || id == "ExtResource") {
|
2015-11-24 13:42:05 +00:00
|
|
|
get_token(p_stream, token, line, r_err_str);
|
|
|
|
if (token.type != TK_PARENTHESIS_OPEN) {
|
|
|
|
r_err_str = "Expected '('";
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
|
|
|
|
2015-11-28 23:56:14 +00:00
|
|
|
if (p_res_parser && id == "Resource" && p_res_parser->func) {
|
2022-05-02 23:43:50 +00:00
|
|
|
Ref<Resource> res;
|
2015-11-28 23:56:14 +00:00
|
|
|
Error err = p_res_parser->func(p_res_parser->userdata, p_stream, res, line, r_err_str);
|
2020-05-14 14:41:43 +00:00
|
|
|
if (err) {
|
2015-11-28 23:56:14 +00:00
|
|
|
return err;
|
2020-05-14 14:41:43 +00:00
|
|
|
}
|
2015-11-24 13:42:05 +00:00
|
|
|
|
|
|
|
value = res;
|
2015-11-28 23:56:14 +00:00
|
|
|
} else if (p_res_parser && id == "ExtResource" && p_res_parser->ext_func) {
|
2022-05-02 23:43:50 +00:00
|
|
|
Ref<Resource> res;
|
2015-11-28 23:56:14 +00:00
|
|
|
Error err = p_res_parser->ext_func(p_res_parser->userdata, p_stream, res, line, r_err_str);
|
2020-05-14 14:41:43 +00:00
|
|
|
if (err) {
|
2023-11-22 11:13:28 +00:00
|
|
|
// If the file is missing, the error can be ignored.
|
2024-04-05 15:49:11 +00:00
|
|
|
if (err != ERR_FILE_NOT_FOUND && err != ERR_CANT_OPEN && err != ERR_FILE_CANT_OPEN) {
|
2023-11-22 11:13:28 +00:00
|
|
|
return err;
|
|
|
|
}
|
2020-05-14 14:41:43 +00:00
|
|
|
}
|
2015-11-28 23:56:14 +00:00
|
|
|
|
|
|
|
value = res;
|
|
|
|
} else if (p_res_parser && id == "SubResource" && p_res_parser->sub_func) {
|
2022-05-02 23:43:50 +00:00
|
|
|
Ref<Resource> res;
|
2015-11-28 23:56:14 +00:00
|
|
|
Error err = p_res_parser->sub_func(p_res_parser->userdata, p_stream, res, line, r_err_str);
|
2020-05-14 14:41:43 +00:00
|
|
|
if (err) {
|
2015-11-24 13:42:05 +00:00
|
|
|
return err;
|
2020-05-14 14:41:43 +00:00
|
|
|
}
|
2015-11-24 13:42:05 +00:00
|
|
|
|
|
|
|
value = res;
|
|
|
|
} else {
|
2015-11-28 23:56:14 +00:00
|
|
|
get_token(p_stream, token, line, r_err_str);
|
|
|
|
if (token.type == TK_STRING) {
|
|
|
|
String path = token.value;
|
2022-05-02 23:43:50 +00:00
|
|
|
Ref<Resource> res = ResourceLoader::load(path);
|
2015-11-28 23:56:14 +00:00
|
|
|
if (res.is_null()) {
|
|
|
|
r_err_str = "Can't load resource at path: '" + path + "'.";
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
get_token(p_stream, token, line, r_err_str);
|
|
|
|
if (token.type != TK_PARENTHESIS_CLOSE) {
|
|
|
|
r_err_str = "Expected ')'";
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
value = res;
|
|
|
|
} else {
|
|
|
|
r_err_str = "Expected string as argument for Resource().";
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
2015-11-24 13:42:05 +00:00
|
|
|
}
|
2023-06-24 18:03:28 +00:00
|
|
|
} else if (id == "Dictionary") {
|
|
|
|
Error err = OK;
|
|
|
|
|
|
|
|
get_token(p_stream, token, line, r_err_str);
|
|
|
|
if (token.type != TK_BRACKET_OPEN) {
|
|
|
|
r_err_str = "Expected '['";
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
get_token(p_stream, token, line, r_err_str);
|
|
|
|
if (token.type != TK_IDENTIFIER) {
|
|
|
|
r_err_str = "Expected type identifier for key";
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
static HashMap<StringName, Variant::Type> builtin_types;
|
|
|
|
if (builtin_types.is_empty()) {
|
|
|
|
for (int i = 1; i < Variant::VARIANT_MAX; i++) {
|
|
|
|
builtin_types[Variant::get_type_name((Variant::Type)i)] = (Variant::Type)i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Dictionary dict;
|
|
|
|
Variant::Type key_type = Variant::NIL;
|
|
|
|
StringName key_class_name;
|
|
|
|
Variant key_script;
|
|
|
|
bool got_comma_token = false;
|
|
|
|
if (builtin_types.has(token.value)) {
|
|
|
|
key_type = builtin_types.get(token.value);
|
|
|
|
} else if (token.value == "Resource" || token.value == "SubResource" || token.value == "ExtResource") {
|
|
|
|
Variant resource;
|
|
|
|
err = parse_value(token, resource, p_stream, line, r_err_str, p_res_parser);
|
|
|
|
if (err) {
|
|
|
|
if (token.value == "Resource" && err == ERR_PARSE_ERROR && r_err_str == "Expected '('" && token.type == TK_COMMA) {
|
|
|
|
err = OK;
|
|
|
|
r_err_str = String();
|
|
|
|
key_type = Variant::OBJECT;
|
|
|
|
key_class_name = token.value;
|
|
|
|
got_comma_token = true;
|
|
|
|
} else {
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
Ref<Script> script = resource;
|
|
|
|
if (script.is_valid() && script->is_valid()) {
|
|
|
|
key_type = Variant::OBJECT;
|
|
|
|
key_class_name = script->get_instance_base_type();
|
|
|
|
key_script = script;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else if (ClassDB::class_exists(token.value)) {
|
|
|
|
key_type = Variant::OBJECT;
|
|
|
|
key_class_name = token.value;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!got_comma_token) {
|
|
|
|
get_token(p_stream, token, line, r_err_str);
|
|
|
|
if (token.type != TK_COMMA) {
|
|
|
|
r_err_str = "Expected ',' after key type";
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
get_token(p_stream, token, line, r_err_str);
|
|
|
|
if (token.type != TK_IDENTIFIER) {
|
|
|
|
r_err_str = "Expected type identifier for value";
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
Variant::Type value_type = Variant::NIL;
|
|
|
|
StringName value_class_name;
|
|
|
|
Variant value_script;
|
|
|
|
bool got_bracket_token = false;
|
|
|
|
if (builtin_types.has(token.value)) {
|
|
|
|
value_type = builtin_types.get(token.value);
|
|
|
|
} else if (token.value == "Resource" || token.value == "SubResource" || token.value == "ExtResource") {
|
|
|
|
Variant resource;
|
|
|
|
err = parse_value(token, resource, p_stream, line, r_err_str, p_res_parser);
|
|
|
|
if (err) {
|
|
|
|
if (token.value == "Resource" && err == ERR_PARSE_ERROR && r_err_str == "Expected '('" && token.type == TK_BRACKET_CLOSE) {
|
|
|
|
err = OK;
|
|
|
|
r_err_str = String();
|
|
|
|
value_type = Variant::OBJECT;
|
|
|
|
value_class_name = token.value;
|
|
|
|
got_comma_token = true;
|
|
|
|
} else {
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
Ref<Script> script = resource;
|
|
|
|
if (script.is_valid() && script->is_valid()) {
|
|
|
|
value_type = Variant::OBJECT;
|
|
|
|
value_class_name = script->get_instance_base_type();
|
|
|
|
value_script = script;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else if (ClassDB::class_exists(token.value)) {
|
|
|
|
value_type = Variant::OBJECT;
|
|
|
|
value_class_name = token.value;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (key_type != Variant::NIL || value_type != Variant::NIL) {
|
|
|
|
dict.set_typed(key_type, key_class_name, key_script, value_type, value_class_name, value_script);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!got_bracket_token) {
|
|
|
|
get_token(p_stream, token, line, r_err_str);
|
|
|
|
if (token.type != TK_BRACKET_CLOSE) {
|
|
|
|
r_err_str = "Expected ']'";
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
get_token(p_stream, token, line, r_err_str);
|
|
|
|
if (token.type != TK_PARENTHESIS_OPEN) {
|
|
|
|
r_err_str = "Expected '('";
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
get_token(p_stream, token, line, r_err_str);
|
|
|
|
if (token.type != TK_CURLY_BRACKET_OPEN) {
|
|
|
|
r_err_str = "Expected '{'";
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
Dictionary values;
|
|
|
|
err = _parse_dictionary(values, p_stream, line, r_err_str, p_res_parser);
|
|
|
|
if (err) {
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
get_token(p_stream, token, line, r_err_str);
|
|
|
|
if (token.type != TK_PARENTHESIS_CLOSE) {
|
|
|
|
r_err_str = "Expected ')'";
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
dict.assign(values);
|
|
|
|
|
|
|
|
value = dict;
|
2022-11-27 07:56:53 +00:00
|
|
|
} else if (id == "Array") {
|
|
|
|
Error err = OK;
|
|
|
|
|
|
|
|
get_token(p_stream, token, line, r_err_str);
|
|
|
|
if (token.type != TK_BRACKET_OPEN) {
|
|
|
|
r_err_str = "Expected '['";
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
get_token(p_stream, token, line, r_err_str);
|
|
|
|
if (token.type != TK_IDENTIFIER) {
|
|
|
|
r_err_str = "Expected type identifier";
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
|
|
|
|
2023-10-19 15:11:42 +00:00
|
|
|
static HashMap<String, Variant::Type> builtin_types;
|
2022-11-27 07:56:53 +00:00
|
|
|
if (builtin_types.is_empty()) {
|
|
|
|
for (int i = 1; i < Variant::VARIANT_MAX; i++) {
|
|
|
|
builtin_types[Variant::get_type_name((Variant::Type)i)] = (Variant::Type)i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Array array = Array();
|
|
|
|
bool got_bracket_token = false;
|
|
|
|
if (builtin_types.has(token.value)) {
|
|
|
|
array.set_typed(builtin_types.get(token.value), StringName(), Variant());
|
|
|
|
} else if (token.value == "Resource" || token.value == "SubResource" || token.value == "ExtResource") {
|
|
|
|
Variant resource;
|
|
|
|
err = parse_value(token, resource, p_stream, line, r_err_str, p_res_parser);
|
|
|
|
if (err) {
|
|
|
|
if (token.value == "Resource" && err == ERR_PARSE_ERROR && r_err_str == "Expected '('" && token.type == TK_BRACKET_CLOSE) {
|
|
|
|
err = OK;
|
|
|
|
r_err_str = String();
|
|
|
|
array.set_typed(Variant::OBJECT, token.value, Variant());
|
|
|
|
got_bracket_token = true;
|
|
|
|
} else {
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
Ref<Script> script = resource;
|
|
|
|
if (script.is_valid() && script->is_valid()) {
|
|
|
|
array.set_typed(Variant::OBJECT, script->get_instance_base_type(), script);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else if (ClassDB::class_exists(token.value)) {
|
|
|
|
array.set_typed(Variant::OBJECT, token.value, Variant());
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!got_bracket_token) {
|
|
|
|
get_token(p_stream, token, line, r_err_str);
|
|
|
|
if (token.type != TK_BRACKET_CLOSE) {
|
|
|
|
r_err_str = "Expected ']'";
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
get_token(p_stream, token, line, r_err_str);
|
|
|
|
if (token.type != TK_PARENTHESIS_OPEN) {
|
|
|
|
r_err_str = "Expected '('";
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
get_token(p_stream, token, line, r_err_str);
|
|
|
|
if (token.type != TK_BRACKET_OPEN) {
|
|
|
|
r_err_str = "Expected '['";
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
Array values;
|
|
|
|
err = _parse_array(values, p_stream, line, r_err_str, p_res_parser);
|
|
|
|
if (err) {
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
get_token(p_stream, token, line, r_err_str);
|
|
|
|
if (token.type != TK_PARENTHESIS_CLOSE) {
|
|
|
|
r_err_str = "Expected ')'";
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
array.assign(values);
|
|
|
|
|
|
|
|
value = array;
|
2020-02-17 21:06:54 +00:00
|
|
|
} else if (id == "PackedByteArray" || id == "PoolByteArray" || id == "ByteArray") {
|
2015-11-28 23:56:14 +00:00
|
|
|
Vector<uint8_t> args;
|
2024-03-05 16:53:57 +00:00
|
|
|
Error err = _parse_byte_array(p_stream, args, line, r_err_str);
|
2020-05-14 14:41:43 +00:00
|
|
|
if (err) {
|
2015-11-24 13:42:05 +00:00
|
|
|
return err;
|
2020-05-14 14:41:43 +00:00
|
|
|
}
|
2015-11-24 13:42:05 +00:00
|
|
|
|
2020-02-17 21:06:54 +00:00
|
|
|
Vector<uint8_t> arr;
|
2015-11-24 13:42:05 +00:00
|
|
|
{
|
|
|
|
int len = args.size();
|
|
|
|
arr.resize(len);
|
2020-02-17 21:06:54 +00:00
|
|
|
uint8_t *w = arr.ptrw();
|
2015-11-24 13:42:05 +00:00
|
|
|
for (int i = 0; i < len; i++) {
|
|
|
|
w[i] = args[i];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
value = arr;
|
Variant: Added 64-bit packed arrays, renamed Variant::REAL to FLOAT.
- Renames PackedIntArray to PackedInt32Array.
- Renames PackedFloatArray to PackedFloat32Array.
- Adds PackedInt64Array and PackedFloat64Array.
- Renames Variant::REAL to Variant::FLOAT for consistency.
Packed arrays are for storing large amount of data and creating stuff like
meshes, buffers. textures, etc. Forcing them to be 64 is a huge waste of
memory. That said, many users requested the ability to have 64 bits packed
arrays for their games, so this is just an optional added type.
For Variant, the float datatype is always 64 bits, and exposed as `float`.
We still have `real_t` which is the datatype that can change from 32 to 64
bits depending on a compile flag (not entirely working right now, but that's
the idea). It affects math related datatypes and code only.
Neither Variant nor PackedArray make use of real_t, which is only intended
for math precision, so the term is removed from there to keep only float.
2020-02-24 18:20:53 +00:00
|
|
|
} else if (id == "PackedInt32Array" || id == "PackedIntArray" || id == "PoolIntArray" || id == "IntArray") {
|
|
|
|
Vector<int32_t> args;
|
|
|
|
Error err = _parse_construct<int32_t>(p_stream, args, line, r_err_str);
|
2020-05-14 14:41:43 +00:00
|
|
|
if (err) {
|
2015-11-24 13:42:05 +00:00
|
|
|
return err;
|
2020-05-14 14:41:43 +00:00
|
|
|
}
|
2015-11-24 13:42:05 +00:00
|
|
|
|
Variant: Added 64-bit packed arrays, renamed Variant::REAL to FLOAT.
- Renames PackedIntArray to PackedInt32Array.
- Renames PackedFloatArray to PackedFloat32Array.
- Adds PackedInt64Array and PackedFloat64Array.
- Renames Variant::REAL to Variant::FLOAT for consistency.
Packed arrays are for storing large amount of data and creating stuff like
meshes, buffers. textures, etc. Forcing them to be 64 is a huge waste of
memory. That said, many users requested the ability to have 64 bits packed
arrays for their games, so this is just an optional added type.
For Variant, the float datatype is always 64 bits, and exposed as `float`.
We still have `real_t` which is the datatype that can change from 32 to 64
bits depending on a compile flag (not entirely working right now, but that's
the idea). It affects math related datatypes and code only.
Neither Variant nor PackedArray make use of real_t, which is only intended
for math precision, so the term is removed from there to keep only float.
2020-02-24 18:20:53 +00:00
|
|
|
Vector<int32_t> arr;
|
2015-11-24 13:42:05 +00:00
|
|
|
{
|
Variant: Added 64-bit packed arrays, renamed Variant::REAL to FLOAT.
- Renames PackedIntArray to PackedInt32Array.
- Renames PackedFloatArray to PackedFloat32Array.
- Adds PackedInt64Array and PackedFloat64Array.
- Renames Variant::REAL to Variant::FLOAT for consistency.
Packed arrays are for storing large amount of data and creating stuff like
meshes, buffers. textures, etc. Forcing them to be 64 is a huge waste of
memory. That said, many users requested the ability to have 64 bits packed
arrays for their games, so this is just an optional added type.
For Variant, the float datatype is always 64 bits, and exposed as `float`.
We still have `real_t` which is the datatype that can change from 32 to 64
bits depending on a compile flag (not entirely working right now, but that's
the idea). It affects math related datatypes and code only.
Neither Variant nor PackedArray make use of real_t, which is only intended
for math precision, so the term is removed from there to keep only float.
2020-02-24 18:20:53 +00:00
|
|
|
int32_t len = args.size();
|
2015-11-24 13:42:05 +00:00
|
|
|
arr.resize(len);
|
Variant: Added 64-bit packed arrays, renamed Variant::REAL to FLOAT.
- Renames PackedIntArray to PackedInt32Array.
- Renames PackedFloatArray to PackedFloat32Array.
- Adds PackedInt64Array and PackedFloat64Array.
- Renames Variant::REAL to Variant::FLOAT for consistency.
Packed arrays are for storing large amount of data and creating stuff like
meshes, buffers. textures, etc. Forcing them to be 64 is a huge waste of
memory. That said, many users requested the ability to have 64 bits packed
arrays for their games, so this is just an optional added type.
For Variant, the float datatype is always 64 bits, and exposed as `float`.
We still have `real_t` which is the datatype that can change from 32 to 64
bits depending on a compile flag (not entirely working right now, but that's
the idea). It affects math related datatypes and code only.
Neither Variant nor PackedArray make use of real_t, which is only intended
for math precision, so the term is removed from there to keep only float.
2020-02-24 18:20:53 +00:00
|
|
|
int32_t *w = arr.ptrw();
|
|
|
|
for (int32_t i = 0; i < len; i++) {
|
|
|
|
w[i] = int32_t(args[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
value = arr;
|
|
|
|
} else if (id == "PackedInt64Array") {
|
|
|
|
Vector<int64_t> args;
|
|
|
|
Error err = _parse_construct<int64_t>(p_stream, args, line, r_err_str);
|
2020-05-14 14:41:43 +00:00
|
|
|
if (err) {
|
Variant: Added 64-bit packed arrays, renamed Variant::REAL to FLOAT.
- Renames PackedIntArray to PackedInt32Array.
- Renames PackedFloatArray to PackedFloat32Array.
- Adds PackedInt64Array and PackedFloat64Array.
- Renames Variant::REAL to Variant::FLOAT for consistency.
Packed arrays are for storing large amount of data and creating stuff like
meshes, buffers. textures, etc. Forcing them to be 64 is a huge waste of
memory. That said, many users requested the ability to have 64 bits packed
arrays for their games, so this is just an optional added type.
For Variant, the float datatype is always 64 bits, and exposed as `float`.
We still have `real_t` which is the datatype that can change from 32 to 64
bits depending on a compile flag (not entirely working right now, but that's
the idea). It affects math related datatypes and code only.
Neither Variant nor PackedArray make use of real_t, which is only intended
for math precision, so the term is removed from there to keep only float.
2020-02-24 18:20:53 +00:00
|
|
|
return err;
|
2020-05-14 14:41:43 +00:00
|
|
|
}
|
Variant: Added 64-bit packed arrays, renamed Variant::REAL to FLOAT.
- Renames PackedIntArray to PackedInt32Array.
- Renames PackedFloatArray to PackedFloat32Array.
- Adds PackedInt64Array and PackedFloat64Array.
- Renames Variant::REAL to Variant::FLOAT for consistency.
Packed arrays are for storing large amount of data and creating stuff like
meshes, buffers. textures, etc. Forcing them to be 64 is a huge waste of
memory. That said, many users requested the ability to have 64 bits packed
arrays for their games, so this is just an optional added type.
For Variant, the float datatype is always 64 bits, and exposed as `float`.
We still have `real_t` which is the datatype that can change from 32 to 64
bits depending on a compile flag (not entirely working right now, but that's
the idea). It affects math related datatypes and code only.
Neither Variant nor PackedArray make use of real_t, which is only intended
for math precision, so the term is removed from there to keep only float.
2020-02-24 18:20:53 +00:00
|
|
|
|
|
|
|
Vector<int64_t> arr;
|
|
|
|
{
|
|
|
|
int64_t len = args.size();
|
|
|
|
arr.resize(len);
|
|
|
|
int64_t *w = arr.ptrw();
|
|
|
|
for (int64_t i = 0; i < len; i++) {
|
|
|
|
w[i] = int64_t(args[i]);
|
2015-11-24 13:42:05 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
value = arr;
|
Variant: Added 64-bit packed arrays, renamed Variant::REAL to FLOAT.
- Renames PackedIntArray to PackedInt32Array.
- Renames PackedFloatArray to PackedFloat32Array.
- Adds PackedInt64Array and PackedFloat64Array.
- Renames Variant::REAL to Variant::FLOAT for consistency.
Packed arrays are for storing large amount of data and creating stuff like
meshes, buffers. textures, etc. Forcing them to be 64 is a huge waste of
memory. That said, many users requested the ability to have 64 bits packed
arrays for their games, so this is just an optional added type.
For Variant, the float datatype is always 64 bits, and exposed as `float`.
We still have `real_t` which is the datatype that can change from 32 to 64
bits depending on a compile flag (not entirely working right now, but that's
the idea). It affects math related datatypes and code only.
Neither Variant nor PackedArray make use of real_t, which is only intended
for math precision, so the term is removed from there to keep only float.
2020-02-24 18:20:53 +00:00
|
|
|
} else if (id == "PackedFloat32Array" || id == "PackedRealArray" || id == "PoolRealArray" || id == "FloatArray") {
|
2015-11-24 13:42:05 +00:00
|
|
|
Vector<float> args;
|
2015-11-28 23:56:14 +00:00
|
|
|
Error err = _parse_construct<float>(p_stream, args, line, r_err_str);
|
2020-05-14 14:41:43 +00:00
|
|
|
if (err) {
|
2015-11-24 13:42:05 +00:00
|
|
|
return err;
|
2020-05-14 14:41:43 +00:00
|
|
|
}
|
2015-11-24 13:42:05 +00:00
|
|
|
|
2020-02-17 21:06:54 +00:00
|
|
|
Vector<float> arr;
|
2015-11-24 13:42:05 +00:00
|
|
|
{
|
|
|
|
int len = args.size();
|
|
|
|
arr.resize(len);
|
2020-02-17 21:06:54 +00:00
|
|
|
float *w = arr.ptrw();
|
2015-11-24 13:42:05 +00:00
|
|
|
for (int i = 0; i < len; i++) {
|
|
|
|
w[i] = args[i];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
value = arr;
|
Variant: Added 64-bit packed arrays, renamed Variant::REAL to FLOAT.
- Renames PackedIntArray to PackedInt32Array.
- Renames PackedFloatArray to PackedFloat32Array.
- Adds PackedInt64Array and PackedFloat64Array.
- Renames Variant::REAL to Variant::FLOAT for consistency.
Packed arrays are for storing large amount of data and creating stuff like
meshes, buffers. textures, etc. Forcing them to be 64 is a huge waste of
memory. That said, many users requested the ability to have 64 bits packed
arrays for their games, so this is just an optional added type.
For Variant, the float datatype is always 64 bits, and exposed as `float`.
We still have `real_t` which is the datatype that can change from 32 to 64
bits depending on a compile flag (not entirely working right now, but that's
the idea). It affects math related datatypes and code only.
Neither Variant nor PackedArray make use of real_t, which is only intended
for math precision, so the term is removed from there to keep only float.
2020-02-24 18:20:53 +00:00
|
|
|
} else if (id == "PackedFloat64Array") {
|
|
|
|
Vector<double> args;
|
|
|
|
Error err = _parse_construct<double>(p_stream, args, line, r_err_str);
|
2020-05-14 14:41:43 +00:00
|
|
|
if (err) {
|
Variant: Added 64-bit packed arrays, renamed Variant::REAL to FLOAT.
- Renames PackedIntArray to PackedInt32Array.
- Renames PackedFloatArray to PackedFloat32Array.
- Adds PackedInt64Array and PackedFloat64Array.
- Renames Variant::REAL to Variant::FLOAT for consistency.
Packed arrays are for storing large amount of data and creating stuff like
meshes, buffers. textures, etc. Forcing them to be 64 is a huge waste of
memory. That said, many users requested the ability to have 64 bits packed
arrays for their games, so this is just an optional added type.
For Variant, the float datatype is always 64 bits, and exposed as `float`.
We still have `real_t` which is the datatype that can change from 32 to 64
bits depending on a compile flag (not entirely working right now, but that's
the idea). It affects math related datatypes and code only.
Neither Variant nor PackedArray make use of real_t, which is only intended
for math precision, so the term is removed from there to keep only float.
2020-02-24 18:20:53 +00:00
|
|
|
return err;
|
2020-05-14 14:41:43 +00:00
|
|
|
}
|
Variant: Added 64-bit packed arrays, renamed Variant::REAL to FLOAT.
- Renames PackedIntArray to PackedInt32Array.
- Renames PackedFloatArray to PackedFloat32Array.
- Adds PackedInt64Array and PackedFloat64Array.
- Renames Variant::REAL to Variant::FLOAT for consistency.
Packed arrays are for storing large amount of data and creating stuff like
meshes, buffers. textures, etc. Forcing them to be 64 is a huge waste of
memory. That said, many users requested the ability to have 64 bits packed
arrays for their games, so this is just an optional added type.
For Variant, the float datatype is always 64 bits, and exposed as `float`.
We still have `real_t` which is the datatype that can change from 32 to 64
bits depending on a compile flag (not entirely working right now, but that's
the idea). It affects math related datatypes and code only.
Neither Variant nor PackedArray make use of real_t, which is only intended
for math precision, so the term is removed from there to keep only float.
2020-02-24 18:20:53 +00:00
|
|
|
|
|
|
|
Vector<double> arr;
|
|
|
|
{
|
|
|
|
int len = args.size();
|
|
|
|
arr.resize(len);
|
|
|
|
double *w = arr.ptrw();
|
|
|
|
for (int i = 0; i < len; i++) {
|
|
|
|
w[i] = args[i];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
value = arr;
|
2020-02-17 21:06:54 +00:00
|
|
|
} else if (id == "PackedStringArray" || id == "PoolStringArray" || id == "StringArray") {
|
2015-11-24 13:42:05 +00:00
|
|
|
get_token(p_stream, token, line, r_err_str);
|
|
|
|
if (token.type != TK_PARENTHESIS_OPEN) {
|
|
|
|
r_err_str = "Expected '('";
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
Vector<String> cs;
|
|
|
|
|
|
|
|
bool first = true;
|
|
|
|
while (true) {
|
|
|
|
if (!first) {
|
|
|
|
get_token(p_stream, token, line, r_err_str);
|
|
|
|
if (token.type == TK_COMMA) {
|
|
|
|
//do none
|
2016-01-09 00:09:13 +00:00
|
|
|
} else if (token.type == TK_PARENTHESIS_CLOSE) {
|
2015-11-24 13:42:05 +00:00
|
|
|
break;
|
|
|
|
} else {
|
|
|
|
r_err_str = "Expected ',' or ')'";
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
get_token(p_stream, token, line, r_err_str);
|
2016-01-09 00:09:13 +00:00
|
|
|
|
2016-07-27 14:59:42 +00:00
|
|
|
if (token.type == TK_PARENTHESIS_CLOSE) {
|
|
|
|
break;
|
|
|
|
} else if (token.type != TK_STRING) {
|
2016-03-08 23:00:52 +00:00
|
|
|
r_err_str = "Expected string";
|
2015-11-24 13:42:05 +00:00
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
|
|
|
|
2016-01-09 00:09:13 +00:00
|
|
|
first = false;
|
2015-11-24 13:42:05 +00:00
|
|
|
cs.push_back(token.value);
|
|
|
|
}
|
|
|
|
|
2020-02-17 21:06:54 +00:00
|
|
|
Vector<String> arr;
|
2015-11-24 13:42:05 +00:00
|
|
|
{
|
|
|
|
int len = cs.size();
|
|
|
|
arr.resize(len);
|
2020-02-17 21:06:54 +00:00
|
|
|
String *w = arr.ptrw();
|
2015-11-24 13:42:05 +00:00
|
|
|
for (int i = 0; i < len; i++) {
|
|
|
|
w[i] = cs[i];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
value = arr;
|
2020-02-17 21:06:54 +00:00
|
|
|
} else if (id == "PackedVector2Array" || id == "PoolVector2Array" || id == "Vector2Array") {
|
2021-01-29 19:27:20 +00:00
|
|
|
Vector<real_t> args;
|
|
|
|
Error err = _parse_construct<real_t>(p_stream, args, line, r_err_str);
|
2020-05-14 14:41:43 +00:00
|
|
|
if (err) {
|
2015-11-24 13:42:05 +00:00
|
|
|
return err;
|
2020-05-14 14:41:43 +00:00
|
|
|
}
|
2015-11-24 13:42:05 +00:00
|
|
|
|
2020-02-17 21:06:54 +00:00
|
|
|
Vector<Vector2> arr;
|
2015-11-24 13:42:05 +00:00
|
|
|
{
|
|
|
|
int len = args.size() / 2;
|
|
|
|
arr.resize(len);
|
2020-02-17 21:06:54 +00:00
|
|
|
Vector2 *w = arr.ptrw();
|
2015-11-24 13:42:05 +00:00
|
|
|
for (int i = 0; i < len; i++) {
|
|
|
|
w[i] = Vector2(args[i * 2 + 0], args[i * 2 + 1]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
value = arr;
|
2020-02-17 21:06:54 +00:00
|
|
|
} else if (id == "PackedVector3Array" || id == "PoolVector3Array" || id == "Vector3Array") {
|
2021-01-29 19:27:20 +00:00
|
|
|
Vector<real_t> args;
|
|
|
|
Error err = _parse_construct<real_t>(p_stream, args, line, r_err_str);
|
2020-05-14 14:41:43 +00:00
|
|
|
if (err) {
|
2015-11-24 13:42:05 +00:00
|
|
|
return err;
|
2020-05-14 14:41:43 +00:00
|
|
|
}
|
2015-11-24 13:42:05 +00:00
|
|
|
|
2020-02-17 21:06:54 +00:00
|
|
|
Vector<Vector3> arr;
|
2015-11-24 13:42:05 +00:00
|
|
|
{
|
|
|
|
int len = args.size() / 3;
|
|
|
|
arr.resize(len);
|
2020-02-17 21:06:54 +00:00
|
|
|
Vector3 *w = arr.ptrw();
|
2015-11-24 13:42:05 +00:00
|
|
|
for (int i = 0; i < len; i++) {
|
|
|
|
w[i] = Vector3(args[i * 3 + 0], args[i * 3 + 1], args[i * 3 + 2]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-04-08 14:51:34 +00:00
|
|
|
value = arr;
|
|
|
|
} else if (id == "PackedVector4Array" || id == "PoolVector4Array" || id == "Vector4Array") {
|
|
|
|
Vector<real_t> args;
|
|
|
|
Error err = _parse_construct<real_t>(p_stream, args, line, r_err_str);
|
|
|
|
if (err) {
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
Vector<Vector4> arr;
|
|
|
|
{
|
|
|
|
int len = args.size() / 4;
|
|
|
|
arr.resize(len);
|
|
|
|
Vector4 *w = arr.ptrw();
|
|
|
|
for (int i = 0; i < len; i++) {
|
|
|
|
w[i] = Vector4(args[i * 4 + 0], args[i * 4 + 1], args[i * 4 + 2], args[i * 4 + 3]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-11-24 13:42:05 +00:00
|
|
|
value = arr;
|
2020-02-17 21:06:54 +00:00
|
|
|
} else if (id == "PackedColorArray" || id == "PoolColorArray" || id == "ColorArray") {
|
2015-11-24 13:42:05 +00:00
|
|
|
Vector<float> args;
|
2015-11-28 23:56:14 +00:00
|
|
|
Error err = _parse_construct<float>(p_stream, args, line, r_err_str);
|
2020-05-14 14:41:43 +00:00
|
|
|
if (err) {
|
2015-11-24 13:42:05 +00:00
|
|
|
return err;
|
2020-05-14 14:41:43 +00:00
|
|
|
}
|
2015-11-24 13:42:05 +00:00
|
|
|
|
2020-02-17 21:06:54 +00:00
|
|
|
Vector<Color> arr;
|
2015-11-24 13:42:05 +00:00
|
|
|
{
|
|
|
|
int len = args.size() / 4;
|
|
|
|
arr.resize(len);
|
2020-02-17 21:06:54 +00:00
|
|
|
Color *w = arr.ptrw();
|
2015-11-24 13:42:05 +00:00
|
|
|
for (int i = 0; i < len; i++) {
|
2016-01-31 23:55:48 +00:00
|
|
|
w[i] = Color(args[i * 4 + 0], args[i * 4 + 1], args[i * 4 + 2], args[i * 4 + 3]);
|
2015-11-24 13:42:05 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
value = arr;
|
|
|
|
} else {
|
|
|
|
r_err_str = "Unexpected identifier: '" + id + "'.";
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
|
|
|
|
2020-06-08 09:36:41 +00:00
|
|
|
// All above branches end up here unless they had an early return.
|
2015-11-24 13:42:05 +00:00
|
|
|
return OK;
|
|
|
|
} else if (token.type == TK_NUMBER) {
|
|
|
|
value = token.value;
|
|
|
|
return OK;
|
|
|
|
} else if (token.type == TK_STRING) {
|
2020-02-25 03:28:14 +00:00
|
|
|
value = token.value;
|
|
|
|
return OK;
|
|
|
|
} else if (token.type == TK_STRING_NAME) {
|
2015-12-31 17:30:50 +00:00
|
|
|
value = token.value;
|
|
|
|
return OK;
|
|
|
|
} else if (token.type == TK_COLOR) {
|
2015-11-24 13:42:05 +00:00
|
|
|
value = token.value;
|
|
|
|
return OK;
|
|
|
|
} else {
|
|
|
|
r_err_str = "Expected value, got " + String(tk_name[token.type]) + ".";
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Error VariantParser::_parse_array(Array &array, Stream *p_stream, int &line, String &r_err_str, ResourceParser *p_res_parser) {
|
|
|
|
Token token;
|
|
|
|
bool need_comma = false;
|
|
|
|
|
2015-11-28 23:56:14 +00:00
|
|
|
while (true) {
|
|
|
|
if (p_stream->is_eof()) {
|
|
|
|
r_err_str = "Unexpected End of File while parsing array";
|
|
|
|
return ERR_FILE_CORRUPT;
|
|
|
|
}
|
2015-11-24 13:42:05 +00:00
|
|
|
|
|
|
|
Error err = get_token(p_stream, token, line, r_err_str);
|
2020-05-14 14:41:43 +00:00
|
|
|
if (err != OK) {
|
2015-11-24 13:42:05 +00:00
|
|
|
return err;
|
2020-05-14 14:41:43 +00:00
|
|
|
}
|
2015-11-24 13:42:05 +00:00
|
|
|
|
|
|
|
if (token.type == TK_BRACKET_CLOSE) {
|
|
|
|
return OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (need_comma) {
|
|
|
|
if (token.type != TK_COMMA) {
|
|
|
|
r_err_str = "Expected ','";
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
} else {
|
|
|
|
need_comma = false;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Variant v;
|
|
|
|
err = parse_value(token, v, p_stream, line, r_err_str, p_res_parser);
|
2020-05-14 14:41:43 +00:00
|
|
|
if (err) {
|
2015-11-24 13:42:05 +00:00
|
|
|
return err;
|
2020-05-14 14:41:43 +00:00
|
|
|
}
|
2015-11-24 13:42:05 +00:00
|
|
|
|
|
|
|
array.push_back(v);
|
|
|
|
need_comma = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Error VariantParser::_parse_dictionary(Dictionary &object, Stream *p_stream, int &line, String &r_err_str, ResourceParser *p_res_parser) {
|
|
|
|
bool at_key = true;
|
|
|
|
Variant key;
|
|
|
|
Token token;
|
|
|
|
bool need_comma = false;
|
2015-11-28 23:56:14 +00:00
|
|
|
|
2015-11-24 13:42:05 +00:00
|
|
|
while (true) {
|
2015-11-28 23:56:14 +00:00
|
|
|
if (p_stream->is_eof()) {
|
|
|
|
r_err_str = "Unexpected End of File while parsing dictionary";
|
|
|
|
return ERR_FILE_CORRUPT;
|
|
|
|
}
|
2015-11-24 13:42:05 +00:00
|
|
|
|
|
|
|
if (at_key) {
|
|
|
|
Error err = get_token(p_stream, token, line, r_err_str);
|
2020-05-14 14:41:43 +00:00
|
|
|
if (err != OK) {
|
2015-11-24 13:42:05 +00:00
|
|
|
return err;
|
2020-05-14 14:41:43 +00:00
|
|
|
}
|
2015-11-24 13:42:05 +00:00
|
|
|
|
|
|
|
if (token.type == TK_CURLY_BRACKET_CLOSE) {
|
|
|
|
return OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (need_comma) {
|
|
|
|
if (token.type != TK_COMMA) {
|
|
|
|
r_err_str = "Expected '}' or ','";
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
} else {
|
|
|
|
need_comma = false;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
err = parse_value(token, key, p_stream, line, r_err_str, p_res_parser);
|
|
|
|
|
2020-05-14 14:41:43 +00:00
|
|
|
if (err) {
|
2015-11-24 13:42:05 +00:00
|
|
|
return err;
|
2020-05-14 14:41:43 +00:00
|
|
|
}
|
2015-11-24 13:42:05 +00:00
|
|
|
|
|
|
|
err = get_token(p_stream, token, line, r_err_str);
|
|
|
|
|
2020-05-14 14:41:43 +00:00
|
|
|
if (err != OK) {
|
2015-11-24 13:42:05 +00:00
|
|
|
return err;
|
2020-05-14 14:41:43 +00:00
|
|
|
}
|
2015-11-24 13:42:05 +00:00
|
|
|
if (token.type != TK_COLON) {
|
|
|
|
r_err_str = "Expected ':'";
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
|
|
|
at_key = false;
|
|
|
|
} else {
|
|
|
|
Error err = get_token(p_stream, token, line, r_err_str);
|
2020-05-14 14:41:43 +00:00
|
|
|
if (err != OK) {
|
2015-11-24 13:42:05 +00:00
|
|
|
return err;
|
2020-05-14 14:41:43 +00:00
|
|
|
}
|
2015-11-24 13:42:05 +00:00
|
|
|
|
|
|
|
Variant v;
|
|
|
|
err = parse_value(token, v, p_stream, line, r_err_str, p_res_parser);
|
2022-11-28 15:19:15 +00:00
|
|
|
if (err && err != ERR_FILE_MISSING_DEPENDENCIES) {
|
2015-11-24 13:42:05 +00:00
|
|
|
return err;
|
2020-05-14 14:41:43 +00:00
|
|
|
}
|
2015-11-24 13:42:05 +00:00
|
|
|
object[key] = v;
|
|
|
|
need_comma = true;
|
|
|
|
at_key = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-01-08 22:00:45 +00:00
|
|
|
Error VariantParser::_parse_tag(Token &token, Stream *p_stream, int &line, String &r_err_str, Tag &r_tag, ResourceParser *p_res_parser, bool p_simple_tag) {
|
2015-11-24 13:42:05 +00:00
|
|
|
r_tag.fields.clear();
|
|
|
|
|
|
|
|
if (token.type != TK_BRACKET_OPEN) {
|
|
|
|
r_err_str = "Expected '['";
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
|
|
|
|
2016-01-08 22:00:45 +00:00
|
|
|
if (p_simple_tag) {
|
|
|
|
r_tag.name = "";
|
|
|
|
r_tag.fields.clear();
|
2022-11-09 13:00:51 +00:00
|
|
|
bool escaping = false;
|
2016-01-08 22:00:45 +00:00
|
|
|
|
2021-07-05 12:46:10 +00:00
|
|
|
if (p_stream->is_utf8()) {
|
|
|
|
CharString cs;
|
|
|
|
while (true) {
|
|
|
|
char c = p_stream->get_char();
|
|
|
|
if (p_stream->is_eof()) {
|
|
|
|
r_err_str = "Unexpected EOF while parsing simple tag";
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
|
|
|
if (c == ']') {
|
2022-11-09 13:00:51 +00:00
|
|
|
if (escaping) {
|
|
|
|
escaping = false;
|
|
|
|
} else {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
} else if (c == '\\') {
|
|
|
|
escaping = true;
|
|
|
|
} else {
|
|
|
|
escaping = false;
|
2021-07-05 12:46:10 +00:00
|
|
|
}
|
|
|
|
cs += c;
|
2016-01-08 22:00:45 +00:00
|
|
|
}
|
2021-07-05 12:46:10 +00:00
|
|
|
r_tag.name.parse_utf8(cs.get_data(), cs.length());
|
|
|
|
} else {
|
|
|
|
while (true) {
|
|
|
|
char32_t c = p_stream->get_char();
|
|
|
|
if (p_stream->is_eof()) {
|
|
|
|
r_err_str = "Unexpected EOF while parsing simple tag";
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
|
|
|
if (c == ']') {
|
2022-11-09 13:00:51 +00:00
|
|
|
if (escaping) {
|
|
|
|
escaping = false;
|
|
|
|
} else {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
} else if (c == '\\') {
|
|
|
|
escaping = true;
|
|
|
|
} else {
|
|
|
|
escaping = false;
|
2021-07-05 12:46:10 +00:00
|
|
|
}
|
|
|
|
r_tag.name += String::chr(c);
|
2020-05-14 14:41:43 +00:00
|
|
|
}
|
2016-01-08 22:00:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
r_tag.name = r_tag.name.strip_edges();
|
|
|
|
|
|
|
|
return OK;
|
|
|
|
}
|
|
|
|
|
2015-11-24 13:42:05 +00:00
|
|
|
get_token(p_stream, token, line, r_err_str);
|
|
|
|
|
|
|
|
if (token.type != TK_IDENTIFIER) {
|
|
|
|
r_err_str = "Expected identifier (tag name)";
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
r_tag.name = token.value;
|
2016-01-07 12:04:44 +00:00
|
|
|
bool parsing_tag = true;
|
2015-11-24 13:42:05 +00:00
|
|
|
|
|
|
|
while (true) {
|
2015-11-28 23:56:14 +00:00
|
|
|
if (p_stream->is_eof()) {
|
|
|
|
r_err_str = "Unexpected End of File while parsing tag: " + r_tag.name;
|
|
|
|
return ERR_FILE_CORRUPT;
|
|
|
|
}
|
|
|
|
|
2015-11-24 13:42:05 +00:00
|
|
|
get_token(p_stream, token, line, r_err_str);
|
2020-05-14 14:41:43 +00:00
|
|
|
if (token.type == TK_BRACKET_CLOSE) {
|
2015-11-24 13:42:05 +00:00
|
|
|
break;
|
2020-05-14 14:41:43 +00:00
|
|
|
}
|
2015-11-24 13:42:05 +00:00
|
|
|
|
2016-01-07 12:04:44 +00:00
|
|
|
if (parsing_tag && token.type == TK_PERIOD) {
|
2018-09-13 01:38:39 +00:00
|
|
|
r_tag.name += "."; //support tags such as [someprop.Android] for specific platforms
|
2016-01-07 12:04:44 +00:00
|
|
|
get_token(p_stream, token, line, r_err_str);
|
2016-01-07 12:16:11 +00:00
|
|
|
} else if (parsing_tag && token.type == TK_COLON) {
|
2018-09-13 01:38:39 +00:00
|
|
|
r_tag.name += ":"; //support tags such as [someprop.Android] for specific platforms
|
2016-01-07 12:16:11 +00:00
|
|
|
get_token(p_stream, token, line, r_err_str);
|
2016-01-07 12:04:44 +00:00
|
|
|
} else {
|
|
|
|
parsing_tag = false;
|
|
|
|
}
|
|
|
|
|
2015-11-24 13:42:05 +00:00
|
|
|
if (token.type != TK_IDENTIFIER) {
|
|
|
|
r_err_str = "Expected Identifier";
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
String id = token.value;
|
|
|
|
|
2016-01-07 12:04:44 +00:00
|
|
|
if (parsing_tag) {
|
|
|
|
r_tag.name += id;
|
|
|
|
continue;
|
|
|
|
}
|
2015-11-24 13:42:05 +00:00
|
|
|
|
|
|
|
get_token(p_stream, token, line, r_err_str);
|
|
|
|
if (token.type != TK_EQUAL) {
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
|
|
|
|
|
|
|
get_token(p_stream, token, line, r_err_str);
|
|
|
|
Variant value;
|
2015-11-28 23:56:14 +00:00
|
|
|
Error err = parse_value(token, value, p_stream, line, r_err_str, p_res_parser);
|
2020-05-14 14:41:43 +00:00
|
|
|
if (err) {
|
2015-11-24 13:42:05 +00:00
|
|
|
return err;
|
2020-05-14 14:41:43 +00:00
|
|
|
}
|
2015-11-24 13:42:05 +00:00
|
|
|
|
2016-03-08 23:00:52 +00:00
|
|
|
r_tag.fields[id] = value;
|
2015-11-24 13:42:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return OK;
|
|
|
|
}
|
|
|
|
|
2016-01-08 22:00:45 +00:00
|
|
|
Error VariantParser::parse_tag(Stream *p_stream, int &line, String &r_err_str, Tag &r_tag, ResourceParser *p_res_parser, bool p_simple_tag) {
|
2015-11-24 13:42:05 +00:00
|
|
|
Token token;
|
|
|
|
get_token(p_stream, token, line, r_err_str);
|
2015-11-28 23:56:14 +00:00
|
|
|
|
|
|
|
if (token.type == TK_EOF) {
|
|
|
|
return ERR_FILE_EOF;
|
|
|
|
}
|
|
|
|
|
2015-11-24 13:42:05 +00:00
|
|
|
if (token.type != TK_BRACKET_OPEN) {
|
|
|
|
r_err_str = "Expected '['";
|
|
|
|
return ERR_PARSE_ERROR;
|
|
|
|
}
|
|
|
|
|
2016-01-08 22:00:45 +00:00
|
|
|
return _parse_tag(token, p_stream, line, r_err_str, r_tag, p_res_parser, p_simple_tag);
|
2015-11-24 13:42:05 +00:00
|
|
|
}
|
|
|
|
|
2016-01-08 22:00:45 +00:00
|
|
|
Error VariantParser::parse_tag_assign_eof(Stream *p_stream, int &line, String &r_err_str, Tag &r_tag, String &r_assign, Variant &r_value, ResourceParser *p_res_parser, bool p_simple_tag) {
|
2015-11-28 23:56:14 +00:00
|
|
|
//assign..
|
2016-01-06 08:39:05 +00:00
|
|
|
r_assign = "";
|
2015-11-28 23:56:14 +00:00
|
|
|
String what;
|
|
|
|
|
|
|
|
while (true) {
|
2020-07-27 10:43:20 +00:00
|
|
|
char32_t c;
|
2015-11-28 23:56:14 +00:00
|
|
|
if (p_stream->saved) {
|
|
|
|
c = p_stream->saved;
|
|
|
|
p_stream->saved = 0;
|
|
|
|
|
|
|
|
} else {
|
|
|
|
c = p_stream->get_char();
|
|
|
|
}
|
|
|
|
|
2020-05-14 14:41:43 +00:00
|
|
|
if (p_stream->is_eof()) {
|
2015-11-28 23:56:14 +00:00
|
|
|
return ERR_FILE_EOF;
|
2020-05-14 14:41:43 +00:00
|
|
|
}
|
2015-11-28 23:56:14 +00:00
|
|
|
|
2015-12-31 13:25:21 +00:00
|
|
|
if (c == ';') { //comment
|
|
|
|
while (true) {
|
2020-07-27 10:43:20 +00:00
|
|
|
char32_t ch = p_stream->get_char();
|
2015-12-31 13:25:21 +00:00
|
|
|
if (p_stream->is_eof()) {
|
|
|
|
return ERR_FILE_EOF;
|
|
|
|
}
|
2020-05-14 14:41:43 +00:00
|
|
|
if (ch == '\n') {
|
2021-05-23 07:37:53 +00:00
|
|
|
line++;
|
2015-12-31 13:25:21 +00:00
|
|
|
break;
|
2020-05-14 14:41:43 +00:00
|
|
|
}
|
2015-12-31 13:25:21 +00:00
|
|
|
}
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2015-11-28 23:56:14 +00:00
|
|
|
if (c == '[' && what.length() == 0) {
|
|
|
|
//it's a tag!
|
|
|
|
p_stream->saved = '['; //go back one
|
|
|
|
|
2016-01-08 22:00:45 +00:00
|
|
|
Error err = parse_tag(p_stream, line, r_err_str, r_tag, p_res_parser, p_simple_tag);
|
2015-11-28 23:56:14 +00:00
|
|
|
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (c > 32) {
|
2016-06-20 21:41:59 +00:00
|
|
|
if (c == '"') { //quoted
|
|
|
|
p_stream->saved = '"';
|
|
|
|
Token tk;
|
|
|
|
Error err = get_token(p_stream, tk, line, r_err_str);
|
2020-05-14 14:41:43 +00:00
|
|
|
if (err) {
|
2016-06-20 21:41:59 +00:00
|
|
|
return err;
|
2020-05-14 14:41:43 +00:00
|
|
|
}
|
2016-06-20 21:41:59 +00:00
|
|
|
if (tk.type != TK_STRING) {
|
|
|
|
r_err_str = "Error reading quoted string";
|
2019-09-22 16:45:08 +00:00
|
|
|
return ERR_INVALID_DATA;
|
2016-06-20 21:41:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
what = tk.value;
|
|
|
|
|
|
|
|
} else if (c != '=') {
|
2015-11-28 23:56:14 +00:00
|
|
|
what += String::chr(c);
|
|
|
|
} else {
|
|
|
|
r_assign = what;
|
|
|
|
Token token;
|
|
|
|
get_token(p_stream, token, line, r_err_str);
|
|
|
|
Error err = parse_value(token, r_value, p_stream, line, r_err_str, p_res_parser);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
} else if (c == '\n') {
|
|
|
|
line++;
|
|
|
|
}
|
|
|
|
}
|
2015-11-24 13:42:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Error VariantParser::parse(Stream *p_stream, Variant &r_ret, String &r_err_str, int &r_err_line, ResourceParser *p_res_parser) {
|
|
|
|
Token token;
|
|
|
|
Error err = get_token(p_stream, token, r_err_line, r_err_str);
|
2020-05-14 14:41:43 +00:00
|
|
|
if (err) {
|
2015-11-24 13:42:05 +00:00
|
|
|
return err;
|
2020-05-14 14:41:43 +00:00
|
|
|
}
|
2015-11-28 23:56:14 +00:00
|
|
|
|
|
|
|
if (token.type == TK_EOF) {
|
|
|
|
return ERR_FILE_EOF;
|
|
|
|
}
|
|
|
|
|
2015-11-24 13:42:05 +00:00
|
|
|
return parse_value(token, r_ret, p_stream, r_err_line, r_err_str, p_res_parser);
|
|
|
|
}
|
|
|
|
|
2015-12-31 03:31:00 +00:00
|
|
|
//////////////////////////////////////////////////////////////////////////////////
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////
|
2020-07-23 04:42:19 +00:00
|
|
|
//////////////////////////////////////////////////////////////////////////////////
|
2015-12-31 03:31:00 +00:00
|
|
|
|
2020-07-23 04:42:19 +00:00
|
|
|
static String rtos_fix(double p_value) {
|
2020-05-14 14:41:43 +00:00
|
|
|
if (p_value == 0.0) {
|
2016-06-13 13:58:32 +00:00
|
|
|
return "0"; //avoid negative zero (-0) being written, which may annoy git, svn, etc. for changes when they don't exist.
|
2020-07-23 04:42:19 +00:00
|
|
|
} else if (isnan(p_value)) {
|
|
|
|
return "nan";
|
|
|
|
} else if (isinf(p_value)) {
|
|
|
|
if (p_value > 0) {
|
|
|
|
return "inf";
|
|
|
|
} else {
|
|
|
|
return "inf_neg";
|
|
|
|
}
|
2020-05-14 14:41:43 +00:00
|
|
|
} else {
|
2016-06-13 13:16:43 +00:00
|
|
|
return rtoss(p_value);
|
2020-05-14 14:41:43 +00:00
|
|
|
}
|
2016-06-13 13:16:43 +00:00
|
|
|
}
|
2015-12-31 03:31:00 +00:00
|
|
|
|
2024-04-19 07:55:29 +00:00
|
|
|
Error VariantWriter::write(const Variant &p_variant, StoreStringFunc p_store_string_func, void *p_store_string_ud, EncodeResourceFunc p_encode_res_func, void *p_encode_res_ud, int p_recursion_count, bool p_compat) {
|
2015-12-31 03:31:00 +00:00
|
|
|
switch (p_variant.get_type()) {
|
|
|
|
case Variant::NIL: {
|
|
|
|
p_store_string_func(p_store_string_ud, "null");
|
|
|
|
} break;
|
|
|
|
case Variant::BOOL: {
|
|
|
|
p_store_string_func(p_store_string_ud, p_variant.operator bool() ? "true" : "false");
|
|
|
|
} break;
|
|
|
|
case Variant::INT: {
|
2019-02-21 10:21:41 +00:00
|
|
|
p_store_string_func(p_store_string_ud, itos(p_variant.operator int64_t()));
|
2015-12-31 03:31:00 +00:00
|
|
|
} break;
|
Variant: Added 64-bit packed arrays, renamed Variant::REAL to FLOAT.
- Renames PackedIntArray to PackedInt32Array.
- Renames PackedFloatArray to PackedFloat32Array.
- Adds PackedInt64Array and PackedFloat64Array.
- Renames Variant::REAL to Variant::FLOAT for consistency.
Packed arrays are for storing large amount of data and creating stuff like
meshes, buffers. textures, etc. Forcing them to be 64 is a huge waste of
memory. That said, many users requested the ability to have 64 bits packed
arrays for their games, so this is just an optional added type.
For Variant, the float datatype is always 64 bits, and exposed as `float`.
We still have `real_t` which is the datatype that can change from 32 to 64
bits depending on a compile flag (not entirely working right now, but that's
the idea). It affects math related datatypes and code only.
Neither Variant nor PackedArray make use of real_t, which is only intended
for math precision, so the term is removed from there to keep only float.
2020-02-24 18:20:53 +00:00
|
|
|
case Variant::FLOAT: {
|
2020-07-23 04:42:19 +00:00
|
|
|
String s = rtos_fix(p_variant.operator double());
|
|
|
|
if (s != "inf" && s != "inf_neg" && s != "nan") {
|
2022-02-03 16:03:38 +00:00
|
|
|
if (!s.contains(".") && !s.contains("e")) {
|
2020-02-09 05:46:13 +00:00
|
|
|
s += ".0";
|
2020-05-14 14:41:43 +00:00
|
|
|
}
|
2020-02-09 05:46:13 +00:00
|
|
|
}
|
2016-01-10 14:51:05 +00:00
|
|
|
p_store_string_func(p_store_string_ud, s);
|
2015-12-31 03:31:00 +00:00
|
|
|
} break;
|
|
|
|
case Variant::STRING: {
|
|
|
|
String str = p_variant;
|
2017-01-16 17:03:38 +00:00
|
|
|
str = "\"" + str.c_escape_multiline() + "\"";
|
2015-12-31 03:31:00 +00:00
|
|
|
p_store_string_func(p_store_string_ud, str);
|
|
|
|
} break;
|
2023-08-29 12:05:53 +00:00
|
|
|
|
|
|
|
// Math types.
|
2015-12-31 03:31:00 +00:00
|
|
|
case Variant::VECTOR2: {
|
|
|
|
Vector2 v = p_variant;
|
2020-07-23 04:42:19 +00:00
|
|
|
p_store_string_func(p_store_string_ud, "Vector2(" + rtos_fix(v.x) + ", " + rtos_fix(v.y) + ")");
|
2015-12-31 03:31:00 +00:00
|
|
|
} break;
|
2020-02-22 03:26:41 +00:00
|
|
|
case Variant::VECTOR2I: {
|
|
|
|
Vector2i v = p_variant;
|
2019-09-24 17:45:03 +00:00
|
|
|
p_store_string_func(p_store_string_ud, "Vector2i(" + itos(v.x) + ", " + itos(v.y) + ")");
|
2020-02-22 03:26:41 +00:00
|
|
|
} break;
|
2015-12-31 03:31:00 +00:00
|
|
|
case Variant::RECT2: {
|
|
|
|
Rect2 aabb = p_variant;
|
2020-07-23 04:42:19 +00:00
|
|
|
p_store_string_func(p_store_string_ud, "Rect2(" + rtos_fix(aabb.position.x) + ", " + rtos_fix(aabb.position.y) + ", " + rtos_fix(aabb.size.x) + ", " + rtos_fix(aabb.size.y) + ")");
|
2020-02-22 03:26:41 +00:00
|
|
|
} break;
|
|
|
|
case Variant::RECT2I: {
|
|
|
|
Rect2i aabb = p_variant;
|
2019-09-24 17:45:03 +00:00
|
|
|
p_store_string_func(p_store_string_ud, "Rect2i(" + itos(aabb.position.x) + ", " + itos(aabb.position.y) + ", " + itos(aabb.size.x) + ", " + itos(aabb.size.y) + ")");
|
2015-12-31 03:31:00 +00:00
|
|
|
} break;
|
|
|
|
case Variant::VECTOR3: {
|
|
|
|
Vector3 v = p_variant;
|
2020-07-23 04:42:19 +00:00
|
|
|
p_store_string_func(p_store_string_ud, "Vector3(" + rtos_fix(v.x) + ", " + rtos_fix(v.y) + ", " + rtos_fix(v.z) + ")");
|
2015-12-31 03:31:00 +00:00
|
|
|
} break;
|
2020-02-22 03:26:41 +00:00
|
|
|
case Variant::VECTOR3I: {
|
|
|
|
Vector3i v = p_variant;
|
2019-09-24 17:45:03 +00:00
|
|
|
p_store_string_func(p_store_string_ud, "Vector3i(" + itos(v.x) + ", " + itos(v.y) + ", " + itos(v.z) + ")");
|
2020-02-22 03:26:41 +00:00
|
|
|
} break;
|
Implement Vector4, Vector4i, Projection
Implement built-in classes Vector4, Vector4i and Projection.
* Two versions of Vector4 (float and integer).
* A Projection class, which is a 4x4 matrix specialized in projection types.
These types have been requested for a long time, but given they were very corner case they were not added before.
Because in Godot 4, reimplementing parts of the rendering engine is now possible, access to these types (heavily used by the rendering code) becomes a necessity.
**Q**: Why Projection and not Matrix4?
**A**: Godot does not use Matrix2, Matrix3, Matrix4x3, etc. naming convention because, within the engine, these types always have a *purpose*. As such, Godot names them: Transform2D, Transform3D or Basis. In this case, this 4x4 matrix is _always_ used as a _Projection_, hence the naming.
2022-07-19 23:11:13 +00:00
|
|
|
case Variant::VECTOR4: {
|
|
|
|
Vector4 v = p_variant;
|
2022-07-29 09:57:44 +00:00
|
|
|
p_store_string_func(p_store_string_ud, "Vector4(" + rtos_fix(v.x) + ", " + rtos_fix(v.y) + ", " + rtos_fix(v.z) + ", " + rtos_fix(v.w) + ")");
|
Implement Vector4, Vector4i, Projection
Implement built-in classes Vector4, Vector4i and Projection.
* Two versions of Vector4 (float and integer).
* A Projection class, which is a 4x4 matrix specialized in projection types.
These types have been requested for a long time, but given they were very corner case they were not added before.
Because in Godot 4, reimplementing parts of the rendering engine is now possible, access to these types (heavily used by the rendering code) becomes a necessity.
**Q**: Why Projection and not Matrix4?
**A**: Godot does not use Matrix2, Matrix3, Matrix4x3, etc. naming convention because, within the engine, these types always have a *purpose*. As such, Godot names them: Transform2D, Transform3D or Basis. In this case, this 4x4 matrix is _always_ used as a _Projection_, hence the naming.
2022-07-19 23:11:13 +00:00
|
|
|
} break;
|
|
|
|
case Variant::VECTOR4I: {
|
|
|
|
Vector4i v = p_variant;
|
2022-07-29 09:57:44 +00:00
|
|
|
p_store_string_func(p_store_string_ud, "Vector4i(" + itos(v.x) + ", " + itos(v.y) + ", " + itos(v.z) + ", " + itos(v.w) + ")");
|
Implement Vector4, Vector4i, Projection
Implement built-in classes Vector4, Vector4i and Projection.
* Two versions of Vector4 (float and integer).
* A Projection class, which is a 4x4 matrix specialized in projection types.
These types have been requested for a long time, but given they were very corner case they were not added before.
Because in Godot 4, reimplementing parts of the rendering engine is now possible, access to these types (heavily used by the rendering code) becomes a necessity.
**Q**: Why Projection and not Matrix4?
**A**: Godot does not use Matrix2, Matrix3, Matrix4x3, etc. naming convention because, within the engine, these types always have a *purpose*. As such, Godot names them: Transform2D, Transform3D or Basis. In this case, this 4x4 matrix is _always_ used as a _Projection_, hence the naming.
2022-07-19 23:11:13 +00:00
|
|
|
} break;
|
2015-12-31 03:31:00 +00:00
|
|
|
case Variant::PLANE: {
|
|
|
|
Plane p = p_variant;
|
2020-07-23 04:42:19 +00:00
|
|
|
p_store_string_func(p_store_string_ud, "Plane(" + rtos_fix(p.normal.x) + ", " + rtos_fix(p.normal.y) + ", " + rtos_fix(p.normal.z) + ", " + rtos_fix(p.d) + ")");
|
2015-12-31 03:31:00 +00:00
|
|
|
} break;
|
2017-11-17 02:09:00 +00:00
|
|
|
case Variant::AABB: {
|
|
|
|
AABB aabb = p_variant;
|
2020-07-23 04:42:19 +00:00
|
|
|
p_store_string_func(p_store_string_ud, "AABB(" + rtos_fix(aabb.position.x) + ", " + rtos_fix(aabb.position.y) + ", " + rtos_fix(aabb.position.z) + ", " + rtos_fix(aabb.size.x) + ", " + rtos_fix(aabb.size.y) + ", " + rtos_fix(aabb.size.z) + ")");
|
2015-12-31 03:31:00 +00:00
|
|
|
} break;
|
2021-01-20 07:02:02 +00:00
|
|
|
case Variant::QUATERNION: {
|
|
|
|
Quaternion quaternion = p_variant;
|
2020-07-23 04:42:19 +00:00
|
|
|
p_store_string_func(p_store_string_ud, "Quaternion(" + rtos_fix(quaternion.x) + ", " + rtos_fix(quaternion.y) + ", " + rtos_fix(quaternion.z) + ", " + rtos_fix(quaternion.w) + ")");
|
2015-12-31 03:31:00 +00:00
|
|
|
} break;
|
2017-01-11 03:52:51 +00:00
|
|
|
case Variant::TRANSFORM2D: {
|
2019-09-24 17:45:03 +00:00
|
|
|
String s = "Transform2D(";
|
2017-01-11 03:52:51 +00:00
|
|
|
Transform2D m3 = p_variant;
|
2015-12-31 03:31:00 +00:00
|
|
|
for (int i = 0; i < 3; i++) {
|
|
|
|
for (int j = 0; j < 2; j++) {
|
2020-05-14 14:41:43 +00:00
|
|
|
if (i != 0 || j != 0) {
|
2015-12-31 03:31:00 +00:00
|
|
|
s += ", ";
|
2020-05-14 14:41:43 +00:00
|
|
|
}
|
2022-04-24 21:59:24 +00:00
|
|
|
s += rtos_fix(m3.columns[i][j]);
|
2015-12-31 03:31:00 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-09-24 17:45:03 +00:00
|
|
|
p_store_string_func(p_store_string_ud, s + ")");
|
2015-12-31 03:31:00 +00:00
|
|
|
} break;
|
2017-01-11 03:52:51 +00:00
|
|
|
case Variant::BASIS: {
|
2019-09-24 17:45:03 +00:00
|
|
|
String s = "Basis(";
|
2017-01-11 03:52:51 +00:00
|
|
|
Basis m3 = p_variant;
|
2015-12-31 03:31:00 +00:00
|
|
|
for (int i = 0; i < 3; i++) {
|
|
|
|
for (int j = 0; j < 3; j++) {
|
2020-05-14 14:41:43 +00:00
|
|
|
if (i != 0 || j != 0) {
|
2015-12-31 03:31:00 +00:00
|
|
|
s += ", ";
|
2020-05-14 14:41:43 +00:00
|
|
|
}
|
2022-04-24 22:07:35 +00:00
|
|
|
s += rtos_fix(m3.rows[i][j]);
|
2015-12-31 03:31:00 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-09-24 17:45:03 +00:00
|
|
|
p_store_string_func(p_store_string_ud, s + ")");
|
2015-12-31 03:31:00 +00:00
|
|
|
} break;
|
2021-04-28 07:36:08 +00:00
|
|
|
case Variant::TRANSFORM3D: {
|
2019-09-24 17:45:03 +00:00
|
|
|
String s = "Transform3D(";
|
2020-10-17 05:08:21 +00:00
|
|
|
Transform3D t = p_variant;
|
2017-01-11 03:52:51 +00:00
|
|
|
Basis &m3 = t.basis;
|
2015-12-31 03:31:00 +00:00
|
|
|
for (int i = 0; i < 3; i++) {
|
|
|
|
for (int j = 0; j < 3; j++) {
|
2020-05-14 14:41:43 +00:00
|
|
|
if (i != 0 || j != 0) {
|
2015-12-31 03:31:00 +00:00
|
|
|
s += ", ";
|
2020-05-14 14:41:43 +00:00
|
|
|
}
|
2022-04-24 22:07:35 +00:00
|
|
|
s += rtos_fix(m3.rows[i][j]);
|
2015-12-31 03:31:00 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-07-23 04:42:19 +00:00
|
|
|
s = s + ", " + rtos_fix(t.origin.x) + ", " + rtos_fix(t.origin.y) + ", " + rtos_fix(t.origin.z);
|
2015-12-31 03:31:00 +00:00
|
|
|
|
2019-09-24 17:45:03 +00:00
|
|
|
p_store_string_func(p_store_string_ud, s + ")");
|
2015-12-31 03:31:00 +00:00
|
|
|
} break;
|
Implement Vector4, Vector4i, Projection
Implement built-in classes Vector4, Vector4i and Projection.
* Two versions of Vector4 (float and integer).
* A Projection class, which is a 4x4 matrix specialized in projection types.
These types have been requested for a long time, but given they were very corner case they were not added before.
Because in Godot 4, reimplementing parts of the rendering engine is now possible, access to these types (heavily used by the rendering code) becomes a necessity.
**Q**: Why Projection and not Matrix4?
**A**: Godot does not use Matrix2, Matrix3, Matrix4x3, etc. naming convention because, within the engine, these types always have a *purpose*. As such, Godot names them: Transform2D, Transform3D or Basis. In this case, this 4x4 matrix is _always_ used as a _Projection_, hence the naming.
2022-07-19 23:11:13 +00:00
|
|
|
case Variant::PROJECTION: {
|
|
|
|
String s = "Projection(";
|
|
|
|
Projection t = p_variant;
|
|
|
|
for (int i = 0; i < 4; i++) {
|
|
|
|
for (int j = 0; j < 4; j++) {
|
|
|
|
if (i != 0 || j != 0) {
|
|
|
|
s += ", ";
|
|
|
|
}
|
2022-10-04 16:44:48 +00:00
|
|
|
s += rtos_fix(t.columns[i][j]);
|
Implement Vector4, Vector4i, Projection
Implement built-in classes Vector4, Vector4i and Projection.
* Two versions of Vector4 (float and integer).
* A Projection class, which is a 4x4 matrix specialized in projection types.
These types have been requested for a long time, but given they were very corner case they were not added before.
Because in Godot 4, reimplementing parts of the rendering engine is now possible, access to these types (heavily used by the rendering code) becomes a necessity.
**Q**: Why Projection and not Matrix4?
**A**: Godot does not use Matrix2, Matrix3, Matrix4x3, etc. naming convention because, within the engine, these types always have a *purpose*. As such, Godot names them: Transform2D, Transform3D or Basis. In this case, this 4x4 matrix is _always_ used as a _Projection_, hence the naming.
2022-07-19 23:11:13 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
p_store_string_func(p_store_string_ud, s + ")");
|
|
|
|
} break;
|
2015-12-31 03:31:00 +00:00
|
|
|
|
2023-08-29 12:05:53 +00:00
|
|
|
// Misc types.
|
2015-12-31 03:31:00 +00:00
|
|
|
case Variant::COLOR: {
|
|
|
|
Color c = p_variant;
|
2020-07-23 04:42:19 +00:00
|
|
|
p_store_string_func(p_store_string_ud, "Color(" + rtos_fix(c.r) + ", " + rtos_fix(c.g) + ", " + rtos_fix(c.b) + ", " + rtos_fix(c.a) + ")");
|
2020-02-20 21:58:05 +00:00
|
|
|
} break;
|
|
|
|
case Variant::STRING_NAME: {
|
|
|
|
String str = p_variant;
|
2021-06-05 19:24:24 +00:00
|
|
|
str = "&\"" + str.c_escape() + "\"";
|
2020-02-20 21:58:05 +00:00
|
|
|
p_store_string_func(p_store_string_ud, str);
|
2015-12-31 03:31:00 +00:00
|
|
|
} break;
|
|
|
|
case Variant::NODE_PATH: {
|
|
|
|
String str = p_variant;
|
|
|
|
str = "NodePath(\"" + str.c_escape() + "\")";
|
|
|
|
p_store_string_func(p_store_string_ud, str);
|
|
|
|
} break;
|
2023-06-21 09:06:32 +00:00
|
|
|
case Variant::RID: {
|
|
|
|
RID rid = p_variant;
|
|
|
|
if (rid == RID()) {
|
|
|
|
p_store_string_func(p_store_string_ud, "RID()");
|
|
|
|
} else {
|
|
|
|
p_store_string_func(p_store_string_ud, "RID(" + itos(rid.get_id()) + ")");
|
|
|
|
}
|
|
|
|
} break;
|
|
|
|
|
|
|
|
// Do not really store these, but ensure that assignments are not empty.
|
|
|
|
case Variant::SIGNAL: {
|
|
|
|
p_store_string_func(p_store_string_ud, "Signal()");
|
|
|
|
} break;
|
|
|
|
case Variant::CALLABLE: {
|
|
|
|
p_store_string_func(p_store_string_ud, "Callable()");
|
|
|
|
} break;
|
|
|
|
|
2015-12-31 03:31:00 +00:00
|
|
|
case Variant::OBJECT: {
|
2023-08-29 12:05:53 +00:00
|
|
|
if (unlikely(p_recursion_count > MAX_RECURSION)) {
|
|
|
|
ERR_PRINT("Max recursion reached");
|
|
|
|
p_store_string_func(p_store_string_ud, "null");
|
|
|
|
return OK;
|
|
|
|
}
|
|
|
|
p_recursion_count++;
|
|
|
|
|
2022-09-04 11:56:37 +00:00
|
|
|
Object *obj = p_variant.get_validated_object();
|
2017-05-29 00:46:48 +00:00
|
|
|
|
|
|
|
if (!obj) {
|
2015-12-31 03:31:00 +00:00
|
|
|
p_store_string_func(p_store_string_ud, "null");
|
|
|
|
break; // don't save it
|
|
|
|
}
|
|
|
|
|
2022-05-02 23:43:50 +00:00
|
|
|
Ref<Resource> res = p_variant;
|
2017-05-29 00:46:48 +00:00
|
|
|
if (res.is_valid()) {
|
|
|
|
//is resource
|
|
|
|
String res_text;
|
2015-12-31 03:31:00 +00:00
|
|
|
|
2017-05-29 00:46:48 +00:00
|
|
|
//try external function
|
|
|
|
if (p_encode_res_func) {
|
|
|
|
res_text = p_encode_res_func(p_encode_res_ud, res);
|
|
|
|
}
|
|
|
|
|
|
|
|
//try path because it's a file
|
2021-12-09 09:42:46 +00:00
|
|
|
if (res_text.is_empty() && res->get_path().is_resource_file()) {
|
2017-05-29 00:46:48 +00:00
|
|
|
//external resource
|
|
|
|
String path = res->get_path();
|
2019-09-24 17:45:03 +00:00
|
|
|
res_text = "Resource(\"" + path + "\")";
|
2017-05-29 00:46:48 +00:00
|
|
|
}
|
2015-12-31 03:31:00 +00:00
|
|
|
|
2017-05-29 00:46:48 +00:00
|
|
|
//could come up with some sort of text
|
2021-12-09 09:42:46 +00:00
|
|
|
if (!res_text.is_empty()) {
|
2017-05-29 00:46:48 +00:00
|
|
|
p_store_string_func(p_store_string_ud, res_text);
|
|
|
|
break;
|
|
|
|
}
|
2015-12-31 03:31:00 +00:00
|
|
|
}
|
|
|
|
|
2017-05-29 00:46:48 +00:00
|
|
|
//store as generic object
|
|
|
|
|
|
|
|
p_store_string_func(p_store_string_ud, "Object(" + obj->get_class() + ",");
|
|
|
|
|
|
|
|
List<PropertyInfo> props;
|
|
|
|
obj->get_property_list(&props);
|
|
|
|
bool first = true;
|
2021-07-24 13:46:25 +00:00
|
|
|
for (const PropertyInfo &E : props) {
|
2021-07-16 03:45:57 +00:00
|
|
|
if (E.usage & PROPERTY_USAGE_STORAGE || E.usage & PROPERTY_USAGE_SCRIPT_VARIABLE) {
|
2017-05-29 00:46:48 +00:00
|
|
|
//must be serialized
|
|
|
|
|
|
|
|
if (first) {
|
|
|
|
first = false;
|
|
|
|
} else {
|
|
|
|
p_store_string_func(p_store_string_ud, ",");
|
|
|
|
}
|
|
|
|
|
2021-07-16 03:45:57 +00:00
|
|
|
p_store_string_func(p_store_string_ud, "\"" + E.name + "\":");
|
2024-04-19 07:55:29 +00:00
|
|
|
write(obj->get(E.name), p_store_string_func, p_store_string_ud, p_encode_res_func, p_encode_res_ud, p_recursion_count, p_compat);
|
2017-05-29 00:46:48 +00:00
|
|
|
}
|
|
|
|
}
|
2015-12-31 03:31:00 +00:00
|
|
|
|
2017-05-29 00:46:48 +00:00
|
|
|
p_store_string_func(p_store_string_ud, ")\n");
|
2015-12-31 03:31:00 +00:00
|
|
|
} break;
|
2016-06-05 00:31:29 +00:00
|
|
|
|
2015-12-31 03:31:00 +00:00
|
|
|
case Variant::DICTIONARY: {
|
|
|
|
Dictionary dict = p_variant;
|
2023-06-24 18:03:28 +00:00
|
|
|
|
|
|
|
if (dict.is_typed()) {
|
|
|
|
p_store_string_func(p_store_string_ud, "Dictionary[");
|
|
|
|
|
|
|
|
Variant::Type key_builtin_type = (Variant::Type)dict.get_typed_key_builtin();
|
|
|
|
StringName key_class_name = dict.get_typed_key_class_name();
|
|
|
|
Ref<Script> key_script = dict.get_typed_key_script();
|
|
|
|
|
|
|
|
if (key_script.is_valid()) {
|
|
|
|
String resource_text;
|
|
|
|
if (p_encode_res_func) {
|
|
|
|
resource_text = p_encode_res_func(p_encode_res_ud, key_script);
|
|
|
|
}
|
|
|
|
if (resource_text.is_empty() && key_script->get_path().is_resource_file()) {
|
|
|
|
resource_text = "Resource(\"" + key_script->get_path() + "\")";
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!resource_text.is_empty()) {
|
|
|
|
p_store_string_func(p_store_string_ud, resource_text);
|
|
|
|
} else {
|
|
|
|
ERR_PRINT("Failed to encode a path to a custom script for a dictionary key type.");
|
|
|
|
p_store_string_func(p_store_string_ud, key_class_name);
|
|
|
|
}
|
|
|
|
} else if (key_class_name != StringName()) {
|
|
|
|
p_store_string_func(p_store_string_ud, key_class_name);
|
|
|
|
} else if (key_builtin_type == Variant::NIL) {
|
|
|
|
p_store_string_func(p_store_string_ud, "Variant");
|
|
|
|
} else {
|
|
|
|
p_store_string_func(p_store_string_ud, Variant::get_type_name(key_builtin_type));
|
|
|
|
}
|
|
|
|
|
|
|
|
p_store_string_func(p_store_string_ud, ", ");
|
|
|
|
|
|
|
|
Variant::Type value_builtin_type = (Variant::Type)dict.get_typed_value_builtin();
|
|
|
|
StringName value_class_name = dict.get_typed_value_class_name();
|
|
|
|
Ref<Script> value_script = dict.get_typed_value_script();
|
|
|
|
|
|
|
|
if (value_script.is_valid()) {
|
|
|
|
String resource_text;
|
|
|
|
if (p_encode_res_func) {
|
|
|
|
resource_text = p_encode_res_func(p_encode_res_ud, value_script);
|
|
|
|
}
|
|
|
|
if (resource_text.is_empty() && value_script->get_path().is_resource_file()) {
|
|
|
|
resource_text = "Resource(\"" + value_script->get_path() + "\")";
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!resource_text.is_empty()) {
|
|
|
|
p_store_string_func(p_store_string_ud, resource_text);
|
|
|
|
} else {
|
|
|
|
ERR_PRINT("Failed to encode a path to a custom script for a dictionary value type.");
|
|
|
|
p_store_string_func(p_store_string_ud, value_class_name);
|
|
|
|
}
|
|
|
|
} else if (value_class_name != StringName()) {
|
|
|
|
p_store_string_func(p_store_string_ud, value_class_name);
|
|
|
|
} else if (value_builtin_type == Variant::NIL) {
|
|
|
|
p_store_string_func(p_store_string_ud, "Variant");
|
|
|
|
} else {
|
|
|
|
p_store_string_func(p_store_string_ud, Variant::get_type_name(value_builtin_type));
|
|
|
|
}
|
|
|
|
|
|
|
|
p_store_string_func(p_store_string_ud, "](");
|
|
|
|
}
|
|
|
|
|
2023-08-29 12:05:53 +00:00
|
|
|
if (unlikely(p_recursion_count > MAX_RECURSION)) {
|
2020-02-01 06:04:14 +00:00
|
|
|
ERR_PRINT("Max recursion reached");
|
|
|
|
p_store_string_func(p_store_string_ud, "{}");
|
|
|
|
} else {
|
|
|
|
List<Variant> keys;
|
|
|
|
dict.get_key_list(&keys);
|
|
|
|
keys.sort();
|
|
|
|
|
2023-06-24 18:03:28 +00:00
|
|
|
if (keys.is_empty()) {
|
|
|
|
// Avoid unnecessary line break.
|
2022-01-18 10:31:15 +00:00
|
|
|
p_store_string_func(p_store_string_ud, "{}");
|
2023-06-24 18:03:28 +00:00
|
|
|
} else {
|
|
|
|
p_recursion_count++;
|
2022-01-18 10:31:15 +00:00
|
|
|
|
2023-06-24 18:03:28 +00:00
|
|
|
p_store_string_func(p_store_string_ud, "{\n");
|
|
|
|
|
|
|
|
for (List<Variant>::Element *E = keys.front(); E; E = E->next()) {
|
|
|
|
write(E->get(), p_store_string_func, p_store_string_ud, p_encode_res_func, p_encode_res_ud, p_recursion_count, p_compat);
|
|
|
|
p_store_string_func(p_store_string_ud, ": ");
|
|
|
|
write(dict[E->get()], p_store_string_func, p_store_string_ud, p_encode_res_func, p_encode_res_ud, p_recursion_count, p_compat);
|
|
|
|
if (E->next()) {
|
|
|
|
p_store_string_func(p_store_string_ud, ",\n");
|
|
|
|
} else {
|
|
|
|
p_store_string_func(p_store_string_ud, "\n");
|
|
|
|
}
|
2020-02-01 06:04:14 +00:00
|
|
|
}
|
2023-06-24 18:03:28 +00:00
|
|
|
|
|
|
|
p_store_string_func(p_store_string_ud, "}");
|
2020-05-14 14:41:43 +00:00
|
|
|
}
|
2023-06-24 18:03:28 +00:00
|
|
|
}
|
2015-12-31 03:31:00 +00:00
|
|
|
|
2023-06-24 18:03:28 +00:00
|
|
|
if (dict.is_typed()) {
|
|
|
|
p_store_string_func(p_store_string_ud, ")");
|
2020-02-01 06:04:14 +00:00
|
|
|
}
|
2015-12-31 03:31:00 +00:00
|
|
|
} break;
|
2020-02-01 06:04:14 +00:00
|
|
|
|
2015-12-31 03:31:00 +00:00
|
|
|
case Variant::ARRAY: {
|
2022-11-27 07:56:53 +00:00
|
|
|
Array array = p_variant;
|
2023-06-24 18:03:28 +00:00
|
|
|
|
|
|
|
if (array.is_typed()) {
|
2022-11-27 07:56:53 +00:00
|
|
|
p_store_string_func(p_store_string_ud, "Array[");
|
|
|
|
|
|
|
|
Variant::Type builtin_type = (Variant::Type)array.get_typed_builtin();
|
|
|
|
StringName class_name = array.get_typed_class_name();
|
|
|
|
Ref<Script> script = array.get_typed_script();
|
|
|
|
|
|
|
|
if (script.is_valid()) {
|
|
|
|
String resource_text = String();
|
|
|
|
if (p_encode_res_func) {
|
|
|
|
resource_text = p_encode_res_func(p_encode_res_ud, script);
|
|
|
|
}
|
|
|
|
if (resource_text.is_empty() && script->get_path().is_resource_file()) {
|
|
|
|
resource_text = "Resource(\"" + script->get_path() + "\")";
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!resource_text.is_empty()) {
|
|
|
|
p_store_string_func(p_store_string_ud, resource_text);
|
|
|
|
} else {
|
|
|
|
ERR_PRINT("Failed to encode a path to a custom script for an array type.");
|
|
|
|
p_store_string_func(p_store_string_ud, class_name);
|
|
|
|
}
|
|
|
|
} else if (class_name != StringName()) {
|
|
|
|
p_store_string_func(p_store_string_ud, class_name);
|
|
|
|
} else {
|
|
|
|
p_store_string_func(p_store_string_ud, Variant::get_type_name(builtin_type));
|
|
|
|
}
|
|
|
|
|
|
|
|
p_store_string_func(p_store_string_ud, "](");
|
|
|
|
}
|
|
|
|
|
2023-08-29 12:05:53 +00:00
|
|
|
if (unlikely(p_recursion_count > MAX_RECURSION)) {
|
2020-02-01 06:04:14 +00:00
|
|
|
ERR_PRINT("Max recursion reached");
|
|
|
|
p_store_string_func(p_store_string_ud, "[]");
|
|
|
|
} else {
|
2023-08-29 12:05:53 +00:00
|
|
|
p_recursion_count++;
|
2020-02-01 06:04:14 +00:00
|
|
|
|
|
|
|
p_store_string_func(p_store_string_ud, "[");
|
2023-06-24 18:03:28 +00:00
|
|
|
|
2023-12-24 12:44:21 +00:00
|
|
|
bool first = true;
|
|
|
|
for (const Variant &var : array) {
|
|
|
|
if (first) {
|
|
|
|
first = false;
|
|
|
|
} else {
|
2020-02-01 06:04:14 +00:00
|
|
|
p_store_string_func(p_store_string_ud, ", ");
|
|
|
|
}
|
2024-04-19 07:55:29 +00:00
|
|
|
write(var, p_store_string_func, p_store_string_ud, p_encode_res_func, p_encode_res_ud, p_recursion_count, p_compat);
|
2020-05-14 14:41:43 +00:00
|
|
|
}
|
2020-02-01 06:04:14 +00:00
|
|
|
|
|
|
|
p_store_string_func(p_store_string_ud, "]");
|
2015-12-31 03:31:00 +00:00
|
|
|
}
|
|
|
|
|
2023-06-24 18:03:28 +00:00
|
|
|
if (array.is_typed()) {
|
2022-11-27 07:56:53 +00:00
|
|
|
p_store_string_func(p_store_string_ud, ")");
|
|
|
|
}
|
2015-12-31 03:31:00 +00:00
|
|
|
} break;
|
|
|
|
|
2020-02-17 21:06:54 +00:00
|
|
|
case Variant::PACKED_BYTE_ARRAY: {
|
2019-09-24 17:45:03 +00:00
|
|
|
p_store_string_func(p_store_string_ud, "PackedByteArray(");
|
2020-02-17 21:06:54 +00:00
|
|
|
Vector<uint8_t> data = p_variant;
|
2024-04-19 07:55:29 +00:00
|
|
|
if (p_compat) {
|
|
|
|
int len = data.size();
|
|
|
|
const uint8_t *ptr = data.ptr();
|
|
|
|
for (int i = 0; i < len; i++) {
|
|
|
|
if (i > 0) {
|
|
|
|
p_store_string_func(p_store_string_ud, ", ");
|
|
|
|
}
|
|
|
|
p_store_string_func(p_store_string_ud, itos(ptr[i]));
|
|
|
|
}
|
|
|
|
} else if (data.size() > 0) {
|
2024-03-05 16:53:57 +00:00
|
|
|
p_store_string_func(p_store_string_ud, "\"");
|
|
|
|
p_store_string_func(p_store_string_ud, CryptoCore::b64_encode_str(data.ptr(), data.size()));
|
|
|
|
p_store_string_func(p_store_string_ud, "\"");
|
2015-12-31 03:31:00 +00:00
|
|
|
}
|
2019-09-24 17:45:03 +00:00
|
|
|
p_store_string_func(p_store_string_ud, ")");
|
2015-12-31 03:31:00 +00:00
|
|
|
} break;
|
Variant: Added 64-bit packed arrays, renamed Variant::REAL to FLOAT.
- Renames PackedIntArray to PackedInt32Array.
- Renames PackedFloatArray to PackedFloat32Array.
- Adds PackedInt64Array and PackedFloat64Array.
- Renames Variant::REAL to Variant::FLOAT for consistency.
Packed arrays are for storing large amount of data and creating stuff like
meshes, buffers. textures, etc. Forcing them to be 64 is a huge waste of
memory. That said, many users requested the ability to have 64 bits packed
arrays for their games, so this is just an optional added type.
For Variant, the float datatype is always 64 bits, and exposed as `float`.
We still have `real_t` which is the datatype that can change from 32 to 64
bits depending on a compile flag (not entirely working right now, but that's
the idea). It affects math related datatypes and code only.
Neither Variant nor PackedArray make use of real_t, which is only intended
for math precision, so the term is removed from there to keep only float.
2020-02-24 18:20:53 +00:00
|
|
|
case Variant::PACKED_INT32_ARRAY: {
|
2019-09-24 17:45:03 +00:00
|
|
|
p_store_string_func(p_store_string_ud, "PackedInt32Array(");
|
Variant: Added 64-bit packed arrays, renamed Variant::REAL to FLOAT.
- Renames PackedIntArray to PackedInt32Array.
- Renames PackedFloatArray to PackedFloat32Array.
- Adds PackedInt64Array and PackedFloat64Array.
- Renames Variant::REAL to Variant::FLOAT for consistency.
Packed arrays are for storing large amount of data and creating stuff like
meshes, buffers. textures, etc. Forcing them to be 64 is a huge waste of
memory. That said, many users requested the ability to have 64 bits packed
arrays for their games, so this is just an optional added type.
For Variant, the float datatype is always 64 bits, and exposed as `float`.
We still have `real_t` which is the datatype that can change from 32 to 64
bits depending on a compile flag (not entirely working right now, but that's
the idea). It affects math related datatypes and code only.
Neither Variant nor PackedArray make use of real_t, which is only intended
for math precision, so the term is removed from there to keep only float.
2020-02-24 18:20:53 +00:00
|
|
|
Vector<int32_t> data = p_variant;
|
|
|
|
int32_t len = data.size();
|
|
|
|
const int32_t *ptr = data.ptr();
|
2015-12-31 03:31:00 +00:00
|
|
|
|
Variant: Added 64-bit packed arrays, renamed Variant::REAL to FLOAT.
- Renames PackedIntArray to PackedInt32Array.
- Renames PackedFloatArray to PackedFloat32Array.
- Adds PackedInt64Array and PackedFloat64Array.
- Renames Variant::REAL to Variant::FLOAT for consistency.
Packed arrays are for storing large amount of data and creating stuff like
meshes, buffers. textures, etc. Forcing them to be 64 is a huge waste of
memory. That said, many users requested the ability to have 64 bits packed
arrays for their games, so this is just an optional added type.
For Variant, the float datatype is always 64 bits, and exposed as `float`.
We still have `real_t` which is the datatype that can change from 32 to 64
bits depending on a compile flag (not entirely working right now, but that's
the idea). It affects math related datatypes and code only.
Neither Variant nor PackedArray make use of real_t, which is only intended
for math precision, so the term is removed from there to keep only float.
2020-02-24 18:20:53 +00:00
|
|
|
for (int32_t i = 0; i < len; i++) {
|
2020-05-14 14:41:43 +00:00
|
|
|
if (i > 0) {
|
2015-12-31 03:31:00 +00:00
|
|
|
p_store_string_func(p_store_string_ud, ", ");
|
2020-05-14 14:41:43 +00:00
|
|
|
}
|
2015-12-31 03:31:00 +00:00
|
|
|
|
|
|
|
p_store_string_func(p_store_string_ud, itos(ptr[i]));
|
|
|
|
}
|
|
|
|
|
2019-09-24 17:45:03 +00:00
|
|
|
p_store_string_func(p_store_string_ud, ")");
|
2015-12-31 03:31:00 +00:00
|
|
|
} break;
|
Variant: Added 64-bit packed arrays, renamed Variant::REAL to FLOAT.
- Renames PackedIntArray to PackedInt32Array.
- Renames PackedFloatArray to PackedFloat32Array.
- Adds PackedInt64Array and PackedFloat64Array.
- Renames Variant::REAL to Variant::FLOAT for consistency.
Packed arrays are for storing large amount of data and creating stuff like
meshes, buffers. textures, etc. Forcing them to be 64 is a huge waste of
memory. That said, many users requested the ability to have 64 bits packed
arrays for their games, so this is just an optional added type.
For Variant, the float datatype is always 64 bits, and exposed as `float`.
We still have `real_t` which is the datatype that can change from 32 to 64
bits depending on a compile flag (not entirely working right now, but that's
the idea). It affects math related datatypes and code only.
Neither Variant nor PackedArray make use of real_t, which is only intended
for math precision, so the term is removed from there to keep only float.
2020-02-24 18:20:53 +00:00
|
|
|
case Variant::PACKED_INT64_ARRAY: {
|
2019-09-24 17:45:03 +00:00
|
|
|
p_store_string_func(p_store_string_ud, "PackedInt64Array(");
|
Variant: Added 64-bit packed arrays, renamed Variant::REAL to FLOAT.
- Renames PackedIntArray to PackedInt32Array.
- Renames PackedFloatArray to PackedFloat32Array.
- Adds PackedInt64Array and PackedFloat64Array.
- Renames Variant::REAL to Variant::FLOAT for consistency.
Packed arrays are for storing large amount of data and creating stuff like
meshes, buffers. textures, etc. Forcing them to be 64 is a huge waste of
memory. That said, many users requested the ability to have 64 bits packed
arrays for their games, so this is just an optional added type.
For Variant, the float datatype is always 64 bits, and exposed as `float`.
We still have `real_t` which is the datatype that can change from 32 to 64
bits depending on a compile flag (not entirely working right now, but that's
the idea). It affects math related datatypes and code only.
Neither Variant nor PackedArray make use of real_t, which is only intended
for math precision, so the term is removed from there to keep only float.
2020-02-24 18:20:53 +00:00
|
|
|
Vector<int64_t> data = p_variant;
|
|
|
|
int64_t len = data.size();
|
|
|
|
const int64_t *ptr = data.ptr();
|
|
|
|
|
|
|
|
for (int64_t i = 0; i < len; i++) {
|
2020-05-14 14:41:43 +00:00
|
|
|
if (i > 0) {
|
Variant: Added 64-bit packed arrays, renamed Variant::REAL to FLOAT.
- Renames PackedIntArray to PackedInt32Array.
- Renames PackedFloatArray to PackedFloat32Array.
- Adds PackedInt64Array and PackedFloat64Array.
- Renames Variant::REAL to Variant::FLOAT for consistency.
Packed arrays are for storing large amount of data and creating stuff like
meshes, buffers. textures, etc. Forcing them to be 64 is a huge waste of
memory. That said, many users requested the ability to have 64 bits packed
arrays for their games, so this is just an optional added type.
For Variant, the float datatype is always 64 bits, and exposed as `float`.
We still have `real_t` which is the datatype that can change from 32 to 64
bits depending on a compile flag (not entirely working right now, but that's
the idea). It affects math related datatypes and code only.
Neither Variant nor PackedArray make use of real_t, which is only intended
for math precision, so the term is removed from there to keep only float.
2020-02-24 18:20:53 +00:00
|
|
|
p_store_string_func(p_store_string_ud, ", ");
|
2020-05-14 14:41:43 +00:00
|
|
|
}
|
Variant: Added 64-bit packed arrays, renamed Variant::REAL to FLOAT.
- Renames PackedIntArray to PackedInt32Array.
- Renames PackedFloatArray to PackedFloat32Array.
- Adds PackedInt64Array and PackedFloat64Array.
- Renames Variant::REAL to Variant::FLOAT for consistency.
Packed arrays are for storing large amount of data and creating stuff like
meshes, buffers. textures, etc. Forcing them to be 64 is a huge waste of
memory. That said, many users requested the ability to have 64 bits packed
arrays for their games, so this is just an optional added type.
For Variant, the float datatype is always 64 bits, and exposed as `float`.
We still have `real_t` which is the datatype that can change from 32 to 64
bits depending on a compile flag (not entirely working right now, but that's
the idea). It affects math related datatypes and code only.
Neither Variant nor PackedArray make use of real_t, which is only intended
for math precision, so the term is removed from there to keep only float.
2020-02-24 18:20:53 +00:00
|
|
|
|
|
|
|
p_store_string_func(p_store_string_ud, itos(ptr[i]));
|
|
|
|
}
|
|
|
|
|
2019-09-24 17:45:03 +00:00
|
|
|
p_store_string_func(p_store_string_ud, ")");
|
Variant: Added 64-bit packed arrays, renamed Variant::REAL to FLOAT.
- Renames PackedIntArray to PackedInt32Array.
- Renames PackedFloatArray to PackedFloat32Array.
- Adds PackedInt64Array and PackedFloat64Array.
- Renames Variant::REAL to Variant::FLOAT for consistency.
Packed arrays are for storing large amount of data and creating stuff like
meshes, buffers. textures, etc. Forcing them to be 64 is a huge waste of
memory. That said, many users requested the ability to have 64 bits packed
arrays for their games, so this is just an optional added type.
For Variant, the float datatype is always 64 bits, and exposed as `float`.
We still have `real_t` which is the datatype that can change from 32 to 64
bits depending on a compile flag (not entirely working right now, but that's
the idea). It affects math related datatypes and code only.
Neither Variant nor PackedArray make use of real_t, which is only intended
for math precision, so the term is removed from there to keep only float.
2020-02-24 18:20:53 +00:00
|
|
|
} break;
|
|
|
|
case Variant::PACKED_FLOAT32_ARRAY: {
|
2019-09-24 17:45:03 +00:00
|
|
|
p_store_string_func(p_store_string_ud, "PackedFloat32Array(");
|
Variant: Added 64-bit packed arrays, renamed Variant::REAL to FLOAT.
- Renames PackedIntArray to PackedInt32Array.
- Renames PackedFloatArray to PackedFloat32Array.
- Adds PackedInt64Array and PackedFloat64Array.
- Renames Variant::REAL to Variant::FLOAT for consistency.
Packed arrays are for storing large amount of data and creating stuff like
meshes, buffers. textures, etc. Forcing them to be 64 is a huge waste of
memory. That said, many users requested the ability to have 64 bits packed
arrays for their games, so this is just an optional added type.
For Variant, the float datatype is always 64 bits, and exposed as `float`.
We still have `real_t` which is the datatype that can change from 32 to 64
bits depending on a compile flag (not entirely working right now, but that's
the idea). It affects math related datatypes and code only.
Neither Variant nor PackedArray make use of real_t, which is only intended
for math precision, so the term is removed from there to keep only float.
2020-02-24 18:20:53 +00:00
|
|
|
Vector<float> data = p_variant;
|
|
|
|
int len = data.size();
|
|
|
|
const float *ptr = data.ptr();
|
|
|
|
|
|
|
|
for (int i = 0; i < len; i++) {
|
2020-05-14 14:41:43 +00:00
|
|
|
if (i > 0) {
|
Variant: Added 64-bit packed arrays, renamed Variant::REAL to FLOAT.
- Renames PackedIntArray to PackedInt32Array.
- Renames PackedFloatArray to PackedFloat32Array.
- Adds PackedInt64Array and PackedFloat64Array.
- Renames Variant::REAL to Variant::FLOAT for consistency.
Packed arrays are for storing large amount of data and creating stuff like
meshes, buffers. textures, etc. Forcing them to be 64 is a huge waste of
memory. That said, many users requested the ability to have 64 bits packed
arrays for their games, so this is just an optional added type.
For Variant, the float datatype is always 64 bits, and exposed as `float`.
We still have `real_t` which is the datatype that can change from 32 to 64
bits depending on a compile flag (not entirely working right now, but that's
the idea). It affects math related datatypes and code only.
Neither Variant nor PackedArray make use of real_t, which is only intended
for math precision, so the term is removed from there to keep only float.
2020-02-24 18:20:53 +00:00
|
|
|
p_store_string_func(p_store_string_ud, ", ");
|
2020-05-14 14:41:43 +00:00
|
|
|
}
|
2020-07-23 04:42:19 +00:00
|
|
|
p_store_string_func(p_store_string_ud, rtos_fix(ptr[i]));
|
Variant: Added 64-bit packed arrays, renamed Variant::REAL to FLOAT.
- Renames PackedIntArray to PackedInt32Array.
- Renames PackedFloatArray to PackedFloat32Array.
- Adds PackedInt64Array and PackedFloat64Array.
- Renames Variant::REAL to Variant::FLOAT for consistency.
Packed arrays are for storing large amount of data and creating stuff like
meshes, buffers. textures, etc. Forcing them to be 64 is a huge waste of
memory. That said, many users requested the ability to have 64 bits packed
arrays for their games, so this is just an optional added type.
For Variant, the float datatype is always 64 bits, and exposed as `float`.
We still have `real_t` which is the datatype that can change from 32 to 64
bits depending on a compile flag (not entirely working right now, but that's
the idea). It affects math related datatypes and code only.
Neither Variant nor PackedArray make use of real_t, which is only intended
for math precision, so the term is removed from there to keep only float.
2020-02-24 18:20:53 +00:00
|
|
|
}
|
|
|
|
|
2019-09-24 17:45:03 +00:00
|
|
|
p_store_string_func(p_store_string_ud, ")");
|
Variant: Added 64-bit packed arrays, renamed Variant::REAL to FLOAT.
- Renames PackedIntArray to PackedInt32Array.
- Renames PackedFloatArray to PackedFloat32Array.
- Adds PackedInt64Array and PackedFloat64Array.
- Renames Variant::REAL to Variant::FLOAT for consistency.
Packed arrays are for storing large amount of data and creating stuff like
meshes, buffers. textures, etc. Forcing them to be 64 is a huge waste of
memory. That said, many users requested the ability to have 64 bits packed
arrays for their games, so this is just an optional added type.
For Variant, the float datatype is always 64 bits, and exposed as `float`.
We still have `real_t` which is the datatype that can change from 32 to 64
bits depending on a compile flag (not entirely working right now, but that's
the idea). It affects math related datatypes and code only.
Neither Variant nor PackedArray make use of real_t, which is only intended
for math precision, so the term is removed from there to keep only float.
2020-02-24 18:20:53 +00:00
|
|
|
} break;
|
|
|
|
case Variant::PACKED_FLOAT64_ARRAY: {
|
2019-09-24 17:45:03 +00:00
|
|
|
p_store_string_func(p_store_string_ud, "PackedFloat64Array(");
|
Variant: Added 64-bit packed arrays, renamed Variant::REAL to FLOAT.
- Renames PackedIntArray to PackedInt32Array.
- Renames PackedFloatArray to PackedFloat32Array.
- Adds PackedInt64Array and PackedFloat64Array.
- Renames Variant::REAL to Variant::FLOAT for consistency.
Packed arrays are for storing large amount of data and creating stuff like
meshes, buffers. textures, etc. Forcing them to be 64 is a huge waste of
memory. That said, many users requested the ability to have 64 bits packed
arrays for their games, so this is just an optional added type.
For Variant, the float datatype is always 64 bits, and exposed as `float`.
We still have `real_t` which is the datatype that can change from 32 to 64
bits depending on a compile flag (not entirely working right now, but that's
the idea). It affects math related datatypes and code only.
Neither Variant nor PackedArray make use of real_t, which is only intended
for math precision, so the term is removed from there to keep only float.
2020-02-24 18:20:53 +00:00
|
|
|
Vector<double> data = p_variant;
|
2015-12-31 03:31:00 +00:00
|
|
|
int len = data.size();
|
Variant: Added 64-bit packed arrays, renamed Variant::REAL to FLOAT.
- Renames PackedIntArray to PackedInt32Array.
- Renames PackedFloatArray to PackedFloat32Array.
- Adds PackedInt64Array and PackedFloat64Array.
- Renames Variant::REAL to Variant::FLOAT for consistency.
Packed arrays are for storing large amount of data and creating stuff like
meshes, buffers. textures, etc. Forcing them to be 64 is a huge waste of
memory. That said, many users requested the ability to have 64 bits packed
arrays for their games, so this is just an optional added type.
For Variant, the float datatype is always 64 bits, and exposed as `float`.
We still have `real_t` which is the datatype that can change from 32 to 64
bits depending on a compile flag (not entirely working right now, but that's
the idea). It affects math related datatypes and code only.
Neither Variant nor PackedArray make use of real_t, which is only intended
for math precision, so the term is removed from there to keep only float.
2020-02-24 18:20:53 +00:00
|
|
|
const double *ptr = data.ptr();
|
2015-12-31 03:31:00 +00:00
|
|
|
|
|
|
|
for (int i = 0; i < len; i++) {
|
2020-05-14 14:41:43 +00:00
|
|
|
if (i > 0) {
|
2015-12-31 03:31:00 +00:00
|
|
|
p_store_string_func(p_store_string_ud, ", ");
|
2020-05-14 14:41:43 +00:00
|
|
|
}
|
2020-07-23 04:42:19 +00:00
|
|
|
p_store_string_func(p_store_string_ud, rtos_fix(ptr[i]));
|
2015-12-31 03:31:00 +00:00
|
|
|
}
|
|
|
|
|
2019-09-24 17:45:03 +00:00
|
|
|
p_store_string_func(p_store_string_ud, ")");
|
2015-12-31 03:31:00 +00:00
|
|
|
} break;
|
2020-02-17 21:06:54 +00:00
|
|
|
case Variant::PACKED_STRING_ARRAY: {
|
2019-09-24 17:45:03 +00:00
|
|
|
p_store_string_func(p_store_string_ud, "PackedStringArray(");
|
2020-02-17 21:06:54 +00:00
|
|
|
Vector<String> data = p_variant;
|
2015-12-31 03:31:00 +00:00
|
|
|
int len = data.size();
|
2020-02-17 21:06:54 +00:00
|
|
|
const String *ptr = data.ptr();
|
|
|
|
|
2015-12-31 03:31:00 +00:00
|
|
|
for (int i = 0; i < len; i++) {
|
2020-05-14 14:41:43 +00:00
|
|
|
if (i > 0) {
|
2015-12-31 03:31:00 +00:00
|
|
|
p_store_string_func(p_store_string_ud, ", ");
|
2020-05-14 14:41:43 +00:00
|
|
|
}
|
2022-11-27 07:56:53 +00:00
|
|
|
p_store_string_func(p_store_string_ud, "\"" + ptr[i].c_escape() + "\"");
|
2015-12-31 03:31:00 +00:00
|
|
|
}
|
|
|
|
|
2019-09-24 17:45:03 +00:00
|
|
|
p_store_string_func(p_store_string_ud, ")");
|
2015-12-31 03:31:00 +00:00
|
|
|
} break;
|
2020-02-17 21:06:54 +00:00
|
|
|
case Variant::PACKED_VECTOR2_ARRAY: {
|
2019-09-24 17:45:03 +00:00
|
|
|
p_store_string_func(p_store_string_ud, "PackedVector2Array(");
|
2020-02-17 21:06:54 +00:00
|
|
|
Vector<Vector2> data = p_variant;
|
2015-12-31 03:31:00 +00:00
|
|
|
int len = data.size();
|
2020-02-17 21:06:54 +00:00
|
|
|
const Vector2 *ptr = data.ptr();
|
2015-12-31 03:31:00 +00:00
|
|
|
|
|
|
|
for (int i = 0; i < len; i++) {
|
2020-05-14 14:41:43 +00:00
|
|
|
if (i > 0) {
|
2015-12-31 03:31:00 +00:00
|
|
|
p_store_string_func(p_store_string_ud, ", ");
|
2020-05-14 14:41:43 +00:00
|
|
|
}
|
2020-07-23 04:42:19 +00:00
|
|
|
p_store_string_func(p_store_string_ud, rtos_fix(ptr[i].x) + ", " + rtos_fix(ptr[i].y));
|
2015-12-31 03:31:00 +00:00
|
|
|
}
|
|
|
|
|
2019-09-24 17:45:03 +00:00
|
|
|
p_store_string_func(p_store_string_ud, ")");
|
2015-12-31 03:31:00 +00:00
|
|
|
} break;
|
2020-02-17 21:06:54 +00:00
|
|
|
case Variant::PACKED_VECTOR3_ARRAY: {
|
2019-09-24 17:45:03 +00:00
|
|
|
p_store_string_func(p_store_string_ud, "PackedVector3Array(");
|
2020-02-17 21:06:54 +00:00
|
|
|
Vector<Vector3> data = p_variant;
|
2015-12-31 03:31:00 +00:00
|
|
|
int len = data.size();
|
2020-02-17 21:06:54 +00:00
|
|
|
const Vector3 *ptr = data.ptr();
|
2015-12-31 03:31:00 +00:00
|
|
|
|
|
|
|
for (int i = 0; i < len; i++) {
|
2020-05-14 14:41:43 +00:00
|
|
|
if (i > 0) {
|
2015-12-31 03:31:00 +00:00
|
|
|
p_store_string_func(p_store_string_ud, ", ");
|
2020-05-14 14:41:43 +00:00
|
|
|
}
|
2020-07-23 04:42:19 +00:00
|
|
|
p_store_string_func(p_store_string_ud, rtos_fix(ptr[i].x) + ", " + rtos_fix(ptr[i].y) + ", " + rtos_fix(ptr[i].z));
|
2015-12-31 03:31:00 +00:00
|
|
|
}
|
|
|
|
|
2019-09-24 17:45:03 +00:00
|
|
|
p_store_string_func(p_store_string_ud, ")");
|
2015-12-31 03:31:00 +00:00
|
|
|
} break;
|
2020-02-17 21:06:54 +00:00
|
|
|
case Variant::PACKED_COLOR_ARRAY: {
|
2019-09-24 17:45:03 +00:00
|
|
|
p_store_string_func(p_store_string_ud, "PackedColorArray(");
|
2020-02-17 21:06:54 +00:00
|
|
|
Vector<Color> data = p_variant;
|
2015-12-31 03:31:00 +00:00
|
|
|
int len = data.size();
|
2020-02-17 21:06:54 +00:00
|
|
|
const Color *ptr = data.ptr();
|
2015-12-31 03:31:00 +00:00
|
|
|
|
|
|
|
for (int i = 0; i < len; i++) {
|
2020-05-14 14:41:43 +00:00
|
|
|
if (i > 0) {
|
2016-06-13 13:16:43 +00:00
|
|
|
p_store_string_func(p_store_string_ud, ", ");
|
2020-05-14 14:41:43 +00:00
|
|
|
}
|
2020-07-23 04:42:19 +00:00
|
|
|
p_store_string_func(p_store_string_ud, rtos_fix(ptr[i].r) + ", " + rtos_fix(ptr[i].g) + ", " + rtos_fix(ptr[i].b) + ", " + rtos_fix(ptr[i].a));
|
2015-12-31 03:31:00 +00:00
|
|
|
}
|
2022-11-27 07:56:53 +00:00
|
|
|
|
2019-09-24 17:45:03 +00:00
|
|
|
p_store_string_func(p_store_string_ud, ")");
|
2015-12-31 03:31:00 +00:00
|
|
|
} break;
|
2024-04-08 14:51:34 +00:00
|
|
|
case Variant::PACKED_VECTOR4_ARRAY: {
|
|
|
|
p_store_string_func(p_store_string_ud, "PackedVector4Array(");
|
|
|
|
Vector<Vector4> data = p_variant;
|
|
|
|
int len = data.size();
|
|
|
|
const Vector4 *ptr = data.ptr();
|
|
|
|
|
|
|
|
for (int i = 0; i < len; i++) {
|
|
|
|
if (i > 0) {
|
|
|
|
p_store_string_func(p_store_string_ud, ", ");
|
|
|
|
}
|
|
|
|
p_store_string_func(p_store_string_ud, rtos_fix(ptr[i].x) + ", " + rtos_fix(ptr[i].y) + ", " + rtos_fix(ptr[i].z) + ", " + rtos_fix(ptr[i].w));
|
|
|
|
}
|
|
|
|
|
|
|
|
p_store_string_func(p_store_string_ud, ")");
|
|
|
|
} break;
|
2023-08-29 12:05:53 +00:00
|
|
|
|
2019-04-09 15:08:36 +00:00
|
|
|
default: {
|
2023-06-21 09:06:32 +00:00
|
|
|
ERR_PRINT("Unknown variant type");
|
|
|
|
return ERR_BUG;
|
2019-04-09 15:08:36 +00:00
|
|
|
}
|
2015-12-31 03:31:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
static Error _write_to_str(void *ud, const String &p_string) {
|
|
|
|
String *str = (String *)ud;
|
|
|
|
(*str) += p_string;
|
|
|
|
return OK;
|
|
|
|
}
|
|
|
|
|
2024-04-19 07:55:29 +00:00
|
|
|
Error VariantWriter::write_to_string(const Variant &p_variant, String &r_string, EncodeResourceFunc p_encode_res_func, void *p_encode_res_ud, bool p_compat) {
|
2015-12-31 03:31:00 +00:00
|
|
|
r_string = String();
|
|
|
|
|
2024-04-19 07:55:29 +00:00
|
|
|
return write(p_variant, _write_to_str, &r_string, p_encode_res_func, p_encode_res_ud, 0, p_compat);
|
2015-12-31 03:31:00 +00:00
|
|
|
}
|