DISTRHO Plugin Framework
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

968 lines
27KB

  1. /*
  2. * DISTRHO Plugin Framework (DPF)
  3. * Copyright (C) 2012-2022 Filipe Coelho <falktx@falktx.com>
  4. *
  5. * Permission to use, copy, modify, and/or distribute this software for any purpose with
  6. * or without fee is hereby granted, provided that the above copyright notice and this
  7. * permission notice appear in all copies.
  8. *
  9. * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH REGARD
  10. * TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS. IN
  11. * NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL
  12. * DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER
  13. * IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN
  14. * CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
  15. */
  16. #include "../ImageBaseWidgets.hpp"
  17. #include "../Color.hpp"
  18. START_NAMESPACE_DGL
  19. // --------------------------------------------------------------------------------------------------------------------
  20. template <class ImageType>
  21. ImageBaseAboutWindow<ImageType>::ImageBaseAboutWindow(Window& transientParentWindow, const ImageType& image)
  22. : StandaloneWindow(transientParentWindow.getApp(), transientParentWindow),
  23. img(image)
  24. {
  25. setResizable(false);
  26. setTitle("About");
  27. if (image.isValid())
  28. {
  29. setSize(image.getSize());
  30. setGeometryConstraints(image.getWidth(), image.getHeight(), true, true);
  31. }
  32. done();
  33. }
  34. template <class ImageType>
  35. ImageBaseAboutWindow<ImageType>::ImageBaseAboutWindow(TopLevelWidget* const topLevelWidget, const ImageType& image)
  36. : StandaloneWindow(topLevelWidget->getApp(), topLevelWidget->getWindow()),
  37. img(image)
  38. {
  39. setResizable(false);
  40. setTitle("About");
  41. if (image.isValid())
  42. {
  43. setSize(image.getSize());
  44. setGeometryConstraints(image.getWidth(), image.getHeight(), true, true);
  45. }
  46. done();
  47. }
  48. template <class ImageType>
  49. void ImageBaseAboutWindow<ImageType>::setImage(const ImageType& image)
  50. {
  51. if (img == image)
  52. return;
  53. if (image.isInvalid())
  54. {
  55. img = image;
  56. return;
  57. }
  58. reinit();
  59. img = image;
  60. setSize(image.getSize());
  61. setGeometryConstraints(image.getWidth(), image.getHeight(), true, true);
  62. done();
  63. }
  64. template <class ImageType>
  65. void ImageBaseAboutWindow<ImageType>::onDisplay()
  66. {
  67. img.draw(getGraphicsContext());
  68. }
  69. template <class ImageType>
  70. bool ImageBaseAboutWindow<ImageType>::onKeyboard(const KeyboardEvent& ev)
  71. {
  72. if (ev.press && ev.key == kKeyEscape)
  73. {
  74. close();
  75. return true;
  76. }
  77. return false;
  78. }
  79. template <class ImageType>
  80. bool ImageBaseAboutWindow<ImageType>::onMouse(const MouseEvent& ev)
  81. {
  82. if (ev.press)
  83. {
  84. close();
  85. return true;
  86. }
  87. return false;
  88. }
  89. // --------------------------------------------------------------------------------------------------------------------
  90. template <class ImageType>
  91. struct ImageBaseButton<ImageType>::PrivateData : public ButtonEventHandler::Callback {
  92. ImageBaseButton<ImageType>::Callback* callback;
  93. ImageType imageNormal;
  94. ImageType imageHover;
  95. ImageType imageDown;
  96. PrivateData(const ImageType& normal, const ImageType& hover, const ImageType& down)
  97. : callback(nullptr),
  98. imageNormal(normal),
  99. imageHover(hover),
  100. imageDown(down) {}
  101. void buttonClicked(SubWidget* widget, int button) override
  102. {
  103. if (callback != nullptr)
  104. if (ImageBaseButton* const imageButton = dynamic_cast<ImageBaseButton*>(widget))
  105. callback->imageButtonClicked(imageButton, button);
  106. }
  107. DISTRHO_DECLARE_NON_COPYABLE(PrivateData)
  108. };
  109. // --------------------------------------------------------------------------------------------------------------------
  110. template <class ImageType>
  111. ImageBaseButton<ImageType>::ImageBaseButton(Widget* const parentWidget, const ImageType& image)
  112. : SubWidget(parentWidget),
  113. ButtonEventHandler(this),
  114. pData(new PrivateData(image, image, image))
  115. {
  116. ButtonEventHandler::setCallback(pData);
  117. setSize(image.getSize());
  118. }
  119. template <class ImageType>
  120. ImageBaseButton<ImageType>::ImageBaseButton(Widget* const parentWidget, const ImageType& imageNormal, const ImageType& imageDown)
  121. : SubWidget(parentWidget),
  122. ButtonEventHandler(this),
  123. pData(new PrivateData(imageNormal, imageNormal, imageDown))
  124. {
  125. DISTRHO_SAFE_ASSERT(imageNormal.getSize() == imageDown.getSize());
  126. ButtonEventHandler::setCallback(pData);
  127. setSize(imageNormal.getSize());
  128. }
  129. template <class ImageType>
  130. ImageBaseButton<ImageType>::ImageBaseButton(Widget* const parentWidget, const ImageType& imageNormal, const ImageType& imageHover, const ImageType& imageDown)
  131. : SubWidget(parentWidget),
  132. ButtonEventHandler(this),
  133. pData(new PrivateData(imageNormal, imageHover, imageDown))
  134. {
  135. DISTRHO_SAFE_ASSERT(imageNormal.getSize() == imageHover.getSize() && imageHover.getSize() == imageDown.getSize());
  136. ButtonEventHandler::setCallback(pData);
  137. setSize(imageNormal.getSize());
  138. }
  139. template <class ImageType>
  140. ImageBaseButton<ImageType>::~ImageBaseButton()
  141. {
  142. delete pData;
  143. }
  144. template <class ImageType>
  145. void ImageBaseButton<ImageType>::setCallback(Callback* callback) noexcept
  146. {
  147. pData->callback = callback;
  148. }
  149. template <class ImageType>
  150. void ImageBaseButton<ImageType>::onDisplay()
  151. {
  152. const GraphicsContext& context(getGraphicsContext());
  153. const State state = ButtonEventHandler::getState();
  154. if (ButtonEventHandler::isCheckable())
  155. {
  156. if (ButtonEventHandler::isChecked())
  157. pData->imageDown.draw(context);
  158. else if (state & kButtonStateHover)
  159. pData->imageHover.draw(context);
  160. else
  161. pData->imageNormal.draw(context);
  162. }
  163. else
  164. {
  165. if (state & kButtonStateActive)
  166. pData->imageDown.draw(context);
  167. else if (state & kButtonStateHover)
  168. pData->imageHover.draw(context);
  169. else
  170. pData->imageNormal.draw(context);
  171. }
  172. }
  173. template <class ImageType>
  174. bool ImageBaseButton<ImageType>::onMouse(const MouseEvent& ev)
  175. {
  176. if (SubWidget::onMouse(ev))
  177. return true;
  178. return ButtonEventHandler::mouseEvent(ev);
  179. }
  180. template <class ImageType>
  181. bool ImageBaseButton<ImageType>::onMotion(const MotionEvent& ev)
  182. {
  183. if (SubWidget::onMotion(ev))
  184. return true;
  185. return ButtonEventHandler::motionEvent(ev);
  186. }
  187. // --------------------------------------------------------------------------------------------------------------------
  188. template <class ImageType>
  189. struct ImageBaseKnob<ImageType>::PrivateData : public KnobEventHandler::Callback {
  190. ImageBaseKnob<ImageType>::Callback* callback;
  191. ImageType image;
  192. int rotationAngle;
  193. bool alwaysRepaint;
  194. bool isImgVertical;
  195. uint imgLayerWidth;
  196. uint imgLayerHeight;
  197. uint imgLayerCount;
  198. bool isReady;
  199. union {
  200. uint glTextureId;
  201. void* cairoSurface;
  202. };
  203. explicit PrivateData(const ImageType& img)
  204. : callback(nullptr),
  205. image(img),
  206. rotationAngle(0),
  207. alwaysRepaint(false),
  208. isImgVertical(img.getHeight() > img.getWidth()),
  209. imgLayerWidth(isImgVertical ? img.getWidth() : img.getHeight()),
  210. imgLayerHeight(imgLayerWidth),
  211. imgLayerCount(isImgVertical ? img.getHeight()/imgLayerHeight : img.getWidth()/imgLayerWidth),
  212. isReady(false)
  213. {
  214. init();
  215. }
  216. explicit PrivateData(PrivateData* const other)
  217. : callback(other->callback),
  218. image(other->image),
  219. rotationAngle(other->rotationAngle),
  220. alwaysRepaint(other->alwaysRepaint),
  221. isImgVertical(other->isImgVertical),
  222. imgLayerWidth(other->imgLayerWidth),
  223. imgLayerHeight(other->imgLayerHeight),
  224. imgLayerCount(other->imgLayerCount),
  225. isReady(false)
  226. {
  227. init();
  228. }
  229. void assignFrom(PrivateData* const other)
  230. {
  231. cleanup();
  232. image = other->image;
  233. rotationAngle = other->rotationAngle;
  234. callback = other->callback;
  235. alwaysRepaint = other->alwaysRepaint;
  236. isImgVertical = other->isImgVertical;
  237. imgLayerWidth = other->imgLayerWidth;
  238. imgLayerHeight = other->imgLayerHeight;
  239. imgLayerCount = other->imgLayerCount;
  240. isReady = false;
  241. init();
  242. }
  243. ~PrivateData()
  244. {
  245. cleanup();
  246. }
  247. void knobDragStarted(SubWidget* const widget) override
  248. {
  249. if (callback != nullptr)
  250. if (ImageBaseKnob* const imageKnob = dynamic_cast<ImageBaseKnob*>(widget))
  251. callback->imageKnobDragStarted(imageKnob);
  252. }
  253. void knobDragFinished(SubWidget* const widget) override
  254. {
  255. if (callback != nullptr)
  256. if (ImageBaseKnob* const imageKnob = dynamic_cast<ImageBaseKnob*>(widget))
  257. callback->imageKnobDragFinished(imageKnob);
  258. }
  259. void knobValueChanged(SubWidget* const widget, const float value) override
  260. {
  261. if (rotationAngle == 0 || alwaysRepaint)
  262. isReady = false;
  263. if (callback != nullptr)
  264. if (ImageBaseKnob* const imageKnob = dynamic_cast<ImageBaseKnob*>(widget))
  265. callback->imageKnobValueChanged(imageKnob, value);
  266. }
  267. // implemented independently per graphics backend
  268. void init();
  269. void cleanup();
  270. DISTRHO_DECLARE_NON_COPYABLE(PrivateData)
  271. };
  272. // --------------------------------------------------------------------------------------------------------------------
  273. template <class ImageType>
  274. ImageBaseKnob<ImageType>::ImageBaseKnob(Widget* const parentWidget,
  275. const ImageType& image,
  276. const Orientation orientation) noexcept
  277. : SubWidget(parentWidget),
  278. KnobEventHandler(this),
  279. pData(new PrivateData(image))
  280. {
  281. KnobEventHandler::setCallback(pData);
  282. setOrientation(orientation);
  283. setSize(pData->imgLayerWidth, pData->imgLayerHeight);
  284. }
  285. template <class ImageType>
  286. ImageBaseKnob<ImageType>::ImageBaseKnob(const ImageBaseKnob<ImageType>& imageKnob)
  287. : SubWidget(imageKnob.getParentWidget()),
  288. KnobEventHandler(this, imageKnob),
  289. pData(new PrivateData(imageKnob.pData))
  290. {
  291. KnobEventHandler::setCallback(pData);
  292. setOrientation(imageKnob.getOrientation());
  293. setSize(pData->imgLayerWidth, pData->imgLayerHeight);
  294. }
  295. template <class ImageType>
  296. ImageBaseKnob<ImageType>& ImageBaseKnob<ImageType>::operator=(const ImageBaseKnob<ImageType>& imageKnob)
  297. {
  298. KnobEventHandler::operator=(imageKnob);
  299. pData->assignFrom(imageKnob.pData);
  300. setSize(pData->imgLayerWidth, pData->imgLayerHeight);
  301. return *this;
  302. }
  303. template <class ImageType>
  304. ImageBaseKnob<ImageType>::~ImageBaseKnob()
  305. {
  306. delete pData;
  307. }
  308. template <class ImageType>
  309. void ImageBaseKnob<ImageType>::setCallback(Callback* callback) noexcept
  310. {
  311. pData->callback = callback;
  312. }
  313. template <class ImageType>
  314. void ImageBaseKnob<ImageType>::setImageLayerCount(uint count) noexcept
  315. {
  316. DISTRHO_SAFE_ASSERT_RETURN(count > 1,);
  317. pData->imgLayerCount = count;
  318. if (pData->isImgVertical)
  319. pData->imgLayerHeight = pData->image.getHeight()/count;
  320. else
  321. pData->imgLayerWidth = pData->image.getWidth()/count;
  322. setSize(pData->imgLayerWidth, pData->imgLayerHeight);
  323. }
  324. template <class ImageType>
  325. void ImageBaseKnob<ImageType>::setRotationAngle(int angle)
  326. {
  327. if (pData->rotationAngle == angle)
  328. return;
  329. pData->rotationAngle = angle;
  330. pData->isReady = false;
  331. }
  332. template <class ImageType>
  333. bool ImageBaseKnob<ImageType>::setValue(float value, bool sendCallback) noexcept
  334. {
  335. if (KnobEventHandler::setValue(value, sendCallback))
  336. {
  337. if (pData->rotationAngle == 0 || pData->alwaysRepaint)
  338. pData->isReady = false;
  339. return true;
  340. }
  341. return false;
  342. }
  343. template <class ImageType>
  344. bool ImageBaseKnob<ImageType>::onMouse(const MouseEvent& ev)
  345. {
  346. if (SubWidget::onMouse(ev))
  347. return true;
  348. return KnobEventHandler::mouseEvent(ev, getTopLevelWidget()->getScaleFactor());
  349. }
  350. template <class ImageType>
  351. bool ImageBaseKnob<ImageType>::onMotion(const MotionEvent& ev)
  352. {
  353. if (SubWidget::onMotion(ev))
  354. return true;
  355. return KnobEventHandler::motionEvent(ev, getTopLevelWidget()->getScaleFactor());
  356. }
  357. template <class ImageType>
  358. bool ImageBaseKnob<ImageType>::onScroll(const ScrollEvent& ev)
  359. {
  360. if (SubWidget::onScroll(ev))
  361. return true;
  362. return KnobEventHandler::scrollEvent(ev);
  363. }
  364. // --------------------------------------------------------------------------------------------------------------------
  365. template <class ImageType>
  366. struct ImageBaseSlider<ImageType>::PrivateData {
  367. ImageType image;
  368. float minimum;
  369. float maximum;
  370. float step;
  371. float value;
  372. float valueDef;
  373. float valueTmp;
  374. bool usingDefault;
  375. bool dragging;
  376. bool checkable;
  377. bool inverted;
  378. bool valueIsSet;
  379. double startedX;
  380. double startedY;
  381. Callback* callback;
  382. Point<int> startPos;
  383. Point<int> endPos;
  384. Rectangle<double> sliderArea;
  385. PrivateData(const ImageType& img)
  386. : image(img),
  387. minimum(0.0f),
  388. maximum(1.0f),
  389. step(0.0f),
  390. value(0.5f),
  391. valueDef(value),
  392. valueTmp(value),
  393. usingDefault(false),
  394. dragging(false),
  395. checkable(false),
  396. inverted(false),
  397. valueIsSet(false),
  398. startedX(0.0),
  399. startedY(0.0),
  400. callback(nullptr),
  401. startPos(),
  402. endPos(),
  403. sliderArea() {}
  404. void recheckArea() noexcept
  405. {
  406. if (startPos.getY() == endPos.getY())
  407. {
  408. // horizontal
  409. sliderArea = Rectangle<double>(startPos.getX(),
  410. startPos.getY(),
  411. endPos.getX() + static_cast<int>(image.getWidth()) - startPos.getX(),
  412. static_cast<int>(image.getHeight()));
  413. }
  414. else
  415. {
  416. // vertical
  417. sliderArea = Rectangle<double>(startPos.getX(),
  418. startPos.getY(),
  419. static_cast<int>(image.getWidth()),
  420. endPos.getY() + static_cast<int>(image.getHeight()) - startPos.getY());
  421. }
  422. }
  423. DISTRHO_DECLARE_NON_COPYABLE(PrivateData)
  424. };
  425. // --------------------------------------------------------------------------------------------------------------------
  426. template <class ImageType>
  427. ImageBaseSlider<ImageType>::ImageBaseSlider(Widget* const parentWidget, const ImageType& image) noexcept
  428. : SubWidget(parentWidget),
  429. pData(new PrivateData(image))
  430. {
  431. setNeedsFullViewportDrawing();
  432. }
  433. template <class ImageType>
  434. ImageBaseSlider<ImageType>::~ImageBaseSlider()
  435. {
  436. delete pData;
  437. }
  438. template <class ImageType>
  439. float ImageBaseSlider<ImageType>::getValue() const noexcept
  440. {
  441. return pData->value;
  442. }
  443. template <class ImageType>
  444. void ImageBaseSlider<ImageType>::setValue(float value, bool sendCallback) noexcept
  445. {
  446. if (! pData->valueIsSet)
  447. pData->valueIsSet = true;
  448. if (d_isEqual(pData->value, value))
  449. return;
  450. pData->value = value;
  451. if (d_isZero(pData->step))
  452. pData->valueTmp = value;
  453. repaint();
  454. if (sendCallback && pData->callback != nullptr)
  455. {
  456. try {
  457. pData->callback->imageSliderValueChanged(this, pData->value);
  458. } DISTRHO_SAFE_EXCEPTION("ImageBaseSlider::setValue");
  459. }
  460. }
  461. template <class ImageType>
  462. void ImageBaseSlider<ImageType>::setStartPos(const Point<int>& startPos) noexcept
  463. {
  464. pData->startPos = startPos;
  465. pData->recheckArea();
  466. }
  467. template <class ImageType>
  468. void ImageBaseSlider<ImageType>::setStartPos(int x, int y) noexcept
  469. {
  470. setStartPos(Point<int>(x, y));
  471. }
  472. template <class ImageType>
  473. void ImageBaseSlider<ImageType>::setEndPos(const Point<int>& endPos) noexcept
  474. {
  475. pData->endPos = endPos;
  476. pData->recheckArea();
  477. }
  478. template <class ImageType>
  479. void ImageBaseSlider<ImageType>::setEndPos(int x, int y) noexcept
  480. {
  481. setEndPos(Point<int>(x, y));
  482. }
  483. template <class ImageType>
  484. void ImageBaseSlider<ImageType>::setCheckable(bool checkable) noexcept
  485. {
  486. if (pData->checkable == checkable)
  487. return;
  488. pData->checkable = checkable;
  489. repaint();
  490. }
  491. template <class ImageType>
  492. void ImageBaseSlider<ImageType>::setInverted(bool inverted) noexcept
  493. {
  494. if (pData->inverted == inverted)
  495. return;
  496. pData->inverted = inverted;
  497. repaint();
  498. }
  499. template <class ImageType>
  500. void ImageBaseSlider<ImageType>::setDefault(float value) noexcept
  501. {
  502. pData->valueDef = value;
  503. pData->usingDefault = true;
  504. }
  505. template <class ImageType>
  506. void ImageBaseSlider<ImageType>::setRange(float min, float max) noexcept
  507. {
  508. pData->minimum = min;
  509. pData->maximum = max;
  510. if (pData->value < min)
  511. {
  512. pData->value = min;
  513. repaint();
  514. if (pData->callback != nullptr && pData->valueIsSet)
  515. {
  516. try {
  517. pData->callback->imageSliderValueChanged(this, pData->value);
  518. } DISTRHO_SAFE_EXCEPTION("ImageBaseSlider::setRange < min");
  519. }
  520. }
  521. else if (pData->value > max)
  522. {
  523. pData->value = max;
  524. repaint();
  525. if (pData->callback != nullptr && pData->valueIsSet)
  526. {
  527. try {
  528. pData->callback->imageSliderValueChanged(this, pData->value);
  529. } DISTRHO_SAFE_EXCEPTION("ImageBaseSlider::setRange > max");
  530. }
  531. }
  532. }
  533. template <class ImageType>
  534. void ImageBaseSlider<ImageType>::setStep(float step) noexcept
  535. {
  536. pData->step = step;
  537. }
  538. template <class ImageType>
  539. void ImageBaseSlider<ImageType>::setCallback(Callback* callback) noexcept
  540. {
  541. pData->callback = callback;
  542. }
  543. template <class ImageType>
  544. void ImageBaseSlider<ImageType>::onDisplay()
  545. {
  546. const GraphicsContext& context(getGraphicsContext());
  547. #if 0 // DEBUG, paints slider area
  548. Color(1.0f, 1.0f, 1.0f, 0.5f).setFor(context, true);
  549. Rectangle<int>(pData->sliderArea.getX(),
  550. pData->sliderArea.getY(),
  551. pData->sliderArea.getX()+pData->sliderArea.getWidth(),
  552. pData->sliderArea.getY()+pData->sliderArea.getHeight()).draw(context);
  553. Color(1.0f, 1.0f, 1.0f, 1.0f).setFor(context, true);
  554. #endif
  555. const float normValue = (pData->value - pData->minimum) / (pData->maximum - pData->minimum);
  556. int x, y;
  557. if (pData->startPos.getY() == pData->endPos.getY())
  558. {
  559. // horizontal
  560. if (pData->inverted)
  561. x = pData->endPos.getX() - static_cast<int>(normValue*static_cast<float>(pData->endPos.getX()-pData->startPos.getX()));
  562. else
  563. x = pData->startPos.getX() + static_cast<int>(normValue*static_cast<float>(pData->endPos.getX()-pData->startPos.getX()));
  564. y = pData->startPos.getY();
  565. }
  566. else
  567. {
  568. // vertical
  569. x = pData->startPos.getX();
  570. if (pData->inverted)
  571. y = pData->endPos.getY() - static_cast<int>(normValue*static_cast<float>(pData->endPos.getY()-pData->startPos.getY()));
  572. else
  573. y = pData->startPos.getY() + static_cast<int>(normValue*static_cast<float>(pData->endPos.getY()-pData->startPos.getY()));
  574. }
  575. pData->image.drawAt(context, x, y);
  576. }
  577. template <class ImageType>
  578. bool ImageBaseSlider<ImageType>::onMouse(const MouseEvent& ev)
  579. {
  580. if (ev.button != 1)
  581. return false;
  582. if (ev.press)
  583. {
  584. if (! pData->sliderArea.contains(ev.pos))
  585. return false;
  586. if ((ev.mod & kModifierShift) != 0 && pData->usingDefault)
  587. {
  588. setValue(pData->valueDef, true);
  589. pData->valueTmp = pData->value;
  590. return true;
  591. }
  592. if (pData->checkable)
  593. {
  594. const float value = d_isEqual(pData->valueTmp, pData->minimum) ? pData->maximum : pData->minimum;
  595. setValue(value, true);
  596. pData->valueTmp = pData->value;
  597. return true;
  598. }
  599. float vper;
  600. const double x = ev.pos.getX();
  601. const double y = ev.pos.getY();
  602. if (pData->startPos.getY() == pData->endPos.getY())
  603. {
  604. // horizontal
  605. vper = float(x - pData->sliderArea.getX()) / float(pData->sliderArea.getWidth());
  606. }
  607. else
  608. {
  609. // vertical
  610. vper = float(y - pData->sliderArea.getY()) / float(pData->sliderArea.getHeight());
  611. }
  612. float value;
  613. if (pData->inverted)
  614. value = pData->maximum - vper * (pData->maximum - pData->minimum);
  615. else
  616. value = pData->minimum + vper * (pData->maximum - pData->minimum);
  617. if (value < pData->minimum)
  618. {
  619. pData->valueTmp = value = pData->minimum;
  620. }
  621. else if (value > pData->maximum)
  622. {
  623. pData->valueTmp = value = pData->maximum;
  624. }
  625. else if (d_isNotZero(pData->step))
  626. {
  627. pData->valueTmp = value;
  628. const float rest = std::fmod(value, pData->step);
  629. value = value - rest + (rest > pData->step/2.0f ? pData->step : 0.0f);
  630. }
  631. pData->dragging = true;
  632. pData->startedX = x;
  633. pData->startedY = y;
  634. if (pData->callback != nullptr)
  635. pData->callback->imageSliderDragStarted(this);
  636. setValue(value, true);
  637. return true;
  638. }
  639. else if (pData->dragging)
  640. {
  641. if (pData->callback != nullptr)
  642. pData->callback->imageSliderDragFinished(this);
  643. pData->dragging = false;
  644. return true;
  645. }
  646. return false;
  647. }
  648. template <class ImageType>
  649. bool ImageBaseSlider<ImageType>::onMotion(const MotionEvent& ev)
  650. {
  651. if (! pData->dragging)
  652. return false;
  653. const bool horizontal = pData->startPos.getY() == pData->endPos.getY();
  654. const double x = ev.pos.getX();
  655. const double y = ev.pos.getY();
  656. if ((horizontal && pData->sliderArea.containsX(x)) || (pData->sliderArea.containsY(y) && ! horizontal))
  657. {
  658. float vper;
  659. if (horizontal)
  660. {
  661. // horizontal
  662. vper = float(x - pData->sliderArea.getX()) / float(pData->sliderArea.getWidth());
  663. }
  664. else
  665. {
  666. // vertical
  667. vper = float(y - pData->sliderArea.getY()) / float(pData->sliderArea.getHeight());
  668. }
  669. float value;
  670. if (pData->inverted)
  671. value = pData->maximum - vper * (pData->maximum - pData->minimum);
  672. else
  673. value = pData->minimum + vper * (pData->maximum - pData->minimum);
  674. if (value < pData->minimum)
  675. {
  676. pData->valueTmp = value = pData->minimum;
  677. }
  678. else if (value > pData->maximum)
  679. {
  680. pData->valueTmp = value = pData->maximum;
  681. }
  682. else if (d_isNotZero(pData->step))
  683. {
  684. pData->valueTmp = value;
  685. const float rest = std::fmod(value, pData->step);
  686. value = value - rest + (rest > pData->step/2.0f ? pData->step : 0.0f);
  687. }
  688. setValue(value, true);
  689. }
  690. else if (horizontal)
  691. {
  692. if (x < pData->sliderArea.getX())
  693. setValue(pData->inverted ? pData->maximum : pData->minimum, true);
  694. else
  695. setValue(pData->inverted ? pData->minimum : pData->maximum, true);
  696. }
  697. else
  698. {
  699. if (y < pData->sliderArea.getY())
  700. setValue(pData->inverted ? pData->maximum : pData->minimum, true);
  701. else
  702. setValue(pData->inverted ? pData->minimum : pData->maximum, true);
  703. }
  704. return true;
  705. }
  706. // --------------------------------------------------------------------------------------------------------------------
  707. template <class ImageType>
  708. struct ImageBaseSwitch<ImageType>::PrivateData {
  709. ImageType imageNormal;
  710. ImageType imageDown;
  711. bool isDown;
  712. Callback* callback;
  713. PrivateData(const ImageType& normal, const ImageType& down)
  714. : imageNormal(normal),
  715. imageDown(down),
  716. isDown(false),
  717. callback(nullptr)
  718. {
  719. DISTRHO_SAFE_ASSERT(imageNormal.getSize() == imageDown.getSize());
  720. }
  721. PrivateData(PrivateData* const other)
  722. : imageNormal(other->imageNormal),
  723. imageDown(other->imageDown),
  724. isDown(other->isDown),
  725. callback(other->callback)
  726. {
  727. DISTRHO_SAFE_ASSERT(imageNormal.getSize() == imageDown.getSize());
  728. }
  729. void assignFrom(PrivateData* const other)
  730. {
  731. imageNormal = other->imageNormal;
  732. imageDown = other->imageDown;
  733. isDown = other->isDown;
  734. callback = other->callback;
  735. DISTRHO_SAFE_ASSERT(imageNormal.getSize() == imageDown.getSize());
  736. }
  737. DISTRHO_DECLARE_NON_COPYABLE(PrivateData)
  738. };
  739. // --------------------------------------------------------------------------------------------------------------------
  740. template <class ImageType>
  741. ImageBaseSwitch<ImageType>::ImageBaseSwitch(Widget* const parentWidget, const ImageType& imageNormal, const ImageType& imageDown) noexcept
  742. : SubWidget(parentWidget),
  743. pData(new PrivateData(imageNormal, imageDown))
  744. {
  745. setSize(imageNormal.getSize());
  746. }
  747. template <class ImageType>
  748. ImageBaseSwitch<ImageType>::ImageBaseSwitch(const ImageBaseSwitch<ImageType>& imageSwitch) noexcept
  749. : SubWidget(imageSwitch.getParentWidget()),
  750. pData(new PrivateData(imageSwitch.pData))
  751. {
  752. setSize(pData->imageNormal.getSize());
  753. }
  754. template <class ImageType>
  755. ImageBaseSwitch<ImageType>& ImageBaseSwitch<ImageType>::operator=(const ImageBaseSwitch<ImageType>& imageSwitch) noexcept
  756. {
  757. pData->assignFrom(imageSwitch.pData);
  758. setSize(pData->imageNormal.getSize());
  759. return *this;
  760. }
  761. template <class ImageType>
  762. ImageBaseSwitch<ImageType>::~ImageBaseSwitch()
  763. {
  764. delete pData;
  765. }
  766. template <class ImageType>
  767. bool ImageBaseSwitch<ImageType>::isDown() const noexcept
  768. {
  769. return pData->isDown;
  770. }
  771. template <class ImageType>
  772. void ImageBaseSwitch<ImageType>::setDown(const bool down) noexcept
  773. {
  774. if (pData->isDown == down)
  775. return;
  776. pData->isDown = down;
  777. repaint();
  778. }
  779. template <class ImageType>
  780. void ImageBaseSwitch<ImageType>::setCallback(Callback* const callback) noexcept
  781. {
  782. pData->callback = callback;
  783. }
  784. template <class ImageType>
  785. void ImageBaseSwitch<ImageType>::onDisplay()
  786. {
  787. const GraphicsContext& context(getGraphicsContext());
  788. if (pData->isDown)
  789. pData->imageDown.draw(context);
  790. else
  791. pData->imageNormal.draw(context);
  792. }
  793. template <class ImageType>
  794. bool ImageBaseSwitch<ImageType>::onMouse(const MouseEvent& ev)
  795. {
  796. if (ev.press && contains(ev.pos))
  797. {
  798. pData->isDown = !pData->isDown;
  799. repaint();
  800. if (pData->callback != nullptr)
  801. pData->callback->imageSwitchClicked(this, pData->isDown);
  802. return true;
  803. }
  804. return false;
  805. }
  806. // --------------------------------------------------------------------------------------------------------------------
  807. END_NAMESPACE_DGL