2021-12-05 17:54:36 +00:00
|
|
|
#include "LDFFormat.h"
|
|
|
|
|
|
|
|
// Custom Classes
|
|
|
|
#include "GeneralUtils.h"
|
|
|
|
|
2023-05-02 22:19:20 +00:00
|
|
|
#include "Game.h"
|
2023-10-21 23:31:55 +00:00
|
|
|
#include "Logger.h"
|
2023-05-02 22:19:20 +00:00
|
|
|
|
2021-12-05 17:54:36 +00:00
|
|
|
// C++
|
2023-05-02 22:19:20 +00:00
|
|
|
#include <string_view>
|
2021-12-05 17:54:36 +00:00
|
|
|
#include <vector>
|
|
|
|
|
2023-05-02 22:19:20 +00:00
|
|
|
using LDFKey = std::string_view;
|
|
|
|
using LDFTypeAndValue = std::string_view;
|
2022-07-28 13:39:57 +00:00
|
|
|
|
2023-05-02 22:19:20 +00:00
|
|
|
using LDFType = std::string_view;
|
|
|
|
using LDFValue = std::string_view;
|
2022-07-28 13:39:57 +00:00
|
|
|
|
2023-05-02 22:19:20 +00:00
|
|
|
//! Returns a pointer to a LDFData value based on string format
|
|
|
|
LDFBaseData* LDFBaseData::DataFromString(const std::string_view& format) {
|
|
|
|
// A valid LDF must be at least 3 characters long (=0:) is the shortest valid LDF (empty UTF-16 key with no initial value)
|
|
|
|
if (format.empty() || format.length() <= 2) return nullptr;
|
|
|
|
auto equalsPosition = format.find('=');
|
|
|
|
// You can have an empty key, just make sure the type and value might exist
|
|
|
|
if (equalsPosition == std::string::npos || equalsPosition == (format.size() - 1)) return nullptr;
|
|
|
|
|
|
|
|
std::pair<LDFKey, LDFTypeAndValue> keyValue;
|
|
|
|
keyValue.first = format.substr(0, equalsPosition);
|
|
|
|
keyValue.second = format.substr(equalsPosition + 1, format.size());
|
|
|
|
|
|
|
|
std::u16string key = GeneralUtils::ASCIIToUTF16(keyValue.first);
|
|
|
|
|
|
|
|
auto colonPosition = keyValue.second.find(':');
|
|
|
|
|
|
|
|
// If : is the first thing after an =, then this is an invalid LDF since
|
|
|
|
// we dont have a type to use.
|
|
|
|
if (colonPosition == std::string::npos || colonPosition == 0) return nullptr;
|
|
|
|
|
|
|
|
std::pair<LDFType, LDFValue> ldfTypeAndValue;
|
|
|
|
ldfTypeAndValue.first = keyValue.second.substr(0, colonPosition);
|
|
|
|
ldfTypeAndValue.second = keyValue.second.substr(colonPosition + 1, keyValue.second.size());
|
|
|
|
|
|
|
|
// Only allow empty values for string values.
|
|
|
|
if (ldfTypeAndValue.second.size() == 0 && !(ldfTypeAndValue.first == "0" || ldfTypeAndValue.first == "13")) return nullptr;
|
|
|
|
|
|
|
|
eLDFType type;
|
|
|
|
char* storage;
|
|
|
|
try {
|
|
|
|
type = static_cast<eLDFType>(strtol(ldfTypeAndValue.first.data(), &storage, 10));
|
|
|
|
} catch (std::exception) {
|
2023-10-21 23:31:55 +00:00
|
|
|
LOG("Attempted to process invalid ldf type (%s) from string (%s)", ldfTypeAndValue.first.data(), format.data());
|
2023-05-02 22:19:20 +00:00
|
|
|
return nullptr;
|
2021-12-05 17:54:36 +00:00
|
|
|
}
|
2022-07-28 13:39:57 +00:00
|
|
|
|
2023-05-02 22:19:20 +00:00
|
|
|
LDFBaseData* returnValue = nullptr;
|
|
|
|
switch (type) {
|
|
|
|
case LDF_TYPE_UTF_16: {
|
|
|
|
std::u16string data = GeneralUtils::UTF8ToUTF16(ldfTypeAndValue.second);
|
|
|
|
returnValue = new LDFData<std::u16string>(key, data);
|
|
|
|
break;
|
|
|
|
}
|
2022-07-28 13:39:57 +00:00
|
|
|
|
2023-05-02 22:19:20 +00:00
|
|
|
case LDF_TYPE_S32: {
|
2024-02-10 11:05:25 +00:00
|
|
|
const auto data = GeneralUtils::TryParse<int32_t>(ldfTypeAndValue.second);
|
|
|
|
if (!data) {
|
2023-10-21 23:31:55 +00:00
|
|
|
LOG("Warning: Attempted to process invalid int32 value (%s) from string (%s)", ldfTypeAndValue.second.data(), format.data());
|
2023-05-02 22:19:20 +00:00
|
|
|
return nullptr;
|
2021-12-05 17:54:36 +00:00
|
|
|
}
|
2024-02-10 11:05:25 +00:00
|
|
|
returnValue = new LDFData<int32_t>(key, data.value());
|
2023-10-09 20:22:40 +00:00
|
|
|
|
2023-05-02 22:19:20 +00:00
|
|
|
break;
|
|
|
|
}
|
2021-12-05 17:54:36 +00:00
|
|
|
|
2023-05-02 22:19:20 +00:00
|
|
|
case LDF_TYPE_FLOAT: {
|
2024-02-10 11:05:25 +00:00
|
|
|
const auto data = GeneralUtils::TryParse<float>(ldfTypeAndValue.second);
|
|
|
|
if (!data) {
|
2023-10-21 23:31:55 +00:00
|
|
|
LOG("Warning: Attempted to process invalid float value (%s) from string (%s)", ldfTypeAndValue.second.data(), format.data());
|
2023-05-02 22:19:20 +00:00
|
|
|
return nullptr;
|
2021-12-05 17:54:36 +00:00
|
|
|
}
|
2024-02-10 11:05:25 +00:00
|
|
|
returnValue = new LDFData<float>(key, data.value());
|
2023-05-02 22:19:20 +00:00
|
|
|
break;
|
|
|
|
}
|
2022-07-28 13:39:57 +00:00
|
|
|
|
2023-05-02 22:19:20 +00:00
|
|
|
case LDF_TYPE_DOUBLE: {
|
2024-02-10 11:05:25 +00:00
|
|
|
const auto data = GeneralUtils::TryParse<double>(ldfTypeAndValue.second);
|
|
|
|
if (!data) {
|
2023-10-21 23:31:55 +00:00
|
|
|
LOG("Warning: Attempted to process invalid double value (%s) from string (%s)", ldfTypeAndValue.second.data(), format.data());
|
2023-05-02 22:19:20 +00:00
|
|
|
return nullptr;
|
2021-12-05 17:54:36 +00:00
|
|
|
}
|
2024-02-10 11:05:25 +00:00
|
|
|
returnValue = new LDFData<double>(key, data.value());
|
2023-05-02 22:19:20 +00:00
|
|
|
break;
|
|
|
|
}
|
2022-07-28 13:39:57 +00:00
|
|
|
|
2023-05-02 22:19:20 +00:00
|
|
|
case LDF_TYPE_U32:
|
|
|
|
{
|
|
|
|
uint32_t data;
|
|
|
|
|
|
|
|
if (ldfTypeAndValue.second == "true") {
|
|
|
|
data = 1;
|
|
|
|
} else if (ldfTypeAndValue.second == "false") {
|
|
|
|
data = 0;
|
|
|
|
} else {
|
2024-02-10 11:05:25 +00:00
|
|
|
const auto dataOptional = GeneralUtils::TryParse<uint32_t>(ldfTypeAndValue.second);
|
|
|
|
if (!dataOptional) {
|
2023-10-21 23:31:55 +00:00
|
|
|
LOG("Warning: Attempted to process invalid uint32 value (%s) from string (%s)", ldfTypeAndValue.second.data(), format.data());
|
2023-05-02 22:19:20 +00:00
|
|
|
return nullptr;
|
2021-12-05 17:54:36 +00:00
|
|
|
}
|
2024-02-10 11:05:25 +00:00
|
|
|
data = dataOptional.value();
|
2023-05-02 22:19:20 +00:00
|
|
|
}
|
2022-07-28 13:39:57 +00:00
|
|
|
|
2023-05-02 22:19:20 +00:00
|
|
|
returnValue = new LDFData<uint32_t>(key, data);
|
|
|
|
break;
|
|
|
|
}
|
2022-07-28 13:39:57 +00:00
|
|
|
|
2023-05-02 22:19:20 +00:00
|
|
|
case LDF_TYPE_BOOLEAN: {
|
|
|
|
bool data;
|
|
|
|
|
|
|
|
if (ldfTypeAndValue.second == "true") {
|
|
|
|
data = true;
|
|
|
|
} else if (ldfTypeAndValue.second == "false") {
|
|
|
|
data = false;
|
|
|
|
} else {
|
2024-02-10 11:05:25 +00:00
|
|
|
const auto dataOptional = GeneralUtils::TryParse<bool>(ldfTypeAndValue.second);
|
|
|
|
if (!dataOptional) {
|
2023-10-21 23:31:55 +00:00
|
|
|
LOG("Warning: Attempted to process invalid bool value (%s) from string (%s)", ldfTypeAndValue.second.data(), format.data());
|
2023-05-02 22:19:20 +00:00
|
|
|
return nullptr;
|
2021-12-05 17:54:36 +00:00
|
|
|
}
|
2024-02-10 11:05:25 +00:00
|
|
|
data = dataOptional.value();
|
2023-05-02 22:19:20 +00:00
|
|
|
}
|
2022-07-28 13:39:57 +00:00
|
|
|
|
2023-05-02 22:19:20 +00:00
|
|
|
returnValue = new LDFData<bool>(key, data);
|
|
|
|
break;
|
|
|
|
}
|
2022-07-28 13:39:57 +00:00
|
|
|
|
2023-05-02 22:19:20 +00:00
|
|
|
case LDF_TYPE_U64: {
|
2024-02-10 11:05:25 +00:00
|
|
|
const auto data = GeneralUtils::TryParse<uint64_t>(ldfTypeAndValue.second);
|
|
|
|
if (!data) {
|
2023-10-21 23:31:55 +00:00
|
|
|
LOG("Warning: Attempted to process invalid uint64 value (%s) from string (%s)", ldfTypeAndValue.second.data(), format.data());
|
2023-05-02 22:19:20 +00:00
|
|
|
return nullptr;
|
|
|
|
}
|
2024-02-10 11:05:25 +00:00
|
|
|
returnValue = new LDFData<uint64_t>(key, data.value());
|
2023-05-02 22:19:20 +00:00
|
|
|
break;
|
|
|
|
}
|
2022-07-28 13:39:57 +00:00
|
|
|
|
2023-05-02 22:19:20 +00:00
|
|
|
case LDF_TYPE_OBJID: {
|
2024-02-10 11:05:25 +00:00
|
|
|
const auto data = GeneralUtils::TryParse<LWOOBJID>(ldfTypeAndValue.second);
|
|
|
|
if (!data) {
|
2023-10-21 23:31:55 +00:00
|
|
|
LOG("Warning: Attempted to process invalid LWOOBJID value (%s) from string (%s)", ldfTypeAndValue.second.data(), format.data());
|
2023-05-02 22:19:20 +00:00
|
|
|
return nullptr;
|
2021-12-05 17:54:36 +00:00
|
|
|
}
|
2024-02-10 11:05:25 +00:00
|
|
|
returnValue = new LDFData<LWOOBJID>(key, data.value());
|
2023-05-02 22:19:20 +00:00
|
|
|
break;
|
2021-12-05 17:54:36 +00:00
|
|
|
}
|
2022-07-28 13:39:57 +00:00
|
|
|
|
2023-05-02 22:19:20 +00:00
|
|
|
case LDF_TYPE_UTF_8: {
|
|
|
|
std::string data = ldfTypeAndValue.second.data();
|
|
|
|
returnValue = new LDFData<std::string>(key, data);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case LDF_TYPE_UNKNOWN: {
|
2023-10-21 23:31:55 +00:00
|
|
|
LOG("Warning: Attempted to process invalid unknown value (%s) from string (%s)", ldfTypeAndValue.second.data(), format.data());
|
2023-05-02 22:19:20 +00:00
|
|
|
break;
|
|
|
|
}
|
2022-07-28 13:39:57 +00:00
|
|
|
|
2023-05-02 22:19:20 +00:00
|
|
|
default: {
|
2023-10-21 23:31:55 +00:00
|
|
|
LOG("Warning: Attempted to process invalid LDF type (%d) from string (%s)", type, format.data());
|
2023-05-02 22:19:20 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return returnValue;
|
2021-12-05 17:54:36 +00:00
|
|
|
}
|