Update contrib.
1 #ifndef DATE_TIME_PERIOD_HPP___
2 #define DATE_TIME_PERIOD_HPP___
4 /* Copyright (c) 2002,2003 CrystalClear Software, Inc.
6 * Distributed under the Boost Software License, Version 1.0.
7 * (See accompanying file LICENSE_1_0.txt or copy at
8 * http://www.boost.org/LICENSE_1_0.txt)
11 * Author: Jeff Garland, Bart Garst
12 * $Date: 2006/07/17 03:56:05 $
16 This file contain the implementation of the period abstraction. This is
17 basically the same idea as a range. Although this class is intended for
18 use in the time library, it is pretty close to general enough for other
23 #include "boost/operators.hpp"
28 //!Provides generalized period type useful in date-time systems
29 /*!This template uses a class to represent a time point within the period
30 and another class to represent a duration. As a result, this class is
31 not appropriate for use when the number and duration representation
32 are the same (eg: in the regular number domain).
34 A period can be specified by providing either the begining point and
35 a duration or the begining point and the end point( end is NOT part
36 of the period but 1 unit past it. A period will be "invalid" if either
37 end_point <= begin_point or the given duration is <= 0. Any valid period
38 will return false for is_null().
40 Zero length periods are also considered invalid. Zero length periods are
41 periods where the begining and end points are the same, or, the given
42 duration is zero. For a zero length period, the last point will be one
43 unit less than the begining point.
45 In the case that the begin and last are the same, the period has a
48 The best way to handle periods is usually to provide a begining point and
49 a duration. So, day1 + 7 days is a week period which includes all of the
50 first day and 6 more days (eg: Sun to Sat).
53 template<class point_rep, class duration_rep>
54 class period : private
55 boost::less_than_comparable<period<point_rep, duration_rep>
56 , boost::equality_comparable< period<point_rep, duration_rep>
60 typedef point_rep point_type;
61 typedef duration_rep duration_type;
63 period(point_rep first_point, point_rep end_point);
64 period(point_rep first_point, duration_rep len);
65 point_rep begin() const;
66 point_rep end() const;
67 point_rep last() const;
68 duration_rep length() const;
70 bool operator==(const period& rhs) const;
71 bool operator<(const period& rhs) const;
72 void shift(const duration_rep& d);
73 bool contains(const point_rep& point) const;
74 bool contains(const period& other) const;
75 bool intersects(const period& other) const;
76 bool is_adjacent(const period& other) const;
77 bool is_before(const point_rep& point) const;
78 bool is_after(const point_rep& point) const;
79 period intersection(const period& other) const;
80 period merge(const period& other) const;
81 period span(const period& other) const;
87 //! create a period from begin to last eg: [begin,end)
88 /*! If end <= begin then the period will be invalid
90 template<class point_rep, class duration_rep>
92 period<point_rep,duration_rep>::period(point_rep first_point,
93 point_rep end_point) :
95 last_(end_point - duration_rep::unit())
98 //! create a period as [begin, begin+len)
99 /*! If len is <= 0 then the period will be invalid
101 template<class point_rep, class duration_rep>
103 period<point_rep,duration_rep>::period(point_rep first_point, duration_rep len) :
105 last_(first_point + len-duration_rep::unit())
109 //! Return the first element in the period
110 template<class point_rep, class duration_rep>
112 point_rep period<point_rep,duration_rep>::begin() const
117 //! Return one past the last element
118 template<class point_rep, class duration_rep>
120 point_rep period<point_rep,duration_rep>::end() const
122 return last_ + duration_rep::unit();
125 //! Return the last item in the period
126 template<class point_rep, class duration_rep>
128 point_rep period<point_rep,duration_rep>::last() const
133 //! True if period is ill formed (length is zero or less)
134 template<class point_rep, class duration_rep>
136 bool period<point_rep,duration_rep>::is_null() const
138 return end() <= begin_;
141 //! Return the length of the period
142 template<class point_rep, class duration_rep>
144 duration_rep period<point_rep,duration_rep>::length() const
146 if(last_ < begin_){ // invalid period
147 return last_+duration_rep::unit() - begin_;
150 return end() - begin_; // normal case
154 //! Equality operator
155 template<class point_rep, class duration_rep>
157 bool period<point_rep,duration_rep>::operator==(const period& rhs) const
159 return ((begin_ == rhs.begin_) &&
160 (last_ == rhs.last_));
163 //! Strict as defined by rhs.last <= lhs.last
164 template<class point_rep, class duration_rep>
166 bool period<point_rep,duration_rep>::operator<(const period& rhs) const
168 return (last_ < rhs.begin_);
172 //! Shift the start and end by the specified amount
173 template<class point_rep, class duration_rep>
175 void period<point_rep,duration_rep>::shift(const duration_rep& d)
181 //! True if the point is inside the period, zero length periods contain no points
182 template<class point_rep, class duration_rep>
184 bool period<point_rep,duration_rep>::contains(const point_rep& point) const
186 return ((point >= begin_) &&
191 //! True if this period fully contains (or equals) the other period
192 template<class point_rep, class duration_rep>
194 bool period<point_rep,duration_rep>::contains(const period<point_rep,duration_rep>& other) const
196 return ((begin_ <= other.begin_) && (last_ >= other.last_));
200 //! True if periods are next to each other without a gap.
201 /* In the example below, p1 and p2 are adjacent, but p3 is not adjacent
202 * with either of p1 or p2.
209 template<class point_rep, class duration_rep>
212 period<point_rep,duration_rep>::is_adjacent(const period<point_rep,duration_rep>& other) const
214 return (other.begin() == end() ||
215 begin_ == other.end());
219 //! True if all of the period is prior or t < start
220 /* In the example below only point 1 would evaluate to true.
228 template<class point_rep, class duration_rep>
231 period<point_rep,duration_rep>::is_after(const point_rep& t) const
235 return false; //null period isn't after
241 //! True if all of the period is prior to the passed point or end <= t
242 /* In the example below points 4 and 5 return true.
250 template<class point_rep, class duration_rep>
253 period<point_rep,duration_rep>::is_before(const point_rep& t) const
257 return false; //null period isn't before anything
264 //! True if the periods overlap in any way
265 /* In the example below p1 intersects with p2, p4, and p6.
275 template<class point_rep, class duration_rep>
277 bool period<point_rep,duration_rep>::intersects(const period<point_rep,duration_rep>& other) const
279 return ( contains(other.begin_) ||
280 other.contains(begin_) ||
281 ((other.begin_ < begin_) && (other.last_ >= begin_)));
284 //! Returns the period of intersection or invalid range no intersection
285 template<class point_rep, class duration_rep>
287 period<point_rep,duration_rep>
288 period<point_rep,duration_rep>::intersection(const period<point_rep,duration_rep>& other) const
290 if (begin_ > other.begin_) {
291 if (last_ <= other.last_) { //case2
295 return period<point_rep,duration_rep>(begin_, other.end());
298 if (last_ <= other.last_) { //case3
299 return period<point_rep,duration_rep>(other.begin_, this->end());
307 //! Returns the union of intersecting periods -- or null period
310 template<class point_rep, class duration_rep>
312 period<point_rep,duration_rep>
313 period<point_rep,duration_rep>::merge(const period<point_rep,duration_rep>& other) const
315 if (this->intersects(other)) {
316 if (begin_ < other.begin_) {
317 return period<point_rep,duration_rep>(begin_, last_ > other.last_ ? this->end() : other.end());
320 return period<point_rep,duration_rep>(other.begin_, last_ > other.last_ ? this->end() : other.end());
323 return period<point_rep,duration_rep>(begin_,begin_); // no intersect return null
326 //! Combine two periods with earliest start and latest end.
327 /*! Combines two periods and any gap between them such that
328 * start = minimum(p1.start, p2.start)
329 * end = maximum(p1.end , p2.end)
334 * [-----------p3----------)
337 template<class point_rep, class duration_rep>
339 period<point_rep,duration_rep>
340 period<point_rep,duration_rep>::span(const period<point_rep,duration_rep>& other) const
342 point_rep start((begin_ < other.begin_) ? begin() : other.begin());
343 point_rep newend((last_ < other.last_) ? other.end() : this->end());
344 return period<point_rep,duration_rep>(start, newend);
348 } } //namespace date_time