1
0
mirror of synced 2025-02-07 14:41:21 +01:00

impr: Move language string interpretation to compile time

This commit is contained in:
WerWolv 2024-07-27 11:23:21 +02:00
parent a8ad045248
commit ce26fe1db7
2 changed files with 67 additions and 25 deletions

View File

@ -1,11 +1,14 @@
#pragma once #pragma once
#include <hex.hpp>
#include <map> #include <map>
#include <string> #include <string>
#include <string_view> #include <string_view>
#include <vector> #include <vector>
#include <fmt/format.h> #include <fmt/format.h>
#include <wolv/types/static_string.hpp>
namespace hex { namespace hex {
@ -22,8 +25,10 @@ namespace hex {
}; };
namespace impl { namespace impl {
void setFallbackLanguage(const std::string &language); void setFallbackLanguage(const std::string &language);
void resetLanguageStrings(); void resetLanguageStrings();
} }
void loadLanguage(const std::string &language); void loadLanguage(const std::string &language);
@ -32,6 +37,7 @@ namespace hex {
[[nodiscard]] const std::map<std::string, std::string> &getSupportedLanguages(); [[nodiscard]] const std::map<std::string, std::string> &getSupportedLanguages();
[[nodiscard]] const std::string &getFallbackLanguage(); [[nodiscard]] const std::string &getFallbackLanguage();
[[nodiscard]] const std::string &getSelectedLanguage(); [[nodiscard]] const std::string &getSelectedLanguage();
} }
struct UnlocalizedString; struct UnlocalizedString;
@ -47,10 +53,30 @@ namespace hex {
[[nodiscard]] operator std::string_view() const; [[nodiscard]] operator std::string_view() const;
[[nodiscard]] operator const char *() const; [[nodiscard]] operator const char *() const;
[[nodiscard]] const std::string &get() const; const std::string &get() const;
constexpr static size_t hash(std::string_view string){
constexpr u64 p = 131;
constexpr u64 m = std::numeric_limits<std::uint32_t>::max() - 4; // Largest 32 bit prime
u64 total = 0;
u64 currentMultiplier = 1;
for (char c : string) {
total = (total + currentMultiplier * c) % m;
currentMultiplier = (currentMultiplier * p) % m;
}
return total;
}
private: private:
std::string m_unlocalizedString; constexpr explicit Lang(std::size_t hash) : m_entryHash(hash) {}
template<wolv::type::StaticString>
friend consteval Lang operator""_lang();
private:
std::size_t m_entryHash;
}; };
[[nodiscard]] std::string operator+(const std::string &&left, const Lang &&right); [[nodiscard]] std::string operator+(const std::string &&left, const Lang &&right);
@ -61,14 +87,16 @@ namespace hex {
[[nodiscard]] std::string operator+(const Lang &&left, const char *right); [[nodiscard]] std::string operator+(const Lang &&left, const char *right);
[[nodiscard]] std::string operator+(const Lang &&left, const Lang &&right); [[nodiscard]] std::string operator+(const Lang &&left, const Lang &&right);
[[nodiscard]] inline Lang operator""_lang(const char *string, size_t) { template<wolv::type::StaticString String>
return Lang(string); [[nodiscard]] consteval Lang operator""_lang() {
return Lang(Lang::hash(String.value.data()));
} }
struct UnlocalizedString { struct UnlocalizedString {
public: public:
UnlocalizedString() = default; UnlocalizedString() = default;
UnlocalizedString(auto && arg) : m_unlocalizedString(std::forward<decltype(arg)>(arg)) { UnlocalizedString(auto && arg) : m_unlocalizedString(std::forward<decltype(arg)>(arg)) {
static_assert(!std::same_as<std::remove_cvref_t<decltype(arg)>, Lang>, "Expected a unlocalized name, got a localized one!"); static_assert(!std::same_as<std::remove_cvref_t<decltype(arg)>, Lang>, "Expected a unlocalized name, got a localized one!");
} }
@ -107,4 +135,4 @@ namespace hex {
return entry.get(); return entry.get();
} }
} }

