#pragma once #include #include #include #include #include #include "util/assertinguithread.hpp" namespace chatterino { namespace util { template class ReadOnlySignalVector : boost::noncopyable { public: ReadOnlySignalVector() { QObject::connect(&this->itemsChangedTimer, &QTimer::timeout, [this] { this->delayedItemsChanged.invoke(); }); this->itemsChangedTimer.setInterval(100); this->itemsChangedTimer.setSingleShot(true); } virtual ~ReadOnlySignalVector() = default; struct ItemArgs { const TVectorItem &item; int index; void *caller; }; pajlada::Signals::Signal itemInserted; pajlada::Signals::Signal itemRemoved; pajlada::Signals::NoArgSignal delayedItemsChanged; const std::vector &getVector() const { util::assertInGuiThread(); return this->vector; } void invokeDelayedItemsChanged() { util::assertInGuiThread(); if (!this->itemsChangedTimer.isActive()) { itemsChangedTimer.start(); } } protected: std::vector vector; QTimer itemsChangedTimer; }; template class BaseSignalVector : public ReadOnlySignalVector { public: // returns the actual index of the inserted item virtual int insertItem(const TVectorItem &item, int proposedIndex = -1, void *caller = 0) = 0; void removeItem(int index, void *caller = 0) { util::assertInGuiThread(); assert(index >= 0 && index < this->vector.size()); TVectorItem item = this->vector[index]; this->vector.erase(this->vector.begin() + index); typename ReadOnlySignalVector::ItemArgs args{item, index, caller}; this->itemRemoved.invoke(args); this->invokeDelayedItemsChanged(); } int appendItem(const TVectorItem &item, void *caller = 0) { return this->insertItem(item, -1, caller); } }; template class UnsortedSignalVector : public BaseSignalVector { public: virtual int insertItem(const TVectorItem &item, int index = -1, void *caller = 0) override { util::assertInGuiThread(); if (index == -1) { index = this->vector.size(); } else { assert(index >= 0 && index <= this->vector.size()); } this->vector.insert(this->vector.begin() + index, item); typename ReadOnlySignalVector::ItemArgs args{item, index, caller}; this->itemInserted.invoke(args); this->invokeDelayedItemsChanged(); return index; } }; template class SortedSignalVector : public BaseSignalVector { public: virtual int insertItem(const TVectorItem &item, int index = -1, void *caller = 0) override { util::assertInGuiThread(); int index = this->vector.insert( std::lower_bound(this->vector.begin(), this->vector.end(), item), item) - this->vector.begin(); typename ReadOnlySignalVector::ItemArgs args{item, index, caller}; this->itemInserted.invoke(args); this->invokeDelayedItemsChanged(); return index; } }; } // namespace util } // namespace chatterino