9#include "absl/strings/ascii.h"
10#include "absl/strings/str_format.h"
11#include "absl/strings/str_split.h"
25 const unsigned char uc =
static_cast<unsigned char>(c);
26 return std::isalnum(uc) || c ==
'_';
30 const bool left_boundary = (pos == 0) || !
IsWordChar(text[pos - 1]);
31 const size_t right_index = pos + len;
32 const bool right_boundary =
33 (right_index >= text.size()) || !
IsWordChar(text[right_index]);
34 return left_boundary && right_boundary;
37std::string LowercaseCopy(std::string_view input) {
38 std::string lowered(input);
39 for (
char& c : lowered) {
40 c =
static_cast<char>(std::tolower(
static_cast<unsigned char>(c)));
60 uint8_t best_match = 0xFF;
61 const wchar_t target =
62 static_cast<wchar_t>(
static_cast<unsigned char>(value));
63 for (
const auto& [key, char_value] : CharEncoder) {
64 if (char_value != target) {
67 if (best_match == 0xFF || key < best_match) {
75 if (value <
DICTOFF || value == 0xFF) {
82 for (
const auto& text_element : TextCommands) {
83 if (text_element.ID == b) {
91 auto it = std::ranges::find_if(SpecialChars,
93 return text_element.
ID == value;
95 if (it != SpecialChars.end()) {
103 std::vector<TextElement> commands_and_chars = TextCommands;
104 commands_and_chars.insert(commands_and_chars.end(), SpecialChars.begin(),
106 for (
auto& text_element : commands_and_chars) {
107 match = text_element.MatchMe(str);
108 if (match.size() > 0) {
109 if (text_element.HasArgument) {
110 std::string arg = match[1].str().substr(1);
112 return ParsedElement(text_element, std::stoi(arg,
nullptr, 16));
113 }
catch (
const std::invalid_argument& e) {
114 util::logf(
"Error parsing argument for %s: %s",
115 text_element.GenericToken.c_str(), arg.c_str());
117 }
catch (
const std::out_of_range& e) {
118 util::logf(
"Argument out of range for %s: %s",
119 text_element.GenericToken.c_str(), arg.c_str());
128 const auto dictionary_element =
131 match = dictionary_element.MatchMe(str);
132 if (match.size() > 0) {
135 std::string dict_arg = match[1].str().substr(1);
137 DICTOFF + std::stoi(dict_arg,
nullptr, 16));
138 }
catch (
const std::exception& e) {
139 util::logf(
"Error parsing dictionary token: %s", match[1].str().c_str());
147 if (CharEncoder.contains(value)) {
148 char c = CharEncoder.at(value);
149 std::string str =
"";
156 text_element != std::nullopt) {
157 return text_element->GenericToken;
162 special_element != std::nullopt) {
163 return special_element->GenericToken;
168 if (dictionary >= 0) {
170 static_cast<unsigned char>(dictionary));
177 std::vector<uint8_t> bytes;
178 std::string temp_string = std::move(str);
180 while (pos < temp_string.size()) {
182 if (temp_string[pos] ==
'[') {
183 int next = temp_string.find(
']', pos);
191 const auto dictionary_element =
194 if (!parsedElement.
Active) {
195 util::logf(
"Error parsing message: %s", temp_string);
197 }
else if (parsedElement.
Parent == dictionary_element) {
198 bytes.push_back(parsedElement.
Value);
200 bytes.push_back(parsedElement.
Parent.
ID);
203 bytes.push_back(parsedElement.
Value);
223 std::string temp_string(str);
225 bool warned_newline =
false;
227 while (pos < temp_string.size()) {
228 char current = temp_string[pos];
229 if (current ==
'\r' || current ==
'\n') {
230 if (!warned_newline) {
232 "Literal newlines are ignored; use [1], [2], [3], [V], or [K] "
233 "tokens for line breaks.");
234 warned_newline =
true;
240 if (current ==
'[') {
241 size_t close = temp_string.find(
']', pos);
242 if (close == std::string::npos) {
244 absl::StrFormat(
"Unclosed token starting at position %zu", pos));
248 std::string token = temp_string.substr(pos, close - pos + 1);
250 const auto dictionary_element =
253 if (!parsed_element.
Active) {
254 result.
errors.push_back(absl::StrFormat(
"Unknown token: %s", token));
261 result.
errors.push_back(absl::StrFormat(
"Unknown token: %s", token));
267 if (parsed_element.
Parent == dictionary_element) {
282 result.
errors.push_back(absl::StrFormat(
283 "Unsupported character '%c' at position %zu", current, pos));
288 result.
bytes.push_back(bb);
306 const std::string lowered = absl::AsciiStrToLower(std::string(value));
307 if (lowered ==
"vanilla") {
310 if (lowered ==
"expanded") {
313 return absl::InvalidArgumentError(
314 absl::StrFormat(
"Unknown message bank: %s", std::string(value)));
318 std::vector<DictionaryEntry> AllDictionaries;
320 std::vector<uint8_t> bytes;
321 std::stringstream stringBuilder;
327 int temppush_backress =
332 while (address < temppush_backress) {
333 uint8_t uint8_tDictionary = rom->
data()[address++];
334 bytes.push_back(uint8_tDictionary);
338 AllDictionaries.push_back(
DictionaryEntry{(uint8_t)i, stringBuilder.str()});
341 std::ranges::sort(AllDictionaries,
346 return AllDictionaries;
350 std::string str,
const std::vector<DictionaryEntry>& dictionary) {
351 std::string temp = std::move(str);
352 for (
const auto& entry : dictionary) {
353 if (entry.ContainedInString(temp)) {
354 temp = entry.ReplaceInstancesOfIn(temp);
361 std::string_view query,
size_t start_pos,
363 bool match_whole_word) {
364 if (query.empty() || start_pos > text.size()) {
368 std::string haystack_storage;
369 std::string query_storage;
370 std::string_view haystack = text;
371 std::string_view needle = query;
372 if (!case_sensitive) {
373 haystack_storage = LowercaseCopy(text);
374 query_storage = LowercaseCopy(query);
375 haystack = haystack_storage;
376 needle = query_storage;
379 size_t pos = haystack.find(needle, start_pos);
380 while (pos != std::string::npos) {
381 if (!match_whole_word || MatchesWholeWordAt(text, pos, query.size())) {
384 pos = haystack.find(needle, pos + 1);
391 std::string_view replacement,
size_t start_pos,
392 bool replace_all,
bool case_sensitive,
393 bool match_whole_word,
394 size_t* first_replaced_pos) {
395 if (!text || query.empty() || start_pos > text->size()) {
399 int replacements = 0;
400 size_t cursor = start_pos;
402 const auto match_pos =
403 FindTextMatch(*text, query, cursor, case_sensitive, match_whole_word);
404 if (!match_pos.has_value()) {
408 text->replace(*match_pos, query.size(), replacement);
409 if (replacements == 0 && first_replaced_pos !=
nullptr) {
410 *first_replaced_pos = *match_pos;
414 cursor = *match_pos + replacement.size();
419 if (cursor > text->size()) {
428 uint8_t value,
const std::vector<DictionaryEntry>& dictionary) {
429 for (
const auto& entry : dictionary) {
430 if (entry.ID +
DICTOFF == value) {
438 const std::vector<uint8_t>& rom_data,
int* current_pos) {
439 if (current_pos ==
nullptr) {
440 return absl::InvalidArgumentError(
"current_pos is null");
442 if (*current_pos < 0 ||
443 static_cast<size_t>(*current_pos) >= rom_data.size()) {
444 return absl::OutOfRangeError(
"current_pos is out of range");
448 int pos = *current_pos;
449 uint8_t current_byte;
450 std::vector<uint8_t> temp_bytes_raw;
451 std::vector<uint8_t> temp_bytes_parsed;
452 std::string current_message_raw;
453 std::string current_message_parsed;
456 while (pos <
static_cast<int>(rom_data.size())) {
457 current_byte = rom_data[pos++];
460 message_data.
ID = message_data.
ID + 1;
462 message_data.
RawString = current_message_raw;
463 message_data.
Data = temp_bytes_raw;
467 temp_bytes_raw.clear();
468 temp_bytes_parsed.clear();
469 current_message_raw.clear();
470 current_message_parsed.clear();
474 }
else if (current_byte == 0xFF) {
475 return absl::InvalidArgumentError(
"message terminator not found");
478 temp_bytes_raw.push_back(current_byte);
482 if (text_element != std::nullopt) {
483 temp_bytes_parsed.push_back(current_byte);
484 if (text_element->HasArgument) {
485 if (pos >=
static_cast<int>(rom_data.size())) {
486 return absl::OutOfRangeError(
"message command argument out of range");
488 uint8_t arg_byte = rom_data[pos++];
489 temp_bytes_raw.push_back(arg_byte);
490 temp_bytes_parsed.push_back(arg_byte);
491 current_message_raw.append(text_element->GetParamToken(arg_byte));
492 current_message_parsed.append(text_element->GetParamToken(arg_byte));
494 current_message_raw.append(text_element->GetParamToken());
495 current_message_parsed.append(text_element->GetParamToken());
502 special_element != std::nullopt) {
503 current_message_raw.append(special_element->GetParamToken());
504 current_message_parsed.append(special_element->GetParamToken());
505 temp_bytes_parsed.push_back(current_byte);
511 if (dictionary >= 0) {
512 std::string token = absl::StrFormat(
514 current_message_raw.append(token);
515 current_message_parsed.append(token);
516 temp_bytes_parsed.push_back(current_byte);
521 if (CharEncoder.contains(current_byte)) {
522 std::string str =
"";
523 str.push_back(CharEncoder.at(current_byte));
524 current_message_raw.append(str);
525 current_message_parsed.append(str);
526 temp_bytes_parsed.push_back(current_byte);
531 return absl::InvalidArgumentError(
"message terminator not found");
535 std::vector<MessageData>& message_data,
536 const std::vector<DictionaryEntry>& dictionary_entries) {
537 std::vector<std::string> parsed_messages;
539 for (
auto& message : message_data) {
540 std::string parsed_message =
"";
542 for (
size_t pos = 0; pos < message.Data.size(); ++pos) {
543 uint8_t
byte = message.Data[pos];
547 if (text_element != std::nullopt) {
550 text_element->ID ==
kLine3) {
551 parsed_message.append(
"\n");
554 if (text_element->HasArgument && pos + 1 < message.Data.size()) {
555 uint8_t arg_byte = message.Data[pos + 1];
556 parsed_message.append(text_element->GetParamToken(arg_byte));
559 parsed_message.append(text_element->GetParamToken());
566 if (special_element != std::nullopt) {
567 parsed_message.append(special_element->GetParamToken());
574 for (
const auto& entry : dictionary_entries) {
575 if (entry.ID ==
byte -
DICTOFF) {
580 parsed_message.append(dic_entry.
Contents);
585 if (CharEncoder.contains(
byte)) {
586 parsed_message.push_back(CharEncoder.at(
byte));
589 parsed_messages.push_back(parsed_message);
592 return parsed_messages;
596 std::vector<MessageData> list_of_texts;
600 return list_of_texts;
602 if (max_pos > 0 && (pos < 0 || pos >= max_pos)) {
603 return list_of_texts;
606 std::vector<uint8_t> raw_message;
607 std::vector<uint8_t> parsed_message;
608 std::string current_raw_message;
609 std::string current_parsed_message;
611 bool did_bank_switch =
false;
612 uint8_t current_byte = 0;
613 while (current_byte != 0xFF) {
614 if (max_pos > 0 && (pos < 0 || pos >= max_pos))
616 current_byte = rom[pos++];
618 list_of_texts.push_back(
619 MessageData(message_id++, pos, current_raw_message, raw_message,
620 current_parsed_message, parsed_message));
622 parsed_message.clear();
623 current_raw_message.clear();
624 current_parsed_message.clear();
626 }
else if (current_byte == 0xFF) {
630 raw_message.push_back(current_byte);
633 if (text_element != std::nullopt) {
634 parsed_message.push_back(current_byte);
635 if (text_element->HasArgument) {
636 if (max_pos > 0 && (pos < 0 || pos >= max_pos))
638 current_byte = rom[pos++];
639 raw_message.push_back(current_byte);
640 parsed_message.push_back(current_byte);
643 current_raw_message.append(text_element->GetParamToken(current_byte));
644 current_parsed_message.append(text_element->GetParamToken(current_byte));
646 if (text_element->Token ==
kBankToken && !did_bank_switch) {
647 did_bank_switch =
true;
656 if (special_element != std::nullopt) {
657 current_raw_message.append(special_element->GetParamToken());
658 current_parsed_message.append(special_element->GetParamToken());
659 parsed_message.push_back(current_byte);
665 if (dictionary >= 0) {
666 current_raw_message.append(absl::StrFormat(
676 if (ptr_a < 0 || ptr_a + 1 >= max_pos || ptr_b < 0 ||
677 ptr_b + 1 >= max_pos) {
684 uint32_t address_end =
688 const uint32_t max_u =
static_cast<uint32_t
>(max_pos);
689 if (address >= max_u || address_end > max_u || address_end < address) {
694 for (uint32_t i = address; i < address_end; i++) {
695 if (max_pos > 0 && i >=
static_cast<uint32_t
>(max_pos))
697 parsed_message.push_back(rom[i]);
705 if (CharEncoder.contains(current_byte)) {
706 std::string str =
"";
707 str.push_back(CharEncoder.at(current_byte));
708 current_raw_message.append(str);
709 current_parsed_message.append(str);
710 parsed_message.push_back(current_byte);
714 return list_of_texts;
718 std::vector<std::string>& parsed_messages,
719 std::vector<MessageData>& expanded_messages,
720 std::vector<DictionaryEntry>& dictionary) {
721 static Rom expanded_message_rom;
722 if (!expanded_message_rom.
LoadFromFile(expanded_message_path).ok()) {
723 return absl::InternalError(
"Failed to load expanded message ROM");
726 auto parsed_expanded_messages =
729 for (
const auto& expanded_message : expanded_messages) {
730 parsed_messages.push_back(parsed_expanded_messages[expanded_message.ID]);
732 return absl::OkStatus();
736 const std::vector<MessageData>& messages) {
737 nlohmann::json j = nlohmann::json::array();
738 for (
const auto& msg : messages) {
739 j.push_back({{
"id", msg.ID},
740 {
"address", msg.Address},
741 {
"raw_string", msg.RawString},
742 {
"parsed_string", msg.ContentsParsed}});
748 const std::vector<MessageData>& messages) {
751 std::ofstream file(path);
752 if (!file.is_open()) {
753 return absl::InternalError(
754 absl::StrFormat(
"Failed to open file for writing: %s", path));
757 return absl::OkStatus();
758 }
catch (
const std::exception& e) {
759 return absl::InternalError(
760 absl::StrFormat(
"JSON export failed: %s", e.what()));
765 const std::vector<MessageData>& vanilla,
766 const std::vector<MessageData>& expanded) {
768 j[
"format"] =
"yaze-message-bundle";
770 j[
"counts"] = {{
"vanilla", vanilla.size()}, {
"expanded", expanded.size()}};
771 j[
"messages"] = nlohmann::json::array();
773 auto append_messages = [&j](
const std::vector<MessageData>& messages,
775 for (
const auto& msg : messages) {
776 nlohmann::json entry;
777 entry[
"id"] = msg.ID;
779 entry[
"address"] = msg.Address;
780 entry[
"raw"] = msg.RawString;
781 entry[
"parsed"] = msg.ContentsParsed;
783 !msg.RawString.empty() ? msg.RawString : msg.ContentsParsed;
784 entry[
"length"] = msg.Data.size();
785 const std::string validation_text =
786 !msg.RawString.empty() ? msg.RawString : msg.ContentsParsed;
788 if (!warnings.empty()) {
789 entry[
"line_width_warnings"] = warnings;
791 j[
"messages"].push_back(entry);
802 const std::string& path,
const std::vector<MessageData>& vanilla,
803 const std::vector<MessageData>& expanded) {
806 std::ofstream file(path);
807 if (!file.is_open()) {
808 return absl::InternalError(
809 absl::StrFormat(
"Failed to open file for writing: %s", path));
812 return absl::OkStatus();
813 }
catch (
const std::exception& e) {
814 return absl::InternalError(
815 absl::StrFormat(
"Message bundle export failed: %s", e.what()));
821 const nlohmann::json& entry,
MessageBank default_bank) {
822 if (!entry.is_object()) {
823 return absl::InvalidArgumentError(
"Message entry must be an object");
827 result.
id = entry.value(
"id", -1);
829 return absl::InvalidArgumentError(
"Message entry missing valid id");
832 if (entry.contains(
"bank")) {
833 if (!entry[
"bank"].is_string()) {
834 return absl::InvalidArgumentError(
"Message entry bank must be string");
838 return bank_or.status();
840 result.
bank = bank_or.value();
842 result.
bank = default_bank;
845 if (entry.contains(
"raw") && entry[
"raw"].is_string()) {
846 result.
raw = entry[
"raw"].get<std::string>();
847 }
else if (entry.contains(
"raw_string") && entry[
"raw_string"].is_string()) {
848 result.
raw = entry[
"raw_string"].get<std::string>();
851 if (entry.contains(
"parsed") && entry[
"parsed"].is_string()) {
852 result.
parsed = entry[
"parsed"].get<std::string>();
853 }
else if (entry.contains(
"parsed_string") &&
854 entry[
"parsed_string"].is_string()) {
855 result.
parsed = entry[
"parsed_string"].get<std::string>();
858 if (entry.contains(
"text") && entry[
"text"].is_string()) {
859 result.
text = entry[
"text"].get<std::string>();
862 if (result.
text.empty()) {
863 if (!result.
raw.empty()) {
865 }
else if (!result.
parsed.empty()) {
870 if (result.
text.empty()) {
871 return absl::InvalidArgumentError(
872 absl::StrFormat(
"Message entry %d missing text content", result.
id));
880 const nlohmann::json& json) {
881 std::vector<MessageBundleEntry> entries;
883 if (json.is_array()) {
884 for (
const auto& entry : json) {
886 if (!parsed_or.ok()) {
887 return parsed_or.status();
889 entries.push_back(parsed_or.value());
894 if (!json.is_object()) {
895 return absl::InvalidArgumentError(
"Message bundle JSON must be object");
898 if (json.contains(
"version") && json[
"version"].is_number_integer()) {
899 int version = json[
"version"].get<
int>();
901 return absl::InvalidArgumentError(
902 absl::StrFormat(
"Unsupported message bundle version: %d", version));
906 if (!json.contains(
"messages") || !json[
"messages"].is_array()) {
907 return absl::InvalidArgumentError(
"Message bundle missing messages array");
910 for (
const auto& entry : json[
"messages"]) {
912 if (!parsed_or.ok()) {
913 return parsed_or.status();
915 entries.push_back(parsed_or.value());
922 const std::string& path) {
923 std::ifstream file(path);
924 if (!file.is_open()) {
925 return absl::NotFoundError(
926 absl::StrFormat(
"Cannot open message bundle: %s", path));
932 }
catch (
const std::exception& e) {
933 return absl::InvalidArgumentError(
934 absl::StrFormat(
"Failed to parse JSON: %s", e.what()));
945 std::vector<std::string> warnings;
950 int visible_chars = 0;
951 bool all_spaces_this_line =
true;
954 while (pos < message.size()) {
955 if (message[pos] ==
'[') {
957 size_t close = message.find(
']', pos);
958 if (close == std::string::npos)
961 std::string token = message.substr(pos, close - pos + 1);
966 if (token ==
"[1]" || token ==
"[2]" || token ==
"[3]" ||
967 token ==
"[V]" || token ==
"[K]") {
970 if (visible_chars >
kMaxLineWidth && !all_spaces_this_line) {
972 absl::StrFormat(
"Line %d: %d visible characters (max %d)",
977 all_spaces_this_line =
true;
988 if (message[pos] !=
' ')
989 all_spaces_this_line =
false;
995 if (visible_chars >
kMaxLineWidth && !all_spaces_this_line) {
997 absl::StrFormat(
"Line %d: %d visible characters (max %d)", line_num,
1009 const std::string& line) {
1012 if (line.size() < 6 || line[0] !=
'*' || line[1] !=
'*' || line[2] !=
' ') {
1013 return std::nullopt;
1017 size_t sep = line.find(
" - ", 3);
1018 if (sep == std::string::npos) {
1019 return std::nullopt;
1023 std::string hex_id = line.substr(3, sep - 3);
1026 message_id = std::stoi(hex_id,
nullptr, 16);
1027 }
catch (
const std::exception&) {
1028 return std::nullopt;
1032 std::string label = line.substr(sep + 3);
1034 return std::make_pair(message_id, label);
1038 const std::string& content) {
1039 std::vector<std::pair<int, std::string>> messages;
1040 std::istringstream stream(content);
1043 int current_id = -1;
1044 std::string current_body;
1046 while (std::getline(stream, line)) {
1049 if (header.has_value()) {
1051 if (current_id >= 0) {
1053 while (!current_body.empty() && current_body.back() ==
'\n') {
1054 current_body.pop_back();
1056 messages.push_back({current_id, current_body});
1059 current_id = header->first;
1060 current_body.clear();
1065 if (!line.empty() && line[0] ==
'*' &&
1066 (line.size() < 2 || line[1] !=
'*')) {
1071 if (current_id >= 0) {
1072 if (!current_body.empty()) {
1073 current_body +=
"\n";
1075 current_body += line;
1080 if (current_id >= 0) {
1081 while (!current_body.empty() && current_body.back() ==
'\n') {
1082 current_body.pop_back();
1084 messages.push_back({current_id, current_body});
1091 const std::vector<std::pair<int, std::string>>& messages,
1092 const std::vector<std::string>& labels) {
1094 output +=
"* Oracle of Secrets English Dialogue\n";
1096 for (
size_t i = 0; i < messages.size(); ++i) {
1097 const auto& [msg_id, body] = messages[i];
1098 std::string label = (i < labels.size())
1100 : absl::StrFormat(
"Message %02X", msg_id);
1102 output += absl::StrFormat(
"** %02X - %s\n", msg_id, label);
1120 const std::vector<std::string>& messages) {
1121 if (rom ==
nullptr || !rom->
is_loaded()) {
1122 return absl::InvalidArgumentError(
"ROM not loaded");
1124 if (start < 0 || end < start) {
1125 return absl::InvalidArgumentError(
"Invalid expanded message region");
1128 const int capacity = end - start + 1;
1129 if (capacity <= 0) {
1130 return absl::InvalidArgumentError(
1131 "Expanded message region has no capacity");
1134 const auto& data = rom->
vector();
1135 if (end >=
static_cast<int>(data.size())) {
1136 return absl::OutOfRangeError(
"Expanded message region out of ROM range");
1141 std::vector<uint8_t> blob;
1142 blob.reserve(
static_cast<size_t>(capacity));
1145 for (
size_t i = 0; i < messages.size(); ++i) {
1147 const int needed =
static_cast<int>(bytes.size()) + 1;
1150 if (used + needed + 1 > capacity) {
1151 return absl::ResourceExhaustedError(absl::StrFormat(
1152 "Expanded message data exceeds bank boundary "
1153 "(at message %d, used=%d, needed=%d, capacity=%d, end=0x%06X)",
1154 static_cast<int>(i), used, needed, capacity, end));
1157 blob.insert(blob.end(), bytes.begin(), bytes.end());
1162 if (used + 1 > capacity) {
1163 return absl::ResourceExhaustedError(
1164 "No space for end-of-region marker (0xFF)");
1166 blob.push_back(0xFF);
1171 const uint32_t fence_start =
static_cast<uint32_t
>(start);
1172 const uint32_t fence_end =
1173 static_cast<uint32_t
>(
static_cast<uint64_t
>(end) + 1ULL);
1181 const std::vector<std::string>& messages) {
1183 int capacity = end - start + 1;
1185 for (
size_t i = 0; i < messages.size(); ++i) {
1189 int needed =
static_cast<int>(bytes.size()) + 1;
1190 if (i == messages.size() - 1) {
1194 if (pos + needed - start > capacity) {
1195 return absl::ResourceExhaustedError(
1196 absl::StrFormat(
"Expanded message data exceeds bank boundary "
1197 "(at message %d, pos 0x%06X, end 0x%06X)",
1198 static_cast<int>(i), pos, end));
1202 for (uint8_t
byte : bytes) {
1210 if (pos - start >= capacity) {
1211 return absl::ResourceExhaustedError(
1212 "No space for end-of-region marker (0xFF)");
1216 return absl::OkStatus();
1220 const std::vector<MessageData>& messages) {
1221 if (rom ==
nullptr || !rom->
is_loaded()) {
1222 return absl::InvalidArgumentError(
"ROM not loaded");
1226 bool in_second_bank =
false;
1228 for (
const auto& message : messages) {
1229 for (uint8_t value : message.Data) {
1234 return absl::ResourceExhaustedError(absl::StrFormat(
1235 "Text data exceeds first bank (pos 0x%06X)", pos));
1238 in_second_bank =
true;
1248 return absl::ResourceExhaustedError(
1249 absl::StrFormat(
"Text data exceeds first bank (pos 0x%06X)", pos));
1253 return absl::ResourceExhaustedError(
1254 absl::StrFormat(
"Text data exceeds second bank (pos 0x%06X)", pos));
1258 return absl::OkStatus();
The Rom class is used to load, save, and modify Rom data. This is a generic SNES ROM container and do...
absl::Status LoadFromFile(const std::string &filename, const LoadOptions &options=LoadOptions::Defaults())
absl::Status WriteByte(int addr, uint8_t value)
const auto & vector() const
absl::Status WriteVector(int addr, std::vector< uint8_t > data)
static RomSettings & Get()
uint32_t GetAddressOr(const std::string &key, uint32_t default_value) const
absl::Status Allow(uint32_t start, uint32_t end, std::string_view label)
constexpr char kExpandedMessageEnd[]
constexpr char kExpandedMessageStart[]
bool MatchesWholeWordAt(std::string_view text, size_t pos, size_t len)
absl::StatusOr< MessageBundleEntry > ParseMessageBundleEntry(const nlohmann::json &entry, MessageBank default_bank)
uint8_t FindMatchingCharacter(char value)
const std::string kBankToken
nlohmann::json SerializeMessagesToJson(const std::vector< MessageData > &messages)
absl::StatusOr< MessageBank > MessageBankFromString(std::string_view value)
DictionaryEntry FindRealDictionaryEntry(uint8_t value, const std::vector< DictionaryEntry > &dictionary)
constexpr int kMaxLineWidth
int GetExpandedTextDataStart()
constexpr int kMessageBundleVersion
const std::string DICTIONARYTOKEN
constexpr uint8_t kScrollVertical
std::string ParseTextDataByte(uint8_t value)
absl::Status WriteAllTextData(Rom *rom, const std::vector< MessageData > &messages)
absl::Status LoadExpandedMessages(std::string &expanded_message_path, std::vector< std::string > &parsed_messages, std::vector< MessageData > &expanded_messages, std::vector< DictionaryEntry > &dictionary)
std::optional< std::pair< int, std::string > > ParseOrgHeader(const std::string &line)
std::string MessageBankToString(MessageBank bank)
constexpr int kExpandedTextDataEndDefault
std::string ReplaceAllDictionaryWords(std::string str, const std::vector< DictionaryEntry > &dictionary)
absl::Status WriteExpandedTextData(Rom *rom, int start, int end, const std::vector< std::string > &messages)
nlohmann::json SerializeMessageBundle(const std::vector< MessageData > &vanilla, const std::vector< MessageData > &expanded)
constexpr int kPointersDictionaries
absl::StatusOr< std::vector< MessageBundleEntry > > LoadMessageBundleFromJson(const std::string &path)
constexpr int kNumDictionaryEntries
absl::StatusOr< MessageData > ParseSingleMessage(const std::vector< uint8_t > &rom_data, int *current_pos)
absl::StatusOr< std::vector< MessageBundleEntry > > ParseMessageBundleJson(const nlohmann::json &json)
std::vector< std::string > ParseMessageData(std::vector< MessageData > &message_data, const std::vector< DictionaryEntry > &dictionary_entries)
std::optional< TextElement > FindMatchingSpecial(uint8_t value)
constexpr uint8_t kMessageTerminator
std::vector< MessageData > ReadAllTextData(uint8_t *rom, int pos, int max_pos)
constexpr int kTextData2End
std::vector< DictionaryEntry > BuildDictionaryEntries(Rom *rom)
constexpr uint8_t kBankSwitchCommand
int ReplaceTextMatches(std::string *text, std::string_view query, std::string_view replacement, size_t start_pos, bool replace_all, bool case_sensitive, bool match_whole_word, size_t *first_replaced_pos)
std::optional< size_t > FindTextMatch(std::string_view text, std::string_view query, size_t start_pos, bool case_sensitive, bool match_whole_word)
std::vector< uint8_t > ParseMessageToData(std::string str)
absl::Status ExportMessagesToJson(const std::string &path, const std::vector< MessageData > &messages)
absl::Status ExportMessageBundleToJson(const std::string &path, const std::vector< MessageData > &vanilla, const std::vector< MessageData > &expanded)
constexpr uint8_t DICTOFF
std::string ExportToOrgFormat(const std::vector< std::pair< int, std::string > > &messages, const std::vector< std::string > &labels)
std::vector< MessageData > ReadExpandedTextData(uint8_t *rom, int pos)
std::optional< TextElement > FindMatchingCommand(uint8_t b)
MessageParseResult ParseMessageToDataWithDiagnostics(std::string_view str)
int GetExpandedTextDataEnd()
ParsedElement FindMatchingElement(const std::string &str)
std::vector< std::string > ValidateMessageLineWidths(const std::string &message)
std::vector< std::pair< int, std::string > > ParseOrgContent(const std::string &content)
constexpr int kExpandedTextDataDefault
int8_t FindDictionaryEntry(uint8_t value)
constexpr int kTextDataEnd
std::string HexByte(uint8_t byte, HexStringParams params)
void logf(const absl::FormatSpec< Args... > &format, Args &&... args)
uint32_t Get24LocalFromPC(uint8_t *data, int addr, bool pc=true)
uint32_t SnesToPc(uint32_t addr) noexcept
#define RETURN_IF_ERROR(expr)
std::vector< uint8_t > Data
std::vector< uint8_t > DataParsed
std::string ContentsParsed
std::vector< uint8_t > bytes
std::vector< std::string > errors
std::vector< std::string > warnings
std::string GetParamToken(uint8_t value=0) const