From dacbe8e3fbda85f7c2e7e7b660795f2a080a9d25 Mon Sep 17 00:00:00 2001 From: Star Rauchenberger Date: Thu, 13 Mar 2025 11:55:55 -0400 Subject: Use sync fields for hunt panels --- src/ap_state.cpp | 53 ++++++++++++++++++++++++++++++++++------------------- 1 file changed, 34 insertions(+), 19 deletions(-) (limited to 'src/ap_state.cpp') diff --git a/src/ap_state.cpp b/src/ap_state.cpp index 023bf7f..8ba6633 100644 --- a/src/ap_state.cpp +++ b/src/ap_state.cpp @@ -10,6 +10,7 @@ #include #include #include +#include #include #include #include @@ -37,6 +38,10 @@ constexpr int ITEM_HANDLING = 7; // <- all constexpr int CONNECTION_TIMEOUT = 50000; // 50 seconds constexpr int CONNECTION_BACKOFF_INTERVAL = 100; +constexpr int PANEL_COUNT = 803; +constexpr int PANEL_BITFIELD_LENGTH = 48; +constexpr int PANEL_BITFIELDS = 17; + namespace { const std::set kNonProgressionItems = { @@ -79,6 +84,7 @@ struct APState { std::set checked_locations; std::map data_storage; std::optional> player_pos; + std::bitset solved_panels; DoorShuffleMode door_shuffle_mode = kNO_DOORS; bool group_doors = false; @@ -142,6 +148,7 @@ struct APState { checked_locations.clear(); data_storage.clear(); player_pos = std::nullopt; + solved_panels.reset(); victory_data_storage_key.clear(); door_shuffle_mode = kNO_DOORS; group_doors = false; @@ -216,14 +223,6 @@ struct APState { return checked_locations.count(location_id); } - bool HasCheckedHuntPanel(int location_id) { - std::lock_guard state_guard(state_mutex); - - std::string key = - fmt::format("{}Hunt|{}", data_storage_prefix, location_id); - return data_storage.count(key) && std::any_cast(data_storage.at(key)); - } - bool HasItem(int item_id, int quantity) { return inventory.count(item_id) && inventory.at(item_id) >= quantity; } @@ -288,6 +287,12 @@ struct APState { 30; // CLIENT_GOAL } + bool IsPanelSolved(int solve_index) { + std::lock_guard state_guard(state_mutex); + + return solved_panels.test(solve_index); + } + private: void Initialize() { if (!initialized) { @@ -300,11 +305,8 @@ struct APState { "Achievement|{}", GD_GetPanel(panel_id).achievement_name)); } - for (const MapArea& map_area : GD_GetMapAreas()) { - for (const Location& location : map_area.locations) { - tracked_data_storage_keys.push_back( - fmt::format("Hunt|{}", location.ap_location_id)); - } + for (int i = 0; i < PANEL_BITFIELDS; i++) { + tracked_data_storage_keys.push_back(fmt::format("Panels_{}", i)); } tracked_data_storage_keys.push_back("PlayerPos"); @@ -610,8 +612,6 @@ struct APState { if (key.find("Achievement|") != std::string::npos) { state_update.achievements = true; - } else if (key.find("Hunt|") != std::string::npos) { - state_update.hunt_panels = true; } } else if (value.is_number()) { data_storage[key] = value.get(); @@ -620,6 +620,21 @@ struct APState { if (key == victory_data_storage_key) { state_update.cleared_locations = true; + } else if (key.find("Panels_") != std::string::npos) { + int bitfield_num = + std::stoi(key.substr(data_storage_prefix.size() + 7)); + uint64_t bitfield_value = value.get(); + for (int i = 0; i < PANEL_BITFIELD_LENGTH; i++) { + if ((bitfield_value & (1LL << i)) != 0) { + int solve_index = bitfield_num * PANEL_BITFIELD_LENGTH + i; + + if (!solved_panels.test(solve_index)) { + state_update.panels.insert(solve_index); + } + + solved_panels.set(solve_index); + } + } } } else if (value.is_object()) { if (key.ends_with("PlayerPos")) { @@ -720,10 +735,6 @@ bool AP_HasCheckedGameLocation(int location_id) { return GetState().HasCheckedGameLocation(location_id); } -bool AP_HasCheckedHuntPanel(int location_id) { - return GetState().HasCheckedHuntPanel(location_id); -} - bool AP_HasItem(int item_id, int quantity) { return GetState().HasItem(item_id, quantity); } @@ -892,3 +903,7 @@ std::optional> AP_GetPlayerPosition() { return GetState().player_pos; } + +bool AP_IsPanelSolved(int solve_index) { + return GetState().IsPanelSolved(solve_index); +} -- cgit 1.4.1