2023-11-18 14:50:43 +01:00
|
|
|
#include <hex/api/event_manager.hpp>
|
2020-12-22 18:10:01 +01:00
|
|
|
|
|
|
|
namespace hex {
|
|
|
|
|
fix: Event unsubscribe not working correcetly when using same key for multiple events (#1309)
<!--
Please provide as much information as possible about what your PR aims
to do.
PRs with no description will most likely be closed until more
information is provided.
If you're planing on changing fundamental behaviour or add big new
features, please open a GitHub Issue first before starting to work on
it.
If it's not something big and you still want to contact us about it,
feel free to do so !
-->
### Problem description
<!-- Describe the bug that you fixed/feature request that you
implemented, or link to an existing issue describing it -->
Fixed possible bug of `EventManager::unsubscribe`
`std::map` only allows unique key, but the same token can subscribe to
multiple events.
https://github.com/WerWolv/ImHex/blob/1a2a926b772f32d4f5e4c723c48e0e6e65a16b0a/lib/libimhex/include/hex/api/event.hpp#L104-L107
If the previous token has already subscribed to an event, then when
subscribing again, `getTokenStore().insert` will not do anything
(Because its type is `std::map`)
https://github.com/WerWolv/ImHex/blob/1a2a926b772f32d4f5e4c723c48e0e6e65a16b0a/lib/libimhex/include/hex/api/event.hpp#L122-L134
At this point in `unsubscribe`, the `iter` may not be able to find the
correct event and erase it
### Implementation description
<!-- Explain what you did to correct the problem -->
Change `tokenStore` to `std::multimap` instead of `std::map`, which
cannot unsubscribe multiple events correctly
### Screenshots
<!-- If your change is visual, take a screenshot showing it. Ideally,
make before/after sceenshots -->
### Additional things
<!-- Anything else you would like to say -->
2023-10-08 05:35:35 +08:00
|
|
|
std::multimap<void *, EventManager::EventList::iterator>& EventManager::getTokenStore() {
|
|
|
|
static std::multimap<void *, EventManager::EventList::iterator> tokenStore;
|
2023-07-26 13:50:51 +02:00
|
|
|
|
|
|
|
return tokenStore;
|
|
|
|
}
|
|
|
|
|
|
|
|
EventManager::EventList& EventManager::getEvents() {
|
|
|
|
static EventManager::EventList events;
|
|
|
|
|
|
|
|
return events;
|
|
|
|
}
|
|
|
|
|
2023-11-04 23:16:38 +01:00
|
|
|
std::recursive_mutex& EventManager::getEventMutex() {
|
|
|
|
static std::recursive_mutex mutex;
|
|
|
|
|
|
|
|
return mutex;
|
|
|
|
}
|
|
|
|
|
2020-12-22 18:10:01 +01:00
|
|
|
|
2024-12-14 19:15:49 +01:00
|
|
|
bool EventManager::isAlreadyRegistered(void *token, impl::EventId id) {
|
|
|
|
if (getTokenStore().contains(token)) {
|
|
|
|
auto&& [begin, end] = getTokenStore().equal_range(token);
|
|
|
|
|
|
|
|
return std::any_of(begin, end, [&](auto &item) {
|
|
|
|
return item.second->first == id;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void EventManager::unsubscribe(void *token, impl::EventId id) {
|
|
|
|
auto &tokenStore = getTokenStore();
|
|
|
|
auto iter = std::find_if(tokenStore.begin(), tokenStore.end(), [&](auto &item) {
|
|
|
|
return item.first == token && item.second->first == id;
|
|
|
|
});
|
|
|
|
|
|
|
|
if (iter != tokenStore.end()) {
|
|
|
|
getEvents().erase(iter->second);
|
|
|
|
tokenStore.erase(iter);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2020-12-22 18:10:01 +01:00
|
|
|
}
|