Core/Signal: Fix disconnection while iterating
This is no longer an issue, you can now disconnect every signal as you wish even while the iteration Former-commit-id: 5caadaa860c62ce09f02f7555aa7c6adebc74fb9
This commit is contained in:
parent
ec9876011f
commit
5ce3c5a0cb
|
|
@ -26,7 +26,7 @@ class NzSignal
|
||||||
class Connection;
|
class Connection;
|
||||||
class ConnectionGuard;
|
class ConnectionGuard;
|
||||||
|
|
||||||
NzSignal() = default;
|
NzSignal();
|
||||||
NzSignal(const NzSignal&) = delete;
|
NzSignal(const NzSignal&) = delete;
|
||||||
NzSignal(NzSignal&& signal);
|
NzSignal(NzSignal&& signal);
|
||||||
~NzSignal() = default;
|
~NzSignal() = default;
|
||||||
|
|
@ -50,6 +50,7 @@ class NzSignal
|
||||||
|
|
||||||
using SlotPtr = std::shared_ptr<Slot>;
|
using SlotPtr = std::shared_ptr<Slot>;
|
||||||
using SlotList = std::vector<SlotPtr>;
|
using SlotList = std::vector<SlotPtr>;
|
||||||
|
using SlotListIndex = typename SlotList::size_type;
|
||||||
|
|
||||||
struct Slot
|
struct Slot
|
||||||
{
|
{
|
||||||
|
|
@ -60,12 +61,13 @@ class NzSignal
|
||||||
|
|
||||||
Callback callback;
|
Callback callback;
|
||||||
NzSignal* signal;
|
NzSignal* signal;
|
||||||
typename SlotList::size_type index;
|
SlotListIndex index;
|
||||||
};
|
};
|
||||||
|
|
||||||
void Disconnect(const SlotPtr& slot);
|
void Disconnect(const SlotPtr& slot);
|
||||||
|
|
||||||
SlotList m_slots;
|
SlotList m_slots;
|
||||||
|
mutable SlotListIndex m_slotIterator;
|
||||||
};
|
};
|
||||||
|
|
||||||
template<typename... Args>
|
template<typename... Args>
|
||||||
|
|
|
||||||
|
|
@ -6,6 +6,12 @@
|
||||||
#include <utility>
|
#include <utility>
|
||||||
#include <Nazara/Core/Debug.hpp>
|
#include <Nazara/Core/Debug.hpp>
|
||||||
|
|
||||||
|
template<typename... Args>
|
||||||
|
NzSignal<Args...>::NzSignal() :
|
||||||
|
m_slotIterator(0)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
template<typename... Args>
|
template<typename... Args>
|
||||||
NzSignal<Args...>::NzSignal(NzSignal&& signal)
|
NzSignal<Args...>::NzSignal(NzSignal&& signal)
|
||||||
{
|
{
|
||||||
|
|
@ -29,11 +35,15 @@ typename NzSignal<Args...>::Connection NzSignal<Args...>::Connect(Callback&& fun
|
||||||
{
|
{
|
||||||
NazaraAssert(func, "Invalid function");
|
NazaraAssert(func, "Invalid function");
|
||||||
|
|
||||||
|
bool resetIt = (m_slotIterator >= m_slots.size());
|
||||||
|
|
||||||
auto tempPtr = std::make_shared<Slot>(this);
|
auto tempPtr = std::make_shared<Slot>(this);
|
||||||
tempPtr->callback = std::move(func);
|
tempPtr->callback = std::move(func);
|
||||||
tempPtr->index = m_slots.size();
|
tempPtr->index = m_slots.size();
|
||||||
|
|
||||||
m_slots.emplace_back(std::move(tempPtr));
|
m_slots.emplace_back(std::move(tempPtr));
|
||||||
|
if (resetIt)
|
||||||
|
m_slotIterator = m_slots.size();
|
||||||
|
|
||||||
return Connection(m_slots.back());
|
return Connection(m_slots.back());
|
||||||
}
|
}
|
||||||
|
|
@ -81,14 +91,15 @@ typename NzSignal<Args...>::Connection NzSignal<Args...>::Connect(const O* objec
|
||||||
template<typename... Args>
|
template<typename... Args>
|
||||||
void NzSignal<Args...>::operator()(Args... args) const
|
void NzSignal<Args...>::operator()(Args... args) const
|
||||||
{
|
{
|
||||||
for (const SlotPtr& slot : m_slots)
|
for (m_slotIterator = 0; m_slotIterator < m_slots.size(); ++m_slotIterator)
|
||||||
slot->callback(args...);
|
m_slots[m_slotIterator]->callback(args...);
|
||||||
}
|
}
|
||||||
|
|
||||||
template<typename... Args>
|
template<typename... Args>
|
||||||
NzSignal<Args...>& NzSignal<Args...>::operator=(NzSignal&& signal)
|
NzSignal<Args...>& NzSignal<Args...>::operator=(NzSignal&& signal)
|
||||||
{
|
{
|
||||||
m_slots = std::move(signal.m_slots);
|
m_slots = std::move(signal.m_slots);
|
||||||
|
m_slotIterator = signal.m_slotIterator;
|
||||||
|
|
||||||
// We need to update the signal pointer inside of each slot
|
// We need to update the signal pointer inside of each slot
|
||||||
for (SlotPtr& slot : m_slots)
|
for (SlotPtr& slot : m_slots)
|
||||||
|
|
@ -105,12 +116,32 @@ void NzSignal<Args...>::Disconnect(const SlotPtr& slot)
|
||||||
|
|
||||||
// "Swap this slot with the last one and pop" idiom
|
// "Swap this slot with the last one and pop" idiom
|
||||||
// This will preserve slot indexes
|
// This will preserve slot indexes
|
||||||
SlotPtr& current = m_slots[slot->index];
|
|
||||||
|
|
||||||
std::swap(current, m_slots.back());
|
// Can we safely "remove" this slot?
|
||||||
|
if (m_slotIterator >= m_slots.size()-1 || slot->index > m_slotIterator)
|
||||||
|
{
|
||||||
|
// Yes we can
|
||||||
|
SlotPtr& newSlot = m_slots[slot->index];
|
||||||
|
newSlot = std::move(m_slots.back());
|
||||||
|
newSlot->index = slot->index; //< Update the moved slot index before resizing (imagine this is the last one)
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
// Nope, let's be tricky
|
||||||
|
SlotPtr& current = m_slots[m_slotIterator];
|
||||||
|
SlotPtr& newSlot = m_slots[slot->index];
|
||||||
|
|
||||||
|
newSlot = std::move(current);
|
||||||
|
newSlot->index = slot->index; //< Update the moved slot index
|
||||||
|
|
||||||
|
current = std::move(m_slots.back());
|
||||||
|
current->index = m_slotIterator; //< Update the moved slot index
|
||||||
|
|
||||||
|
--m_slotIterator;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Pop the last entry (where we moved our slot)
|
||||||
m_slots.pop_back();
|
m_slots.pop_back();
|
||||||
|
|
||||||
current->index = slot->index; //< Update the moved slot index
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
|
||||||
Loading…
Reference in New Issue