connection.hpp 6.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213
  1. // Boost.Signals library
  2. // Copyright Douglas Gregor 2001-2004. Use, modification and
  3. // distribution is subject to the Boost Software License, Version
  4. // 1.0. (See accompanying file LICENSE_1_0.txt or copy at
  5. // http://www.boost.org/LICENSE_1_0.txt)
  6. // For more information, see http://www.boost.org
  7. #ifndef BOOST_SIGNALS_CONNECTION_HPP
  8. #define BOOST_SIGNALS_CONNECTION_HPP
  9. #include <boost/signals/detail/signals_common.hpp>
  10. #include <boost/smart_ptr.hpp>
  11. #include <boost/operators.hpp>
  12. #include <boost/any.hpp>
  13. #include <list>
  14. #include <cassert>
  15. #include <utility>
  16. #ifdef BOOST_HAS_ABI_HEADERS
  17. # include BOOST_ABI_PREFIX
  18. #endif
  19. namespace boost {
  20. namespace BOOST_SIGNALS_NAMESPACE {
  21. class trackable;
  22. namespace detail {
  23. // Represents an object that has been bound as part of a slot, and how
  24. // to notify that object of a disconnect
  25. struct bound_object {
  26. void* obj;
  27. void* data;
  28. void (*disconnect)(void*, void*);
  29. bool operator==(const bound_object& other) const
  30. { return obj == other.obj && data == other.data; }
  31. bool operator<(const bound_object& other) const
  32. { return obj < other.obj; }
  33. // To support intel 80 compiler, 2004/03/18 (Mark Rodgers)
  34. bool operator!=(const bound_object& other) const
  35. { return !(*this==other); }
  36. bool operator>(const bound_object& other) const
  37. { return !(*this < other); }
  38. };
  39. // Describes the connection between a signal and the objects that are
  40. // bound for a specific slot. Enables notification of the signal and the
  41. // slots when a disconnect is requested.
  42. struct basic_connection {
  43. void* signal;
  44. void* signal_data;
  45. void (*signal_disconnect)(void*, void*);
  46. bool blocked_;
  47. std::list<bound_object> bound_objects;
  48. };
  49. } // end namespace detail
  50. // The user may freely pass around the "connection" object and terminate
  51. // the connection at any time using disconnect().
  52. class BOOST_SIGNALS_DECL connection :
  53. private less_than_comparable1<connection>,
  54. private equality_comparable1<connection>
  55. {
  56. public:
  57. connection() : con(), controlling_connection(false) {}
  58. connection(const connection&);
  59. ~connection();
  60. // Block he connection: if the connection is still active, there
  61. // will be no notification
  62. void block(bool should_block = true) { con->blocked_ = should_block; }
  63. void unblock() { con->blocked_ = false; }
  64. bool blocked() const { return !connected() || con->blocked_; }
  65. // Disconnect the signal and slot, if they are connected
  66. void disconnect() const;
  67. // Returns true if the signal and slot are connected
  68. bool connected() const { return con.get() && con->signal_disconnect; }
  69. // Comparison of connections
  70. bool operator==(const connection& other) const;
  71. bool operator<(const connection& other) const;
  72. // Connection assignment
  73. connection& operator=(const connection& other) ;
  74. // Swap connections
  75. void swap(connection& other);
  76. public: // TBD: CHANGE THIS
  77. // Set whether this connection object is controlling or not
  78. void set_controlling(bool control = true)
  79. { controlling_connection = control; }
  80. shared_ptr<BOOST_SIGNALS_NAMESPACE::detail::basic_connection>
  81. get_connection() const
  82. { return con; }
  83. private:
  84. friend class detail::signal_base_impl;
  85. friend class detail::slot_base;
  86. friend class trackable;
  87. // Reset this connection to refer to a different actual connection
  88. void reset(BOOST_SIGNALS_NAMESPACE::detail::basic_connection*);
  89. // Add a bound object to this connection (not for users)
  90. void add_bound_object(const BOOST_SIGNALS_NAMESPACE::detail::bound_object& b);
  91. friend class BOOST_SIGNALS_NAMESPACE::detail::bound_objects_visitor;
  92. // Pointer to the actual contents of the connection
  93. shared_ptr<BOOST_SIGNALS_NAMESPACE::detail::basic_connection> con;
  94. // True if the destruction of this connection object should disconnect
  95. bool controlling_connection;
  96. };
  97. // Similar to connection, but will disconnect the connection when it is
  98. // destroyed unless release() has been called.
  99. class BOOST_SIGNALS_DECL scoped_connection : public connection {
  100. public:
  101. scoped_connection() : connection(), released(false) {}
  102. scoped_connection(const connection&);
  103. scoped_connection(const scoped_connection&);
  104. ~scoped_connection();
  105. connection release();
  106. void swap(scoped_connection&);
  107. scoped_connection& operator=(const connection&);
  108. scoped_connection& operator=(const scoped_connection&);
  109. private:
  110. bool released;
  111. };
  112. namespace detail {
  113. struct connection_slot_pair {
  114. connection first;
  115. any second;
  116. connection_slot_pair() {}
  117. connection_slot_pair(const connection& c, const any& a)
  118. : first(c), second(a)
  119. {
  120. }
  121. // Dummys to allow explicit instantiation to work
  122. bool operator==(const connection_slot_pair&) const { return false; }
  123. bool operator<(const connection_slot_pair&) const { return false;}
  124. };
  125. // Determines if the underlying connection is disconnected
  126. struct is_disconnected {
  127. typedef connection_slot_pair argument_type;
  128. typedef bool result_type;
  129. inline bool operator()(const argument_type& c) const
  130. {
  131. return !c.first.connected();
  132. }
  133. };
  134. // Determines if the underlying connection is callable, ie if
  135. // it is connected and not blocked
  136. struct is_callable {
  137. typedef connection_slot_pair argument_type;
  138. typedef bool result_type;
  139. inline bool operator()(const argument_type& c) const
  140. {
  141. return c.first.connected() && !c.first.blocked() ;
  142. }
  143. };
  144. // Autodisconnects the bound object when it is destroyed unless the
  145. // release method is invoked.
  146. class auto_disconnect_bound_object {
  147. public:
  148. auto_disconnect_bound_object(const bound_object& b) :
  149. binding(b), auto_disconnect(true)
  150. {
  151. }
  152. ~auto_disconnect_bound_object()
  153. {
  154. if (auto_disconnect)
  155. binding.disconnect(binding.obj, binding.data);
  156. }
  157. void release() { auto_disconnect = false; }
  158. private:
  159. bound_object binding;
  160. bool auto_disconnect;
  161. };
  162. } // end namespace detail
  163. } // end namespace BOOST_SIGNALS_NAMESPACE
  164. } // end namespace boost
  165. #ifdef BOOST_HAS_ABI_HEADERS
  166. # include BOOST_ABI_SUFFIX
  167. #endif
  168. #endif // BOOST_SIGNALS_CONNECTION_HPP