ctkDoubleRangeSlider.cpp 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553
  1. /*=========================================================================
  2. Library: CTK
  3. Copyright (c) 2010 Kitware 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.commontk.org/LICENSE
  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 <QHBoxLayout>
  17. // CTK includes
  18. #include "ctkRangeSlider.h"
  19. #include "ctkDoubleRangeSlider.h"
  20. //-----------------------------------------------------------------------------
  21. class ctkDoubleRangeSliderPrivate: public ctkPrivate<ctkDoubleRangeSlider>
  22. {
  23. public:
  24. ctkDoubleRangeSliderPrivate();
  25. int toInt(double _value)const;
  26. double minFromInt(int _value)const;
  27. double maxFromInt(int _value)const;
  28. void init();
  29. void connectSlider();
  30. void updateMinOffset(double value);
  31. void updateMaxOffset(double value);
  32. ctkRangeSlider* Slider;
  33. double Minimum;
  34. double Maximum;
  35. bool SettingRange;
  36. // we should have a MinValueOffset and MinPositionOffset (and MinimumOffset?)
  37. double MinOffset;
  38. // we should have a MaxValueOffset and MaxPositionOffset (and MaximumOffset?)
  39. double MaxOffset;
  40. double SingleStep;
  41. double MinValue;
  42. double MaxValue;
  43. };
  44. // --------------------------------------------------------------------------
  45. ctkDoubleRangeSliderPrivate::ctkDoubleRangeSliderPrivate()
  46. {
  47. // the initial values will be overwritten in
  48. // ctkDoubleRangeSliderPrivate::init()
  49. this->Slider = 0;
  50. this->Minimum = 0.;
  51. this->Maximum = 99.;
  52. this->SettingRange = false;
  53. this->MinOffset = 0.;
  54. this->MaxOffset = 0.;
  55. this->SingleStep = 1.;
  56. this->MinValue = 0.;
  57. this->MaxValue = 99.;
  58. }
  59. // --------------------------------------------------------------------------
  60. void ctkDoubleRangeSliderPrivate::init()
  61. {
  62. CTK_P(ctkDoubleRangeSlider);
  63. this->Slider = new ctkRangeSlider(p);
  64. QHBoxLayout* l = new QHBoxLayout(p);
  65. l->addWidget(this->Slider);
  66. l->setContentsMargins(0,0,0,0);
  67. this->Minimum = this->Slider->minimum();
  68. this->Maximum = this->Slider->maximum();
  69. this->MinValue = this->Slider->minimumValue();
  70. this->MaxValue = this->Slider->maximumValue();
  71. this->SingleStep = this->Slider->singleStep();
  72. this->connectSlider();
  73. }
  74. // --------------------------------------------------------------------------
  75. void ctkDoubleRangeSliderPrivate::connectSlider()
  76. {
  77. CTK_P(ctkDoubleRangeSlider);
  78. p->connect(this->Slider, SIGNAL(minimumValueChanged(int)),
  79. p, SLOT(onMinValueChanged(int)));
  80. p->connect(this->Slider, SIGNAL(maximumValueChanged(int)),
  81. p, SLOT(onMaxValueChanged(int)));
  82. p->connect(this->Slider, SIGNAL(valuesChanged(int,int)),
  83. p, SLOT(onValuesChanged(int,int)));
  84. p->connect(this->Slider, SIGNAL(minimumPositionChanged(int)),
  85. p, SLOT(onMinPosChanged(int)));
  86. p->connect(this->Slider, SIGNAL(maximumPositionChanged(int)),
  87. p, SLOT(onMaxPosChanged(int)));
  88. p->connect(this->Slider, SIGNAL(positionsChanged(int,int)),
  89. p, SLOT(onPositionsChanged(int,int)));
  90. p->connect(this->Slider, SIGNAL(sliderPressed()),
  91. p, SIGNAL(sliderPressed()));
  92. p->connect(this->Slider, SIGNAL(sliderReleased()),
  93. p, SIGNAL(sliderReleased()));
  94. p->connect(this->Slider, SIGNAL(rangeChanged(int, int)),
  95. p, SLOT(onRangeChanged(int, int)));
  96. }
  97. // --------------------------------------------------------------------------
  98. int ctkDoubleRangeSliderPrivate::toInt(double doubleValue)const
  99. {
  100. double tmp = doubleValue / this->SingleStep;
  101. int intValue = qRound(tmp);
  102. return intValue;
  103. }
  104. // --------------------------------------------------------------------------
  105. double ctkDoubleRangeSliderPrivate::minFromInt(int intValue)const
  106. {
  107. double doubleValue = this->SingleStep * (this->MinOffset + intValue) ;
  108. return doubleValue;
  109. }
  110. // --------------------------------------------------------------------------
  111. double ctkDoubleRangeSliderPrivate::maxFromInt(int intValue)const
  112. {
  113. double doubleValue = this->SingleStep * (this->MaxOffset + intValue) ;
  114. return doubleValue;
  115. }
  116. // --------------------------------------------------------------------------
  117. void ctkDoubleRangeSliderPrivate::updateMinOffset(double value)
  118. {
  119. this->MinOffset = (value / this->SingleStep) - this->toInt(value);
  120. }
  121. // --------------------------------------------------------------------------
  122. void ctkDoubleRangeSliderPrivate::updateMaxOffset(double value)
  123. {
  124. this->MaxOffset = (value / this->SingleStep) - this->toInt(value);
  125. }
  126. // --------------------------------------------------------------------------
  127. ctkDoubleRangeSlider::ctkDoubleRangeSlider(QWidget* _parent) : Superclass(_parent)
  128. {
  129. CTK_INIT_PRIVATE(ctkDoubleRangeSlider);
  130. ctk_d()->init();
  131. }
  132. // --------------------------------------------------------------------------
  133. ctkDoubleRangeSlider::ctkDoubleRangeSlider(Qt::Orientation _orientation, QWidget* _parent)
  134. : Superclass(_parent)
  135. {
  136. CTK_INIT_PRIVATE(ctkDoubleRangeSlider);
  137. ctk_d()->init();
  138. this->setOrientation(_orientation);
  139. }
  140. // --------------------------------------------------------------------------
  141. void ctkDoubleRangeSlider::setMinimum(double min)
  142. {
  143. CTK_D(ctkDoubleRangeSlider);
  144. d->Minimum = min;
  145. if (d->Minimum >= d->MinValue)
  146. {// TBD: use same offset
  147. d->updateMinOffset(d->Minimum);
  148. }
  149. if (d->Minimum >= d->MaxValue)
  150. {// TBD: use same offset
  151. d->updateMaxOffset(d->Minimum);
  152. }
  153. d->SettingRange = true;
  154. d->Slider->setMinimum(d->toInt(min));
  155. d->SettingRange = false;
  156. emit this->rangeChanged(d->Minimum, d->Maximum);
  157. }
  158. // --------------------------------------------------------------------------
  159. double ctkDoubleRangeSlider::minimum()const
  160. {
  161. CTK_D(const ctkDoubleRangeSlider);
  162. return d->Minimum;
  163. }
  164. // --------------------------------------------------------------------------
  165. void ctkDoubleRangeSlider::setMaximum(double max)
  166. {
  167. CTK_D(ctkDoubleRangeSlider);
  168. d->Maximum = max;
  169. if (d->Maximum <= d->MinValue)
  170. {// TBD: use same offset
  171. d->updateMinOffset(d->Maximum);
  172. }
  173. if (d->Maximum <= d->MaxValue)
  174. {// TBD: use same offset ?
  175. d->updateMaxOffset(d->Maximum);
  176. }
  177. d->SettingRange = true;
  178. d->Slider->setMaximum(d->toInt(max));
  179. d->SettingRange = false;
  180. emit this->rangeChanged(d->Minimum, d->Maximum);
  181. }
  182. // --------------------------------------------------------------------------
  183. double ctkDoubleRangeSlider::maximum()const
  184. {
  185. CTK_D(const ctkDoubleRangeSlider);
  186. return d->Maximum;
  187. }
  188. // --------------------------------------------------------------------------
  189. void ctkDoubleRangeSlider::setRange(double min, double max)
  190. {
  191. CTK_D(ctkDoubleRangeSlider);
  192. d->Minimum = min;
  193. d->Maximum = max;
  194. if (d->Minimum >= d->MinValue)
  195. {// TBD: use same offset
  196. d->updateMinOffset(d->Minimum);
  197. }
  198. if (d->Minimum >= d->MaxValue)
  199. {// TBD: use same offset
  200. d->updateMaxOffset(d->Minimum);
  201. }
  202. if (d->Maximum <= d->MinValue)
  203. {// TBD: use same offset
  204. d->updateMinOffset(d->Maximum);
  205. }
  206. if (d->Maximum <= d->MaxValue)
  207. {// TBD: use same offset ?
  208. d->updateMaxOffset(d->Maximum);
  209. }
  210. d->SettingRange = true;
  211. d->Slider->setRange(d->toInt(min), d->toInt(max));
  212. d->SettingRange = false;
  213. emit this->rangeChanged(d->Minimum, d->Maximum);
  214. }
  215. // --------------------------------------------------------------------------
  216. double ctkDoubleRangeSlider::minimumPosition()const
  217. {
  218. CTK_D(const ctkDoubleRangeSlider);
  219. return d->minFromInt(d->Slider->minimumPosition());
  220. }
  221. // --------------------------------------------------------------------------
  222. void ctkDoubleRangeSlider::setMinimumPosition(double minPos)
  223. {
  224. CTK_D(ctkDoubleRangeSlider);
  225. d->Slider->setMinimumPosition(d->toInt(minPos));
  226. }
  227. // --------------------------------------------------------------------------
  228. double ctkDoubleRangeSlider::maximumPosition()const
  229. {
  230. CTK_D(const ctkDoubleRangeSlider);
  231. return d->maxFromInt(d->Slider->maximumPosition());
  232. }
  233. // --------------------------------------------------------------------------
  234. void ctkDoubleRangeSlider::setMaximumPosition(double maxPos)
  235. {
  236. CTK_D(ctkDoubleRangeSlider);
  237. d->Slider->setMaximumPosition(d->toInt(maxPos));
  238. }
  239. // --------------------------------------------------------------------------
  240. void ctkDoubleRangeSlider::setPositions(double minPos, double maxPos)
  241. {
  242. CTK_D(ctkDoubleRangeSlider);
  243. d->Slider->setPositions(d->toInt(minPos), d->toInt(maxPos));
  244. }
  245. // --------------------------------------------------------------------------
  246. double ctkDoubleRangeSlider::minimumValue()const
  247. {
  248. CTK_D(const ctkDoubleRangeSlider);
  249. return d->MinValue;
  250. }
  251. // --------------------------------------------------------------------------
  252. void ctkDoubleRangeSlider::setMinimumValue(double newMinValue)
  253. {
  254. CTK_D(ctkDoubleRangeSlider);
  255. newMinValue = qBound(d->Minimum, newMinValue, d->Maximum);
  256. d->updateMinOffset(newMinValue);
  257. if (newMinValue >= d->MaxValue)
  258. {
  259. d->updateMaxOffset(newMinValue);
  260. }
  261. int newIntValue = d->toInt(newMinValue);
  262. if (newIntValue != d->Slider->minimumValue())
  263. {
  264. // d->Slider will emit a minimumValueChanged signal that is connected to
  265. // ctkDoubleSlider::onValueChanged
  266. d->Slider->setMinimumValue(newIntValue);
  267. }
  268. else
  269. {
  270. double oldValue = d->MinValue;
  271. d->MinValue = newMinValue;
  272. // don't emit a valuechanged signal if the new value is quite
  273. // similar to the old value.
  274. if (qAbs(newMinValue - oldValue) > (d->SingleStep * 0.000000001))
  275. {
  276. emit this->minimumValueChanged(newMinValue);
  277. }
  278. }
  279. }
  280. // --------------------------------------------------------------------------
  281. double ctkDoubleRangeSlider::maximumValue()const
  282. {
  283. CTK_D(const ctkDoubleRangeSlider);
  284. return d->MaxValue;
  285. }
  286. // --------------------------------------------------------------------------
  287. void ctkDoubleRangeSlider::setMaximumValue(double newMaxValue)
  288. {
  289. CTK_D(ctkDoubleRangeSlider);
  290. newMaxValue = qBound(d->Minimum, newMaxValue, d->Maximum);
  291. d->updateMaxOffset(newMaxValue);
  292. if (newMaxValue <= d->MinValue)
  293. {
  294. d->updateMinOffset(newMaxValue);
  295. }
  296. int newIntValue = d->toInt(newMaxValue);
  297. if (newIntValue != d->Slider->maximumValue())
  298. {
  299. // d->Slider will emit a maximumValueChanged signal that is connected to
  300. // ctkDoubleSlider::onValueChanged
  301. d->Slider->setMaximumValue(newIntValue);
  302. }
  303. else
  304. {
  305. double oldValue = d->MaxValue;
  306. d->MaxValue = newMaxValue;
  307. // don't emit a valuechanged signal if the new value is quite
  308. // similar to the old value.
  309. if (qAbs(newMaxValue - oldValue) > (d->SingleStep * 0.000000001))
  310. {
  311. emit this->maximumValueChanged(newMaxValue);
  312. }
  313. }
  314. }
  315. // --------------------------------------------------------------------------
  316. void ctkDoubleRangeSlider::setValues(double newMinValue, double newMaxValue)
  317. {
  318. this->setMinimumValue(qMin(newMinValue, newMaxValue));
  319. this->setMaximumValue(qMax(newMinValue, newMaxValue));
  320. }
  321. // --------------------------------------------------------------------------
  322. double ctkDoubleRangeSlider::singleStep()const
  323. {
  324. CTK_D(const ctkDoubleRangeSlider);
  325. return d->SingleStep;
  326. }
  327. // --------------------------------------------------------------------------
  328. void ctkDoubleRangeSlider::setSingleStep(double newStep)
  329. {
  330. CTK_D(ctkDoubleRangeSlider);
  331. d->SingleStep = newStep;
  332. // The following can fire A LOT of signals that shouldn't be
  333. // fired.
  334. bool oldBlockSignals = this->blockSignals(true);
  335. d->updateMinOffset(d->MinValue);
  336. d->updateMaxOffset(d->MaxValue);
  337. // update the new values of the ctkRangeSlider
  338. double _minvalue = d->MinValue;
  339. double _maxvalue = d->MaxValue;
  340. // calling setMinimum or setMaximum can change the values MinimumValue
  341. // and MaximumValue, this is why we re-set them later.
  342. this->setMinimum(d->Minimum);
  343. this->setMaximum(d->Maximum);
  344. this->setMinimumValue(_minvalue);
  345. this->setMinimumPosition(_minvalue);
  346. this->setMaximumValue(_maxvalue);
  347. this->setMaximumPosition(_maxvalue);
  348. this->blockSignals(oldBlockSignals);
  349. }
  350. // --------------------------------------------------------------------------
  351. double ctkDoubleRangeSlider::tickInterval()const
  352. {
  353. CTK_D(const ctkDoubleRangeSlider);
  354. return d->minFromInt(d->Slider->tickInterval());
  355. }
  356. // --------------------------------------------------------------------------
  357. void ctkDoubleRangeSlider::setTickInterval(double newTickInterval)
  358. {
  359. CTK_D(ctkDoubleRangeSlider);
  360. d->Slider->setTickInterval(d->toInt(newTickInterval));
  361. }
  362. // --------------------------------------------------------------------------
  363. bool ctkDoubleRangeSlider::hasTracking()const
  364. {
  365. CTK_D(const ctkDoubleRangeSlider);
  366. return d->Slider->hasTracking();
  367. }
  368. // --------------------------------------------------------------------------
  369. void ctkDoubleRangeSlider::setTracking(bool enable)
  370. {
  371. CTK_D(ctkDoubleRangeSlider);
  372. d->Slider->setTracking(enable);
  373. }
  374. // --------------------------------------------------------------------------
  375. void ctkDoubleRangeSlider::triggerAction( QAbstractSlider::SliderAction action)
  376. {
  377. CTK_D(ctkDoubleRangeSlider);
  378. d->Slider->triggerAction(action);
  379. }
  380. // --------------------------------------------------------------------------
  381. void ctkDoubleRangeSlider::setOrientation(Qt::Orientation newOrientation)
  382. {
  383. CTK_D(ctkDoubleRangeSlider);
  384. d->Slider->setOrientation(newOrientation);
  385. }
  386. // --------------------------------------------------------------------------
  387. Qt::Orientation ctkDoubleRangeSlider::orientation()const
  388. {
  389. CTK_D(const ctkDoubleRangeSlider);
  390. return d->Slider->orientation();
  391. }
  392. // --------------------------------------------------------------------------
  393. void ctkDoubleRangeSlider::onMinValueChanged(int newValue)
  394. {
  395. CTK_D(ctkDoubleRangeSlider);
  396. double doubleNewValue = d->minFromInt(newValue);
  397. if (d->MinValue == doubleNewValue)
  398. {
  399. return;
  400. }
  401. d->MinValue = doubleNewValue;
  402. emit this->valuesChanged(d->MinValue, d->MaxValue);
  403. emit this->minimumValueChanged(d->MinValue);
  404. }
  405. // --------------------------------------------------------------------------
  406. void ctkDoubleRangeSlider::onMaxValueChanged(int newValue)
  407. {
  408. CTK_D(ctkDoubleRangeSlider);
  409. double doubleNewValue = d->maxFromInt(newValue);
  410. if (d->MaxValue == doubleNewValue)
  411. {
  412. return;
  413. }
  414. d->MaxValue = doubleNewValue;
  415. emit this->valuesChanged(d->MinValue, d->MaxValue);
  416. emit this->maximumValueChanged(d->MaxValue);
  417. }
  418. // --------------------------------------------------------------------------
  419. void ctkDoubleRangeSlider::onValuesChanged(int newMinValue, int newMaxValue)
  420. {
  421. CTK_D(ctkDoubleRangeSlider);
  422. double doubleNewMinValue = d->minFromInt(newMinValue);
  423. double doubleNewMaxValue = d->maxFromInt(newMaxValue);
  424. bool emitMinValueChanged = (d->MinValue != doubleNewMinValue);
  425. bool emitMaxValueChanged = (d->MaxValue != doubleNewMaxValue);
  426. if (!emitMinValueChanged && !emitMaxValueChanged)
  427. {
  428. return;
  429. }
  430. d->MinValue = doubleNewMinValue;
  431. d->MaxValue = doubleNewMaxValue;
  432. emit this->valuesChanged(d->MinValue, d->MaxValue);
  433. if (emitMinValueChanged)
  434. {
  435. emit this->minimumValueChanged(d->MinValue);
  436. }
  437. if (emitMaxValueChanged)
  438. {
  439. emit this->maximumValueChanged(d->MaxValue);
  440. }
  441. }
  442. // --------------------------------------------------------------------------
  443. void ctkDoubleRangeSlider::onMinPosChanged(int newPosition)
  444. {
  445. CTK_D(const ctkDoubleRangeSlider);
  446. emit this->minimumPositionChanged(d->minFromInt(newPosition));
  447. }
  448. // --------------------------------------------------------------------------
  449. void ctkDoubleRangeSlider::onMaxPosChanged(int newPosition)
  450. {
  451. CTK_D(const ctkDoubleRangeSlider);
  452. emit this->maximumPositionChanged(d->maxFromInt(newPosition));
  453. }
  454. // --------------------------------------------------------------------------
  455. void ctkDoubleRangeSlider::onPositionsChanged(int min, int max)
  456. {
  457. CTK_D(const ctkDoubleRangeSlider);
  458. emit this->positionsChanged(d->minFromInt(min), d->maxFromInt(max));
  459. }
  460. // --------------------------------------------------------------------------
  461. void ctkDoubleRangeSlider::onRangeChanged(int min, int max)
  462. {
  463. CTK_D(const ctkDoubleRangeSlider);
  464. if (!d->SettingRange)
  465. {
  466. this->setRange(d->minFromInt(min), d->maxFromInt(max));
  467. }
  468. }
  469. // --------------------------------------------------------------------------
  470. ctkRangeSlider* ctkDoubleRangeSlider::slider()const
  471. {
  472. CTK_D(const ctkDoubleRangeSlider);
  473. return d->Slider;
  474. }
  475. // --------------------------------------------------------------------------
  476. void ctkDoubleRangeSlider::setSlider(ctkRangeSlider* slider)
  477. {
  478. CTK_D(ctkDoubleRangeSlider);
  479. slider->setOrientation(d->Slider->orientation());
  480. slider->setMinimum(d->Slider->minimum());
  481. slider->setMaximum(d->Slider->maximum());
  482. slider->setValues(d->Slider->minimumValue(), d->Slider->maximumValue());
  483. slider->setSingleStep(d->Slider->singleStep());
  484. slider->setTracking(d->Slider->hasTracking());
  485. slider->setTickInterval(d->Slider->tickInterval());
  486. delete d->Slider;
  487. qobject_cast<QHBoxLayout*>(this->layout())->addWidget(slider);
  488. d->Slider = slider;
  489. d->connectSlider();
  490. }