blob: 51cc541a4edee3679bbc1b04f49499cc4ab17e00 (
plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
|
// last_value function object (documented as part of Boost.Signals)
// Copyright Frank Mori Hess 2007.
// Copyright Douglas Gregor 2001-2003. 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
#ifndef BOOST_SIGNALS2_LAST_VALUE_HPP
#define BOOST_SIGNALS2_LAST_VALUE_HPP
#include <boost/optional.hpp>
#include <boost/signals2/expired_slot.hpp>
#include <boost/throw_exception.hpp>
#include <stdexcept>
namespace boost {
namespace signals2 {
// no_slots_error is thrown when we are unable to generate a return value
// due to no slots being connected to the signal.
class no_slots_error: public std::exception
{
public:
virtual const char* what() const throw() {return "boost::signals2::no_slots_error";}
};
template<typename T>
class last_value {
public:
typedef T result_type;
template<typename InputIterator>
T operator()(InputIterator first, InputIterator last) const
{
if(first == last)
{
boost::throw_exception(no_slots_error());
}
optional<T> value;
while (first != last)
{
try
{
value = *first;
}
catch(const expired_slot &) {}
++first;
}
if(value) return value.get();
boost::throw_exception(no_slots_error());
}
};
template<>
class last_value<void> {
public:
typedef void result_type;
template<typename InputIterator>
result_type operator()(InputIterator first, InputIterator last) const
{
while (first != last)
{
try
{
*first;
}
catch(const expired_slot &) {}
++first;
}
return;
}
};
} // namespace signals2
} // namespace boost
#endif // BOOST_SIGNALS2_LAST_VALUE_HPP
|