View File

@ -10,7 +10,7 @@ namespace hex {
AutoReset<std::string> s_fallbackLanguage; AutoReset<std::string> s_fallbackLanguage;
AutoReset<std::string> s_selectedLanguage; AutoReset<std::string> s_selectedLanguage;
AutoReset<std::map<std::string, std::string>> s_currStrings; AutoReset<std::map<size_t, std::string>> s_currStrings;
} }
@ -41,6 +41,21 @@ namespace hex {
return m_entries; return m_entries;
} }
static void loadLanguageDefinitions(const std::vector<LanguageDefinition> &definitions) {
for (const auto &definition : definitions) {
const auto &entries = definition.getEntries();
if (entries.empty())
continue;
for (const auto &[key, value] : entries) {
if (value.empty())
continue;
s_currStrings->emplace(Lang::hash(key), value);
}
}
}
void loadLanguage(const std::string &language) { void loadLanguage(const std::string &language) {
auto &definitions = ContentRegistry::Language::impl::getLanguageDefinitions(); auto &definitions = ContentRegistry::Language::impl::getLanguageDefinitions();
@ -49,14 +64,10 @@ namespace hex {
s_currStrings->clear(); s_currStrings->clear();
for (const auto &definition : definitions.at(language)) loadLanguageDefinitions(definitions.at(language));
s_currStrings->insert(definition.getEntries().begin(), definition.getEntries().end());
const auto& fallbackLanguage = getFallbackLanguage(); const auto& fallbackLanguage = getFallbackLanguage();
if (language != fallbackLanguage && definitions.contains(fallbackLanguage)) { loadLanguageDefinitions(definitions.at(fallbackLanguage));
for (const auto &definition : definitions.at(fallbackLanguage))
s_currStrings->insert(definition.getEntries().begin(), definition.getEntries().end());
}
s_selectedLanguage = language; s_selectedLanguage = language;
} }
@ -98,11 +109,10 @@ namespace hex {
} }
Lang::Lang(const char *unlocalizedString) : m_unlocalizedString(unlocalizedString) { } Lang::Lang(const char *unlocalizedString) : m_entryHash(hash(unlocalizedString)) { }
Lang::Lang(const std::string &unlocalizedString) : m_unlocalizedString(unlocalizedString) { } Lang::Lang(const std::string &unlocalizedString) : m_entryHash(hash(unlocalizedString)) { }
Lang::Lang(const UnlocalizedString &unlocalizedString) : m_unlocalizedString(unlocalizedString.get()) { } Lang::Lang(const UnlocalizedString &unlocalizedString) : m_entryHash(hash(unlocalizedString.get())) { }
Lang::Lang(std::string_view unlocalizedString) : m_unlocalizedString(unlocalizedString) { } Lang::Lang(std::string_view unlocalizedString) : m_entryHash(hash(unlocalizedString)) { }
Lang::operator std::string() const { Lang::operator std::string() const {
return get(); return get();
@ -116,6 +126,18 @@ namespace hex {
return get().c_str(); return get().c_str();
} }
const std::string &Lang::get() const {
const auto &lang = *LocalizationManager::s_currStrings;
auto it = lang.find(m_entryHash);
if (it == lang.end()) {
static const std::string invalidString = "[ !!! INVALID LANGUAGE STRING !!! ]";
return invalidString;
} else {
return it->second;
}
}
std::string operator+(const std::string &&left, const Lang &&right) { std::string operator+(const std::string &&left, const Lang &&right) {
return left + static_cast<std::string>(right); return left + static_cast<std::string>(right);
} }
@ -144,12 +166,4 @@ namespace hex {
return static_cast<std::string>(left) + right; return static_cast<std::string>(left) + right;
} }
const std::string &Lang::get() const {
auto &lang = LocalizationManager::s_currStrings;
if (lang->contains(m_unlocalizedString))
return lang->at(m_unlocalizedString);
else
return m_unlocalizedString;
}
} }