ctkDateRangeWidget.cpp 9.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289
  1. /*=========================================================================
  2. Library: CTK
  3. Copyright (c) Isomics Inc.
  4. Licensed under the Apache License, Version 2.0 (the "License");
  5. you may not use this file except in compliance with the License.
  6. You may obtain a copy of the License at
  7. http://www.apache.org/licenses/LICENSE-2.0.txt
  8. Unless required by applicable law or agreed to in writing, software
  9. distributed under the License is distributed on an "AS IS" BASIS,
  10. WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  11. See the License for the specific language governing permissions and
  12. limitations under the License.
  13. =========================================================================*/
  14. // Qt includes
  15. #include <QDebug>
  16. #include <QDate>
  17. #include <QDateTime>
  18. #include <QDateTimeEdit>
  19. // CTK includes
  20. #include "ctkDateRangeWidget.h"
  21. #include "ui_ctkDateRangeWidget.h"
  22. #include "ctkLogger.h"
  23. static ctkLogger logger("org.commontk.libs.widgets.ctkDateRangeWidget");
  24. //-----------------------------------------------------------------------------
  25. class ctkDateRangeWidgetPrivate: public Ui_ctkDateRangeWidget
  26. {
  27. Q_DECLARE_PUBLIC(ctkDateRangeWidget);
  28. protected:
  29. ctkDateRangeWidget* const q_ptr;
  30. public:
  31. ctkDateRangeWidgetPrivate(ctkDateRangeWidget& object);
  32. /// Automatically select the right radio button based on the date range
  33. void autoselectRadioButton();
  34. /// ForceSelectRange is set to true when the user expressively requested
  35. /// to have "Select Range" option active. This property is set only if
  36. /// the user clicks on the option or if setSelectRange() is programatically
  37. /// called
  38. bool ForceSelectRange;
  39. /// DisplayTime is true if the time is displayed in the range widget
  40. bool DisplayTime;
  41. };
  42. // --------------------------------------------------------------------------
  43. ctkDateRangeWidgetPrivate::ctkDateRangeWidgetPrivate(ctkDateRangeWidget& object)
  44. :q_ptr(&object)
  45. {
  46. this->ForceSelectRange = false;
  47. this->DisplayTime = true;
  48. }
  49. // -------------------------------------------------------------------------
  50. void ctkDateRangeWidgetPrivate::autoselectRadioButton()
  51. {
  52. Q_Q(ctkDateRangeWidget);
  53. QDate startDate = q->startDateTime().date();
  54. QDate endDate = q->endDateTime().date();
  55. if (this->ForceSelectRange)
  56. {
  57. this->SelectRangeRadioButton->setChecked(true);
  58. }
  59. else if (q->isAnyDate())
  60. {
  61. this->AnyDateRadioButton->setChecked(true);
  62. }
  63. else if (q->startDateTime() != QDateTime(q->startDateTime().date()) ||
  64. q->endDateTime() != QDateTime(q->endDateTime().date()))
  65. {
  66. this->SelectRangeRadioButton->setChecked(true);
  67. }
  68. else if (startDate.addDays(1) == endDate &&
  69. startDate == QDate::currentDate())
  70. {
  71. this->TodayRadioButton->setChecked(true);
  72. }
  73. else if (startDate.addDays(1) == endDate &&
  74. endDate == QDate::currentDate())
  75. {
  76. this->YesterdayRadioButton->setChecked(true);
  77. }
  78. else if (startDate.addDays(7) == endDate &&
  79. endDate == QDate::currentDate())
  80. {
  81. this->LastWeekRadioButton->setChecked(true);
  82. }
  83. else if (startDate.addDays(31) == endDate &&
  84. endDate == QDate::currentDate())
  85. {
  86. this->LastMonthRadioButton->setChecked(true);
  87. }
  88. else
  89. {
  90. this->SelectRangeRadioButton->setChecked(true);
  91. }
  92. }
  93. // --------------------------------------------------------------------------
  94. ctkDateRangeWidget::ctkDateRangeWidget(QWidget* _parent) : Superclass(_parent)
  95. , d_ptr(new ctkDateRangeWidgetPrivate(*this))
  96. {
  97. Q_D(ctkDateRangeWidget);
  98. d->setupUi(this);
  99. d->DateRangeWidget->setVisible(d->SelectRangeRadioButton->isChecked());
  100. this->setDisplayTime(false);
  101. this->setDateTimeRange(QDateTime(), QDateTime());
  102. // Note that we connect on the clicked() signal and not the toggled.
  103. // The clicked() signal is fired only when the USER clicks the radio button
  104. // and not when the button is checked programatically (except using click()).
  105. QObject::connect(d->AnyDateRadioButton, SIGNAL(clicked()),
  106. this, SLOT(setAnyDate()));
  107. QObject::connect(d->TodayRadioButton, SIGNAL(clicked()),
  108. this, SLOT(setToday()));
  109. QObject::connect(d->YesterdayRadioButton, SIGNAL(clicked()),
  110. this, SLOT(setYesterday()));
  111. QObject::connect(d->LastWeekRadioButton, SIGNAL(clicked()),
  112. this, SLOT(setLastWeek()));
  113. QObject::connect(d->LastMonthRadioButton, SIGNAL(clicked()),
  114. this, SLOT(setLastMonth()));
  115. QObject::connect(d->SelectRangeRadioButton, SIGNAL(clicked()),
  116. this, SLOT(setSelectRange()));
  117. QObject::connect(d->StartDate, SIGNAL(dateTimeChanged(QDateTime)),
  118. this, SIGNAL(startDateTimeChanged(QDateTime)));
  119. QObject::connect(d->EndDate, SIGNAL(dateTimeChanged(QDateTime)),
  120. this, SIGNAL(endDateTimeChanged(QDateTime)));
  121. QObject::connect(d->StartDate, SIGNAL(dateTimeChanged(QDateTime)),
  122. this, SLOT(onDateTimeChanged()));
  123. QObject::connect(d->EndDate, SIGNAL(dateTimeChanged(QDateTime)),
  124. this, SLOT(onDateTimeChanged()));
  125. }
  126. // --------------------------------------------------------------------------
  127. ctkDateRangeWidget::~ctkDateRangeWidget()
  128. {
  129. }
  130. // --------------------------------------------------------------------------
  131. QDateTime ctkDateRangeWidget::startDateTime()const
  132. {
  133. Q_D(const ctkDateRangeWidget);
  134. return d->StartDate->dateTime();
  135. }
  136. // --------------------------------------------------------------------------
  137. QDateTime ctkDateRangeWidget::endDateTime()const
  138. {
  139. Q_D(const ctkDateRangeWidget);
  140. return d->EndDate->dateTime();
  141. }
  142. // --------------------------------------------------------------------------
  143. void ctkDateRangeWidget::setStartDateTime(QDateTime dateTime)
  144. {
  145. Q_D(ctkDateRangeWidget);
  146. d->StartDate->setDateTime(dateTime);
  147. d->autoselectRadioButton();
  148. }
  149. // --------------------------------------------------------------------------
  150. void ctkDateRangeWidget::setEndDateTime(QDateTime dateTime)
  151. {
  152. Q_D(ctkDateRangeWidget);
  153. d->EndDate->setDateTime(dateTime);
  154. d->autoselectRadioButton();
  155. }
  156. // --------------------------------------------------------------------------
  157. void ctkDateRangeWidget::setDateTimeRange(QDateTime startDateTime, QDateTime endDateTime)
  158. {
  159. Q_D(ctkDateRangeWidget);
  160. d->StartDate->setDateTime(startDateTime.isValid() ?
  161. startDateTime : d->StartDate->minimumDateTime());
  162. d->EndDate->setDateTime(endDateTime.isValid() ?
  163. endDateTime : d->EndDate->maximumDateTime());
  164. d->autoselectRadioButton();
  165. }
  166. // --------------------------------------------------------------------------
  167. void ctkDateRangeWidget::setDateRange(QDate startDate, QDate endDate)
  168. {
  169. this->setDateTimeRange(QDateTime(startDate), QDateTime(endDate));
  170. }
  171. // --------------------------------------------------------------------------
  172. void ctkDateRangeWidget::setAnyDate()
  173. {
  174. Q_D(ctkDateRangeWidget);
  175. d->ForceSelectRange = false;
  176. this->setDateTimeRange(QDateTime(), QDateTime());
  177. }
  178. // --------------------------------------------------------------------------
  179. void ctkDateRangeWidget::setToday()
  180. {
  181. Q_D(ctkDateRangeWidget);
  182. d->ForceSelectRange = false;
  183. QDate today = QDate::currentDate();
  184. this->setDateRange(today, today.addDays(1));
  185. }
  186. // --------------------------------------------------------------------------
  187. void ctkDateRangeWidget::setYesterday()
  188. {
  189. Q_D(ctkDateRangeWidget);
  190. d->ForceSelectRange = false;
  191. QDate today = QDate::currentDate();
  192. this->setDateRange(today.addDays(-1), today);
  193. }
  194. // --------------------------------------------------------------------------
  195. void ctkDateRangeWidget::setLastWeek()
  196. {
  197. Q_D(ctkDateRangeWidget);
  198. d->ForceSelectRange = false;
  199. QDate today = QDate::currentDate();
  200. this->setDateRange(today.addDays(-7), today);
  201. }
  202. // --------------------------------------------------------------------------
  203. void ctkDateRangeWidget::setLastMonth()
  204. {
  205. Q_D(ctkDateRangeWidget);
  206. d->ForceSelectRange = false;
  207. QDate today = QDate::currentDate();
  208. this->setDateRange(today.addMonths(-1), today);
  209. }
  210. // --------------------------------------------------------------------------
  211. void ctkDateRangeWidget::setSelectRange()
  212. {
  213. Q_D(ctkDateRangeWidget);
  214. d->SelectRangeRadioButton->setChecked(true);
  215. d->ForceSelectRange = true;
  216. }
  217. // -------------------------------------------------------------------------
  218. bool ctkDateRangeWidget::isAnyDate()const
  219. {
  220. Q_D(const ctkDateRangeWidget);
  221. return this->startDateTime() == d->StartDate->minimumDateTime() &&
  222. this->endDateTime() == d->EndDate->maximumDateTime();
  223. }
  224. // -------------------------------------------------------------------------
  225. void ctkDateRangeWidget::setDisplayTime(bool displayTime)
  226. {
  227. Q_D(ctkDateRangeWidget);
  228. d->DisplayTime = displayTime;
  229. if ( displayTime )
  230. {
  231. d->StartDate->setDisplayFormat( QString( "MMM dd, yyyy HH:mm:ss") );
  232. d->EndDate->setDisplayFormat( QString( "MMM dd, yyyy HH:mm:ss") );
  233. }
  234. else
  235. {
  236. d->StartDate->setDisplayFormat( QString( "MMM dd, yyyy") );
  237. d->EndDate->setDisplayFormat( QString( "MMM dd, yyyy") );
  238. }
  239. }
  240. // -------------------------------------------------------------------------
  241. bool ctkDateRangeWidget::displayTime()const
  242. {
  243. logger.error("including time in the date range is not supported now");
  244. Q_D(const ctkDateRangeWidget);
  245. return d->DisplayTime;
  246. }
  247. // -------------------------------------------------------------------------
  248. void ctkDateRangeWidget::onDateTimeChanged()
  249. {
  250. Q_D(ctkDateRangeWidget);
  251. d->autoselectRadioButton();
  252. }