slot.hpp 5.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157
  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_SLOT_HEADER
  8. #define BOOST_SIGNALS_SLOT_HEADER
  9. #include <boost/signals/detail/signals_common.hpp>
  10. #include <boost/signals/connection.hpp>
  11. #include <boost/signals/trackable.hpp>
  12. #include <boost/visit_each.hpp>
  13. #include <boost/shared_ptr.hpp>
  14. #include <cassert>
  15. #ifdef BOOST_HAS_ABI_HEADERS
  16. # include BOOST_ABI_PREFIX
  17. #endif
  18. namespace boost {
  19. namespace BOOST_SIGNALS_NAMESPACE {
  20. namespace detail {
  21. class BOOST_SIGNALS_DECL slot_base {
  22. // We would have to enumerate all of the signalN classes here as
  23. // friends to make this private (as it otherwise should be). We can't
  24. // name all of them because we don't know how many there are.
  25. public:
  26. struct data_t {
  27. std::vector<const trackable*> bound_objects;
  28. connection watch_bound_objects;
  29. };
  30. shared_ptr<data_t> get_data() const { return data; }
  31. // Get the set of bound objects
  32. std::vector<const trackable*>& get_bound_objects() const
  33. { return data->bound_objects; }
  34. // Determine if this slot is still "active", i.e., all of the bound
  35. // objects still exist
  36. bool is_active() const
  37. { return data->watch_bound_objects.connected(); }
  38. protected:
  39. // Create a connection for this slot
  40. void create_connection();
  41. shared_ptr<data_t> data;
  42. private:
  43. static void bound_object_destructed(void*, void*) {}
  44. };
  45. } // end namespace detail
  46. // Get the slot so that it can be copied
  47. template<typename F>
  48. reference_wrapper<const F>
  49. get_invocable_slot(const F& f, BOOST_SIGNALS_NAMESPACE::detail::signal_tag)
  50. { return reference_wrapper<const F>(f); }
  51. template<typename F>
  52. const F&
  53. get_invocable_slot(const F& f, BOOST_SIGNALS_NAMESPACE::detail::reference_tag)
  54. { return f; }
  55. template<typename F>
  56. const F&
  57. get_invocable_slot(const F& f, BOOST_SIGNALS_NAMESPACE::detail::value_tag)
  58. { return f; }
  59. // Get the slot so that it can be inspected for trackable objects
  60. template<typename F>
  61. const F&
  62. get_inspectable_slot(const F& f, BOOST_SIGNALS_NAMESPACE::detail::signal_tag)
  63. { return f; }
  64. template<typename F>
  65. const F&
  66. get_inspectable_slot(const reference_wrapper<F>& f, BOOST_SIGNALS_NAMESPACE::detail::reference_tag)
  67. { return f.get(); }
  68. template<typename F>
  69. const F&
  70. get_inspectable_slot(const F& f, BOOST_SIGNALS_NAMESPACE::detail::value_tag)
  71. { return f; }
  72. // Determines the type of the slot - is it a signal, a reference to a
  73. // slot or just a normal slot.
  74. template<typename F>
  75. typename BOOST_SIGNALS_NAMESPACE::detail::get_slot_tag<F>::type
  76. tag_type(const F&)
  77. {
  78. typedef typename BOOST_SIGNALS_NAMESPACE::detail::get_slot_tag<F>::type
  79. the_tag_type;
  80. the_tag_type tag = the_tag_type();
  81. return tag;
  82. }
  83. } // end namespace BOOST_SIGNALS_NAMESPACE
  84. template<typename SlotFunction>
  85. class slot : public BOOST_SIGNALS_NAMESPACE::detail::slot_base {
  86. typedef BOOST_SIGNALS_NAMESPACE::detail::slot_base inherited;
  87. typedef typename inherited::data_t data_t;
  88. public:
  89. template<typename F>
  90. slot(const F& f) : slot_function(BOOST_SIGNALS_NAMESPACE::get_invocable_slot(f, BOOST_SIGNALS_NAMESPACE::tag_type(f)))
  91. {
  92. this->data.reset(new data_t);
  93. // Visit each of the bound objects and store them for later use
  94. // An exception thrown here will allow the basic_connection to be
  95. // destroyed when this goes out of scope, and no other connections
  96. // have been made.
  97. BOOST_SIGNALS_NAMESPACE::detail::bound_objects_visitor
  98. do_bind(this->data->bound_objects);
  99. visit_each(do_bind,
  100. BOOST_SIGNALS_NAMESPACE::get_inspectable_slot
  101. (f, BOOST_SIGNALS_NAMESPACE::tag_type(f)));
  102. create_connection();
  103. }
  104. #ifdef __BORLANDC__
  105. template<typename F>
  106. slot(F* f) : slot_function(f)
  107. {
  108. this->data.reset(new data_t);
  109. create_connection();
  110. }
  111. #endif // __BORLANDC__
  112. // We would have to enumerate all of the signalN classes here as friends
  113. // to make this private (as it otherwise should be). We can't name all of
  114. // them because we don't know how many there are.
  115. public:
  116. // Get the slot function to call the actual slot
  117. const SlotFunction& get_slot_function() const { return slot_function; }
  118. void release() const { data->watch_bound_objects.set_controlling(false); }
  119. private:
  120. slot(); // no default constructor
  121. slot& operator=(const slot&); // no assignment operator
  122. SlotFunction slot_function;
  123. };
  124. } // end namespace boost
  125. #ifdef BOOST_HAS_ABI_HEADERS
  126. # include BOOST_ABI_SUFFIX
  127. #endif
  128. #endif // BOOST_SIGNALS_SLOT_HEADER