#include #include #include #include namespace hex { EncodingFile::EncodingFile() : m_mapping(std::make_unique, std::string>>>()) { } EncodingFile::EncodingFile(const hex::EncodingFile &other) { this->m_mapping = std::make_unique, std::string>>>(*other.m_mapping); this->m_tableContent = other.m_tableContent; this->m_longestSequence = other.m_longestSequence; this->m_valid = other.m_valid; this->m_name = other.m_name; } EncodingFile::EncodingFile(EncodingFile &&other) noexcept { this->m_mapping = std::move(other.m_mapping); this->m_tableContent = std::move(other.m_tableContent); this->m_longestSequence = other.m_longestSequence; this->m_valid = other.m_valid; this->m_name = std::move(other.m_name); } EncodingFile::EncodingFile(Type type, const std::fs::path &path) : EncodingFile() { auto file = wolv::io::File(path, wolv::io::File::Mode::Read); switch (type) { case Type::Thingy: parse(file.readString()); break; default: return; } { this->m_name = path.stem().string(); this->m_name = wolv::util::replaceStrings(this->m_name, "_", " "); if (!this->m_name.empty()) this->m_name[0] = std::toupper(this->m_name[0]); } this->m_valid = true; } EncodingFile::EncodingFile(Type type, const std::string &content) : EncodingFile() { switch (type) { case Type::Thingy: parse(content); break; default: return; } this->m_name = "Unknown"; this->m_valid = true; } EncodingFile &EncodingFile::operator=(const hex::EncodingFile &other) { this->m_mapping = std::make_unique, std::string>>>(*other.m_mapping); this->m_tableContent = other.m_tableContent; this->m_longestSequence = other.m_longestSequence; this->m_valid = other.m_valid; this->m_name = other.m_name; return *this; } EncodingFile &EncodingFile::operator=(EncodingFile &&other) noexcept { this->m_mapping = std::move(other.m_mapping); this->m_tableContent = std::move(other.m_tableContent); this->m_longestSequence = other.m_longestSequence; this->m_valid = other.m_valid; this->m_name = std::move(other.m_name); return *this; } std::pair EncodingFile::getEncodingFor(std::span buffer) const { for (auto riter = this->m_mapping->crbegin(); riter != this->m_mapping->crend(); ++riter) { const auto &[size, mapping] = *riter; if (size > buffer.size()) continue; std::vector key(buffer.begin(), buffer.begin() + size); if (mapping.contains(key)) return { mapping.at(key), size }; } return { ".", 1 }; } size_t EncodingFile::getEncodingLengthFor(std::span buffer) const { for (auto riter = this->m_mapping->crbegin(); riter != this->m_mapping->crend(); ++riter) { const auto &[size, mapping] = *riter; if (size > buffer.size()) continue; std::vector key(buffer.begin(), buffer.begin() + size); if (mapping.contains(key)) return size; } return 1; } void EncodingFile::parse(const std::string &content) { this->m_tableContent = content; for (const auto &line : splitString(this->m_tableContent, "\n")) { std::string from, to; { auto delimiterPos = line.find('='); if (delimiterPos >= line.length()) continue; from = line.substr(0, delimiterPos); to = line.substr(delimiterPos + 1); if (from.empty()) continue; } auto fromBytes = hex::parseByteString(from); if (fromBytes.empty()) continue; if (to.length() > 1) to = wolv::util::trim(to); if (to.empty()) to = " "; if (!this->m_mapping->contains(fromBytes.size())) this->m_mapping->insert({ fromBytes.size(), {} }); auto keySize = fromBytes.size(); (*this->m_mapping)[keySize].insert({ std::move(fromBytes), to }); this->m_longestSequence = std::max(this->m_longestSequence, keySize); } } }