diff --git a/include/recomp_ui.h b/include/recomp_ui.h index 8d127fd..b8e4f85 100644 --- a/include/recomp_ui.h +++ b/include/recomp_ui.h @@ -48,9 +48,9 @@ namespace recompui { void show_context(ContextId context, std::string_view param); void hide_context(ContextId context); void hide_all_contexts(); - bool is_context_open(ContextId context); + bool is_context_shown(ContextId context); bool is_context_taking_input(); - bool is_any_context_open(); + bool is_any_context_shown(); ContextId get_launcher_context_id(); ContextId get_config_context_id(); diff --git a/src/game/input.cpp b/src/game/input.cpp index 4f94771..8d59e0c 100644 --- a/src/game/input.cpp +++ b/src/game/input.cpp @@ -157,7 +157,7 @@ bool sdl_event_filter(void* userdata, SDL_Event* event) { } recompui::ContextId config_context_id = recompui::get_config_context_id(); - if (!recompui::is_context_open(config_context_id)) { + if (!recompui::is_context_shown(config_context_id)) { recompui::show_context(config_context_id, ""); } @@ -713,7 +713,7 @@ void recomp::set_right_analog_suppressed(bool suppressed) { bool recomp::game_input_disabled() { // Disable input if any menu that blocks input is open. - return recompui::is_any_context_open(); + return recompui::is_context_taking_input(); } bool recomp::all_input_disabled() { diff --git a/src/ui/core/ui_context.cpp b/src/ui/core/ui_context.cpp index 43d0442..b9fd1d0 100644 --- a/src/ui/core/ui_context.cpp +++ b/src/ui/core/ui_context.cpp @@ -33,6 +33,7 @@ namespace recompui { Rml::ElementDocument* document; Element root_element; std::vector loose_elements; + std::unordered_set to_update; Context(Rml::ElementDocument* document) : document(document), root_element(document) {} }; } // namespace recompui @@ -58,12 +59,15 @@ enum class ContextErrorType { GetContextWithoutOpen, AddResourceWithoutOpen, AddResourceToWrongContext, + UpdateElementWithoutContext, + UpdateElementInWrongContext, GetResourceWithoutOpen, GetResourceFailed, DestroyResourceWithoutOpen, DestroyResourceInWrongContext, DestroyResourceNotFound, GetDocumentInvalidContext, + InternalError, }; enum class SlotTag : uint8_t { @@ -101,6 +105,12 @@ void context_error(recompui::ContextId id, ContextErrorType type) { case ContextErrorType::AddResourceToWrongContext: error_message = "Attempted to create a UI resource in a different UI context than the one that's open"; break; + case ContextErrorType::UpdateElementWithoutContext: + error_message = "Attempted to update a UI element with no open UI context"; + break; + case ContextErrorType::UpdateElementInWrongContext: + error_message = "Attempted to update a UI element in a different UI context than the one that's open"; + break; case ContextErrorType::GetResourceWithoutOpen: error_message = "Attempted to get a UI resource with no open UI context"; break; @@ -119,6 +129,9 @@ void context_error(recompui::ContextId id, ContextErrorType type) { case ContextErrorType::GetDocumentInvalidContext: error_message = "Attempted to get the document of an invalid UI context"; break; + case ContextErrorType::InternalError: + error_message = "Internal error in UI context"; + break; default: error_message = "Unknown UI context error"; break; @@ -317,6 +330,47 @@ void recompui::ContextId::close() { } } +void recompui::ContextId::process_updates() { + // Ensure a context is currently opened by this thread. + if (opened_context_id == ContextId::null()) { + context_error(*this, ContextErrorType::InternalError); + } + + // Check that the context that was specified is the same one that's currently open. + if (*this != opened_context_id) { + context_error(*this, ContextErrorType::InternalError); + } + + // Move the current update set into a local variable. This clears the update set + // and allows it to be used to queue updates from any element callbacks. + std::unordered_set to_update = std::move(opened_context->to_update); + + Event update_event = Event::update_event(); + + for (auto cur_resource_id : to_update) { + resource_slotmap::key cur_key{ cur_resource_id.slot_id }; + + // Ignore any resources that aren't elements. + if (cur_key.get_tag() != static_cast(SlotTag::Element)) { + // Assert to catch errors of queueing other resource types for update. + // This isn't an actual error, so there's no issue with continuing in release builds. + assert(false); + continue; + } + + // Get the resource being updaten from the context. + std::unique_ptr