From 6a96bcf1659d07cd3a7f7e65519007abfaeda2a0 Mon Sep 17 00:00:00 2001 From: Bent Bisballe Nyeng Date: Sat, 15 Jul 2017 11:59:10 +0200 Subject: Ensure that the Notifier call order is the same as the connection order. --- src/notifier.h | 18 ++++++--- test/Makefile.am | 9 ++++- test/notifiertest.cc | 103 +++++++++++++++++++++++++++++++++++++++++++++++++++ 3 files changed, 123 insertions(+), 7 deletions(-) create mode 100644 test/notifiertest.cc diff --git a/src/notifier.h b/src/notifier.h index ff59741..f044d98 100644 --- a/src/notifier.h +++ b/src/notifier.h @@ -27,10 +27,9 @@ #pragma once #include -#include -#include +#include +#include #include -#include namespace aux { @@ -124,7 +123,7 @@ public: template void connect(O* p, const F& fn) { - slots[p] = std::move(construct_mem_fn(fn, p, aux::gen_int_sequence{})); + slots.emplace_back(std::make_pair(p, std::move(construct_mem_fn(fn, p, aux::gen_int_sequence{})))); if(p && dynamic_cast(p)) { dynamic_cast(p)->registerNotifier(this); @@ -134,7 +133,14 @@ public: //! \brief Disconnect object from this Notifier. void disconnect(Listener* object) { - slots.erase(object); + for(auto it = slots.begin(); it != slots.end(); ++it) + { + if(it->first == object) + { + slots.erase(it); + return; + } + } } //! \brief Activate this notifier by pretending it is a function. @@ -148,7 +154,7 @@ public: } private: - std::map slots; + std::list> slots; template callback_type construct_mem_fn(const F& fn, O* p, aux::int_sequence) const diff --git a/test/Makefile.am b/test/Makefile.am index 08ef4fa..871b3b4 100644 --- a/test/Makefile.am +++ b/test/Makefile.am @@ -6,7 +6,7 @@ if ENABLE_TESTS TESTS = resource enginetest paintertest resampler configfile audiocache \ audiocachefile audiocacheidmanager audiocacheeventhandler \ memchecker randomtest atomictest syncedsettingstest imagecachetest \ - semaphoretest drumkitcreatortest bytesizeparsertest + semaphoretest drumkitcreatortest bytesizeparsertest notifiertest if ENABLE_LV2 TESTS += lv2 @@ -205,4 +205,11 @@ bytesizeparsertest_SOURCES = \ bytesizeparsertest.cc \ test.cc +notifiertest_CXXFLAGS = -DOUTPUT=\"notifiertest\" $(CPPUNIT_CFLAGS) \ + -I$(top_srcdir)/src +notifiertest_LDFLAGS = $(CPPUNIT_LIBS) +notifiertest_SOURCES = \ + notifiertest.cc \ + test.cc + endif diff --git a/test/notifiertest.cc b/test/notifiertest.cc new file mode 100644 index 0000000..2b2e92e --- /dev/null +++ b/test/notifiertest.cc @@ -0,0 +1,103 @@ +/* -*- Mode: c++ -*- */ +/*************************************************************************** + * notifiertest.cc + * + * Sat Jul 15 09:56:51 CEST 2017 + * Copyright 2017 Bent Bisballe Nyeng + * deva@aasimon.org + ****************************************************************************/ + +/* + * This file is part of DrumGizmo. + * + * DrumGizmo is free software; you can redistribute it and/or modify + * it under the terms of the GNU Lesser General Public License as published by + * the Free Software Foundation; either version 3 of the License, or + * (at your option) any later version. + * + * DrumGizmo is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public License + * along with DrumGizmo; if not, write to the Free Software + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA. + */ +#include + +#include + +#include + +class Probe + : public Listener +{ +public: + Probe(std::vector& triggers) + : triggers(triggers) + { + } + + void slot() + { + triggers.push_back(this); + } + + std::vector& triggers; +}; + +class NotifierTest + : public CppUnit::TestFixture +{ + CPPUNIT_TEST_SUITE(NotifierTest); + CPPUNIT_TEST(testTest); + CPPUNIT_TEST_SUITE_END(); + +public: + void setUp() { } + void tearDown() { } + + //! This just creates some drumkit. + void testTest() + { + Notifier<> notifier; + std::vector triggers; + Probe foo1(triggers); + Probe foo2(triggers); + + { // Order as initialisation + notifier.connect(&foo1, &Probe::slot); + notifier.connect(&foo2, &Probe::slot); + notifier(); + std::vector ref; + ref.push_back(&foo1); + ref.push_back(&foo2); + CPPUNIT_ASSERT_EQUAL(ref.size(), triggers.size()); + CPPUNIT_ASSERT_EQUAL(ref[0], triggers[0]); + CPPUNIT_ASSERT_EQUAL(ref[1], triggers[1]); + notifier.disconnect(&foo1); + notifier.disconnect(&foo2); + triggers.clear(); + } + + { // Reverse order + notifier.connect(&foo2, &Probe::slot); + notifier.connect(&foo1, &Probe::slot); + notifier(); + std::vector ref; + ref.push_back(&foo2); + ref.push_back(&foo1); + CPPUNIT_ASSERT_EQUAL(ref.size(), triggers.size()); + CPPUNIT_ASSERT_EQUAL(ref[0], triggers[0]); + CPPUNIT_ASSERT_EQUAL(ref[1], triggers[1]); + notifier.disconnect(&foo1); + notifier.disconnect(&foo2); + triggers.clear(); + } + + } +}; + +// Registers the fixture into the 'registry' +CPPUNIT_TEST_SUITE_REGISTRATION(NotifierTest); -- cgit v1.2.3