// Boost.Signals2 library // Copyright Douglas Gregor 2001-2006. // Copyright Frank Mori Hess 2009. // Use, modification and // distribution is subject to the Boost Software License, Version // 1.0. (See accompanying file LICENSE_1_0.txt or copy at // http://www.boost.org/LICENSE_1_0.txt) // For more information, see http://www.boost.org #include #include #define BOOST_TEST_MODULE trackable_test #include #include #include #include using namespace boost::placeholders; struct short_lived : public boost::signals2::trackable { ~short_lived() {} }; struct swallow { typedef int result_type; template int operator()(const T*, int i) { return i; } template int operator()(T &, int i) { return i; } template int operator()(boost::weak_ptr, int i) { return i; } }; template struct max_or_default { typedef T result_type; template T operator()(InputIterator first, InputIterator last) const { if (first == last) return T(); T max = *first++; for (; first != last; ++first) max = (*first > max)? *first : max; return max; } }; struct self_deleting : public boost::signals2::trackable { void delete_myself(boost::signals2::connection connection) { BOOST_CHECK(connection.connected()); delete this; BOOST_CHECK(connection.connected() == false); } }; // test that slot assocated with signals2::trackable // gets disconnected immediately upon deletion of the // signals2::trackable, even when a signal invocation // is in progress. void test_immediate_disconnect_on_delete() { boost::signals2::signal sig; self_deleting *obj = new self_deleting(); sig.connect_extended(boost::bind(&self_deleting::delete_myself, obj, _1)); sig(); } BOOST_AUTO_TEST_CASE(test_main) { typedef boost::signals2::signal > sig_type; sig_type s1; // Test auto-disconnection BOOST_CHECK(s1(5) == 0); { short_lived shorty; s1.connect(boost::bind(swallow(), &shorty, _1)); BOOST_CHECK(s1(5) == 5); } BOOST_CHECK(s1(5) == 0); // Test auto-disconnection of trackable inside reference_wrapper { short_lived shorty; s1.connect(boost::bind(swallow(), boost::ref(shorty), _1)); BOOST_CHECK(s1(5) == 5); } BOOST_CHECK(s1(5) == 0); // Test multiple arg slot constructor { short_lived shorty; s1.connect(sig_type::slot_type(swallow(), &shorty, _1)); BOOST_CHECK(s1(5) == 5); } BOOST_CHECK(s1(5) == 0); // Test auto-disconnection of slot before signal connection { short_lived* shorty = new short_lived(); sig_type::slot_type slot(boost::bind(swallow(), shorty, _1)); delete shorty; BOOST_CHECK(s1(5) == 0); } test_immediate_disconnect_on_delete(); }