period.hpp 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377
  1. #ifndef DATE_TIME_PERIOD_HPP___
  2. #define DATE_TIME_PERIOD_HPP___
  3. /* Copyright (c) 2002,2003 CrystalClear Software, Inc.
  4. * Use, modification and distribution is subject to the
  5. * Boost Software License, Version 1.0. (See accompanying
  6. * file LICENSE_1_0.txt or http://www.boost.org/LICENSE_1_0.txt)
  7. * Author: Jeff Garland, Bart Garst
  8. * $Date$
  9. */
  10. /*! \file period.hpp
  11. This file contain the implementation of the period abstraction. This is
  12. basically the same idea as a range. Although this class is intended for
  13. use in the time library, it is pretty close to general enough for other
  14. numeric uses.
  15. */
  16. #include "boost/operators.hpp"
  17. namespace boost {
  18. namespace date_time {
  19. //!Provides generalized period type useful in date-time systems
  20. /*!This template uses a class to represent a time point within the period
  21. and another class to represent a duration. As a result, this class is
  22. not appropriate for use when the number and duration representation
  23. are the same (eg: in the regular number domain).
  24. A period can be specified by providing either the begining point and
  25. a duration or the begining point and the end point( end is NOT part
  26. of the period but 1 unit past it. A period will be "invalid" if either
  27. end_point <= begin_point or the given duration is <= 0. Any valid period
  28. will return false for is_null().
  29. Zero length periods are also considered invalid. Zero length periods are
  30. periods where the begining and end points are the same, or, the given
  31. duration is zero. For a zero length period, the last point will be one
  32. unit less than the begining point.
  33. In the case that the begin and last are the same, the period has a
  34. length of one unit.
  35. The best way to handle periods is usually to provide a begining point and
  36. a duration. So, day1 + 7 days is a week period which includes all of the
  37. first day and 6 more days (eg: Sun to Sat).
  38. */
  39. template<class point_rep, class duration_rep>
  40. class period : private
  41. boost::less_than_comparable<period<point_rep, duration_rep>
  42. , boost::equality_comparable< period<point_rep, duration_rep>
  43. > >
  44. {
  45. public:
  46. typedef point_rep point_type;
  47. typedef duration_rep duration_type;
  48. period(point_rep first_point, point_rep end_point);
  49. period(point_rep first_point, duration_rep len);
  50. point_rep begin() const;
  51. point_rep end() const;
  52. point_rep last() const;
  53. duration_rep length() const;
  54. bool is_null() const;
  55. bool operator==(const period& rhs) const;
  56. bool operator<(const period& rhs) const;
  57. void shift(const duration_rep& d);
  58. void expand(const duration_rep& d);
  59. bool contains(const point_rep& point) const;
  60. bool contains(const period& other) const;
  61. bool intersects(const period& other) const;
  62. bool is_adjacent(const period& other) const;
  63. bool is_before(const point_rep& point) const;
  64. bool is_after(const point_rep& point) const;
  65. period intersection(const period& other) const;
  66. period merge(const period& other) const;
  67. period span(const period& other) const;
  68. private:
  69. point_rep begin_;
  70. point_rep last_;
  71. };
  72. //! create a period from begin to last eg: [begin,end)
  73. /*! If end <= begin then the period will be invalid
  74. */
  75. template<class point_rep, class duration_rep>
  76. inline
  77. period<point_rep,duration_rep>::period(point_rep first_point,
  78. point_rep end_point) :
  79. begin_(first_point),
  80. last_(end_point - duration_rep::unit())
  81. {}
  82. //! create a period as [begin, begin+len)
  83. /*! If len is <= 0 then the period will be invalid
  84. */
  85. template<class point_rep, class duration_rep>
  86. inline
  87. period<point_rep,duration_rep>::period(point_rep first_point, duration_rep len) :
  88. begin_(first_point),
  89. last_(first_point + len-duration_rep::unit())
  90. { }
  91. //! Return the first element in the period
  92. template<class point_rep, class duration_rep>
  93. inline
  94. point_rep period<point_rep,duration_rep>::begin() const
  95. {
  96. return begin_;
  97. }
  98. //! Return one past the last element
  99. template<class point_rep, class duration_rep>
  100. inline
  101. point_rep period<point_rep,duration_rep>::end() const
  102. {
  103. return last_ + duration_rep::unit();
  104. }
  105. //! Return the last item in the period
  106. template<class point_rep, class duration_rep>
  107. inline
  108. point_rep period<point_rep,duration_rep>::last() const
  109. {
  110. return last_;
  111. }
  112. //! True if period is ill formed (length is zero or less)
  113. template<class point_rep, class duration_rep>
  114. inline
  115. bool period<point_rep,duration_rep>::is_null() const
  116. {
  117. return end() <= begin_;
  118. }
  119. //! Return the length of the period
  120. template<class point_rep, class duration_rep>
  121. inline
  122. duration_rep period<point_rep,duration_rep>::length() const
  123. {
  124. if(last_ < begin_){ // invalid period
  125. return last_+duration_rep::unit() - begin_;
  126. }
  127. else{
  128. return end() - begin_; // normal case
  129. }
  130. }
  131. //! Equality operator
  132. template<class point_rep, class duration_rep>
  133. inline
  134. bool period<point_rep,duration_rep>::operator==(const period& rhs) const
  135. {
  136. return ((begin_ == rhs.begin_) &&
  137. (last_ == rhs.last_));
  138. }
  139. //! Strict as defined by rhs.last <= lhs.last
  140. template<class point_rep, class duration_rep>
  141. inline
  142. bool period<point_rep,duration_rep>::operator<(const period& rhs) const
  143. {
  144. return (last_ < rhs.begin_);
  145. }
  146. //! Shift the start and end by the specified amount
  147. template<class point_rep, class duration_rep>
  148. inline
  149. void period<point_rep,duration_rep>::shift(const duration_rep& d)
  150. {
  151. begin_ = begin_ + d;
  152. last_ = last_ + d;
  153. }
  154. /** Expands the size of the period by the duration on both ends.
  155. *
  156. *So before expand
  157. *@code
  158. *
  159. * [-------]
  160. * ^ ^ ^ ^ ^ ^ ^
  161. * 1 2 3 4 5 6 7
  162. *
  163. *@endcode
  164. * After expand(2)
  165. *@code
  166. *
  167. * [----------------------]
  168. * ^ ^ ^ ^ ^ ^ ^
  169. * 1 2 3 4 5 6 7
  170. *
  171. *@endcode
  172. */
  173. template<class point_rep, class duration_rep>
  174. inline
  175. void period<point_rep,duration_rep>::expand(const duration_rep& d)
  176. {
  177. begin_ = begin_ - d;
  178. last_ = last_ + d;
  179. }
  180. //! True if the point is inside the period, zero length periods contain no points
  181. template<class point_rep, class duration_rep>
  182. inline
  183. bool period<point_rep,duration_rep>::contains(const point_rep& point) const
  184. {
  185. return ((point >= begin_) &&
  186. (point <= last_));
  187. }
  188. //! True if this period fully contains (or equals) the other period
  189. template<class point_rep, class duration_rep>
  190. inline
  191. bool period<point_rep,duration_rep>::contains(const period<point_rep,duration_rep>& other) const
  192. {
  193. return ((begin_ <= other.begin_) && (last_ >= other.last_));
  194. }
  195. //! True if periods are next to each other without a gap.
  196. /* In the example below, p1 and p2 are adjacent, but p3 is not adjacent
  197. * with either of p1 or p2.
  198. *@code
  199. * [-p1-)
  200. * [-p2-)
  201. * [-p3-)
  202. *@endcode
  203. */
  204. template<class point_rep, class duration_rep>
  205. inline
  206. bool
  207. period<point_rep,duration_rep>::is_adjacent(const period<point_rep,duration_rep>& other) const
  208. {
  209. return (other.begin() == end() ||
  210. begin_ == other.end());
  211. }
  212. //! True if all of the period is prior or t < start
  213. /* In the example below only point 1 would evaluate to true.
  214. *@code
  215. * [---------])
  216. * ^ ^ ^ ^ ^
  217. * 1 2 3 4 5
  218. *
  219. *@endcode
  220. */
  221. template<class point_rep, class duration_rep>
  222. inline
  223. bool
  224. period<point_rep,duration_rep>::is_after(const point_rep& t) const
  225. {
  226. if (is_null())
  227. {
  228. return false; //null period isn't after
  229. }
  230. return t < begin_;
  231. }
  232. //! True if all of the period is prior to the passed point or end <= t
  233. /* In the example below points 4 and 5 return true.
  234. *@code
  235. * [---------])
  236. * ^ ^ ^ ^ ^
  237. * 1 2 3 4 5
  238. *
  239. *@endcode
  240. */
  241. template<class point_rep, class duration_rep>
  242. inline
  243. bool
  244. period<point_rep,duration_rep>::is_before(const point_rep& t) const
  245. {
  246. if (is_null())
  247. {
  248. return false; //null period isn't before anything
  249. }
  250. return last_ < t;
  251. }
  252. //! True if the periods overlap in any way
  253. /* In the example below p1 intersects with p2, p4, and p6.
  254. *@code
  255. * [---p1---)
  256. * [---p2---)
  257. * [---p3---)
  258. * [---p4---)
  259. * [-p5-)
  260. * [-p6-)
  261. *@endcode
  262. */
  263. template<class point_rep, class duration_rep>
  264. inline
  265. bool period<point_rep,duration_rep>::intersects(const period<point_rep,duration_rep>& other) const
  266. {
  267. return ( contains(other.begin_) ||
  268. other.contains(begin_) ||
  269. ((other.begin_ < begin_) && (other.last_ >= begin_)));
  270. }
  271. //! Returns the period of intersection or invalid range no intersection
  272. template<class point_rep, class duration_rep>
  273. inline
  274. period<point_rep,duration_rep>
  275. period<point_rep,duration_rep>::intersection(const period<point_rep,duration_rep>& other) const
  276. {
  277. if (begin_ > other.begin_) {
  278. if (last_ <= other.last_) { //case2
  279. return *this;
  280. }
  281. //case 1
  282. return period<point_rep,duration_rep>(begin_, other.end());
  283. }
  284. else {
  285. if (last_ <= other.last_) { //case3
  286. return period<point_rep,duration_rep>(other.begin_, this->end());
  287. }
  288. //case4
  289. return other;
  290. }
  291. //unreachable
  292. }
  293. //! Returns the union of intersecting periods -- or null period
  294. /*!
  295. */
  296. template<class point_rep, class duration_rep>
  297. inline
  298. period<point_rep,duration_rep>
  299. period<point_rep,duration_rep>::merge(const period<point_rep,duration_rep>& other) const
  300. {
  301. if (this->intersects(other)) {
  302. if (begin_ < other.begin_) {
  303. return period<point_rep,duration_rep>(begin_, last_ > other.last_ ? this->end() : other.end());
  304. }
  305. return period<point_rep,duration_rep>(other.begin_, last_ > other.last_ ? this->end() : other.end());
  306. }
  307. return period<point_rep,duration_rep>(begin_,begin_); // no intersect return null
  308. }
  309. //! Combine two periods with earliest start and latest end.
  310. /*! Combines two periods and any gap between them such that
  311. * start = min(p1.start, p2.start)
  312. * end = max(p1.end , p2.end)
  313. *@code
  314. * [---p1---)
  315. * [---p2---)
  316. * result:
  317. * [-----------p3----------)
  318. *@endcode
  319. */
  320. template<class point_rep, class duration_rep>
  321. inline
  322. period<point_rep,duration_rep>
  323. period<point_rep,duration_rep>::span(const period<point_rep,duration_rep>& other) const
  324. {
  325. point_rep start((begin_ < other.begin_) ? begin() : other.begin());
  326. point_rep newend((last_ < other.last_) ? other.end() : this->end());
  327. return period<point_rep,duration_rep>(start, newend);
  328. }
  329. } } //namespace date_time
  330. #endif