juce_MouseInputSource.cpp 27 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702
  1. /*
  2. ==============================================================================
  3. This file is part of the JUCE library.
  4. Copyright (c) 2015 - ROLI Ltd.
  5. Permission is granted to use this software under the terms of either:
  6. a) the GPL v2 (or any later version)
  7. b) the Affero GPL v3
  8. Details of these licenses can be found at: www.gnu.org/licenses
  9. JUCE is distributed in the hope that it will be useful, but WITHOUT ANY
  10. WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR
  11. A PARTICULAR PURPOSE. See the GNU General Public License for more details.
  12. ------------------------------------------------------------------------------
  13. To release a closed-source product which uses JUCE, commercial licenses are
  14. available: visit www.juce.com for more information.
  15. ==============================================================================
  16. */
  17. class MouseInputSourceInternal : private AsyncUpdater
  18. {
  19. public:
  20. //==============================================================================
  21. MouseInputSourceInternal (const int i, const bool isMouse)
  22. : index (i), isMouseDevice (isMouse), pressure (0.0f),
  23. isUnboundedMouseModeOn (false), isCursorVisibleUntilOffscreen (false),
  24. lastPeer (nullptr), currentCursorHandle (nullptr),
  25. mouseEventCounter (0), mouseMovedSignificantlySincePressed (false)
  26. {
  27. }
  28. //==============================================================================
  29. bool isDragging() const noexcept
  30. {
  31. return buttonState.isAnyMouseButtonDown();
  32. }
  33. Component* getComponentUnderMouse() const noexcept
  34. {
  35. return componentUnderMouse.get();
  36. }
  37. ModifierKeys getCurrentModifiers() const noexcept
  38. {
  39. return ModifierKeys::getCurrentModifiers().withoutMouseButtons().withFlags (buttonState.getRawFlags());
  40. }
  41. ComponentPeer* getPeer() noexcept
  42. {
  43. if (! ComponentPeer::isValidPeer (lastPeer))
  44. lastPeer = nullptr;
  45. return lastPeer;
  46. }
  47. static Point<float> screenPosToLocalPos (Component& comp, Point<float> pos)
  48. {
  49. if (ComponentPeer* const peer = comp.getPeer())
  50. {
  51. pos = peer->globalToLocal (pos);
  52. Component& peerComp = peer->getComponent();
  53. return comp.getLocalPoint (&peerComp, ScalingHelpers::unscaledScreenPosToScaled (peerComp, pos));
  54. }
  55. return comp.getLocalPoint (nullptr, ScalingHelpers::unscaledScreenPosToScaled (comp, pos));
  56. }
  57. Component* findComponentAt (Point<float> screenPos)
  58. {
  59. if (ComponentPeer* const peer = getPeer())
  60. {
  61. Point<float> relativePos (ScalingHelpers::unscaledScreenPosToScaled (peer->getComponent(),
  62. peer->globalToLocal (screenPos)));
  63. Component& comp = peer->getComponent();
  64. const Point<int> pos (relativePos.roundToInt());
  65. // (the contains() call is needed to test for overlapping desktop windows)
  66. if (comp.contains (pos))
  67. return comp.getComponentAt (pos);
  68. }
  69. return nullptr;
  70. }
  71. Point<float> getScreenPosition() const
  72. {
  73. // This needs to return the live position if possible, but it mustn't update the lastScreenPos
  74. // value, because that can cause continuity problems.
  75. return ScalingHelpers::unscaledScreenPosToScaled
  76. (unboundedMouseOffset + (isMouseDevice ? MouseInputSource::getCurrentRawMousePosition()
  77. : lastScreenPos));
  78. }
  79. void setScreenPosition (Point<float> p)
  80. {
  81. MouseInputSource::setRawMousePosition (ScalingHelpers::scaledScreenPosToUnscaled (p));
  82. }
  83. bool isPressureValid() const noexcept { return pressure > 0.0f && pressure < 1.0f; }
  84. //==============================================================================
  85. #if JUCE_DUMP_MOUSE_EVENTS
  86. #define JUCE_MOUSE_EVENT_DBG(desc) DBG ("Mouse " << desc << " #" << index \
  87. << ": " << screenPosToLocalPos (comp, screenPos).toString() \
  88. << " - Comp: " << String::toHexString ((pointer_sized_int) &comp));
  89. #else
  90. #define JUCE_MOUSE_EVENT_DBG(desc)
  91. #endif
  92. void sendMouseEnter (Component& comp, Point<float> screenPos, Time time)
  93. {
  94. JUCE_MOUSE_EVENT_DBG ("enter")
  95. comp.internalMouseEnter (MouseInputSource (this), screenPosToLocalPos (comp, screenPos), time);
  96. }
  97. void sendMouseExit (Component& comp, Point<float> screenPos, Time time)
  98. {
  99. JUCE_MOUSE_EVENT_DBG ("exit")
  100. comp.internalMouseExit (MouseInputSource (this), screenPosToLocalPos (comp, screenPos), time);
  101. }
  102. void sendMouseMove (Component& comp, Point<float> screenPos, Time time)
  103. {
  104. JUCE_MOUSE_EVENT_DBG ("move")
  105. comp.internalMouseMove (MouseInputSource (this), screenPosToLocalPos (comp, screenPos), time);
  106. }
  107. void sendMouseDown (Component& comp, Point<float> screenPos, Time time)
  108. {
  109. JUCE_MOUSE_EVENT_DBG ("down")
  110. comp.internalMouseDown (MouseInputSource (this), screenPosToLocalPos (comp, screenPos), time, pressure);
  111. }
  112. void sendMouseDrag (Component& comp, Point<float> screenPos, Time time)
  113. {
  114. JUCE_MOUSE_EVENT_DBG ("drag")
  115. comp.internalMouseDrag (MouseInputSource (this), screenPosToLocalPos (comp, screenPos), time, pressure);
  116. }
  117. void sendMouseUp (Component& comp, Point<float> screenPos, Time time, const ModifierKeys oldMods)
  118. {
  119. JUCE_MOUSE_EVENT_DBG ("up")
  120. comp.internalMouseUp (MouseInputSource (this), screenPosToLocalPos (comp, screenPos), time, oldMods);
  121. }
  122. void sendMouseWheel (Component& comp, Point<float> screenPos, Time time, const MouseWheelDetails& wheel)
  123. {
  124. JUCE_MOUSE_EVENT_DBG ("wheel")
  125. comp.internalMouseWheel (MouseInputSource (this), screenPosToLocalPos (comp, screenPos), time, wheel);
  126. }
  127. void sendMagnifyGesture (Component& comp, Point<float> screenPos, Time time, const float amount)
  128. {
  129. JUCE_MOUSE_EVENT_DBG ("magnify")
  130. comp.internalMagnifyGesture (MouseInputSource (this), screenPosToLocalPos (comp, screenPos), time, amount);
  131. }
  132. //==============================================================================
  133. // (returns true if the button change caused a modal event loop)
  134. bool setButtons (Point<float> screenPos, Time time, const ModifierKeys newButtonState)
  135. {
  136. if (buttonState == newButtonState)
  137. return false;
  138. // (avoid sending a spurious mouse-drag when we receive a mouse-up)
  139. if (! (isDragging() && ! newButtonState.isAnyMouseButtonDown()))
  140. setScreenPos (screenPos, time, false);
  141. // (ignore secondary clicks when there's already a button down)
  142. if (buttonState.isAnyMouseButtonDown() == newButtonState.isAnyMouseButtonDown())
  143. {
  144. buttonState = newButtonState;
  145. return false;
  146. }
  147. const int lastCounter = mouseEventCounter;
  148. if (buttonState.isAnyMouseButtonDown())
  149. {
  150. if (Component* const current = getComponentUnderMouse())
  151. {
  152. const ModifierKeys oldMods (getCurrentModifiers());
  153. buttonState = newButtonState; // must change this before calling sendMouseUp, in case it runs a modal loop
  154. sendMouseUp (*current, screenPos + unboundedMouseOffset, time, oldMods);
  155. if (lastCounter != mouseEventCounter)
  156. return true; // if a modal loop happened, then newButtonState is no longer valid.
  157. }
  158. enableUnboundedMouseMovement (false, false);
  159. }
  160. buttonState = newButtonState;
  161. if (buttonState.isAnyMouseButtonDown())
  162. {
  163. Desktop::getInstance().incrementMouseClickCounter();
  164. if (Component* const current = getComponentUnderMouse())
  165. {
  166. registerMouseDown (screenPos, time, *current, buttonState);
  167. sendMouseDown (*current, screenPos, time);
  168. }
  169. }
  170. return lastCounter != mouseEventCounter;
  171. }
  172. void setComponentUnderMouse (Component* const newComponent, Point<float> screenPos, Time time)
  173. {
  174. Component* current = getComponentUnderMouse();
  175. if (newComponent != current)
  176. {
  177. WeakReference<Component> safeNewComp (newComponent);
  178. const ModifierKeys originalButtonState (buttonState);
  179. if (current != nullptr)
  180. {
  181. WeakReference<Component> safeOldComp (current);
  182. setButtons (screenPos, time, ModifierKeys());
  183. if (safeOldComp != nullptr)
  184. {
  185. componentUnderMouse = safeNewComp;
  186. sendMouseExit (*safeOldComp, screenPos, time);
  187. }
  188. buttonState = originalButtonState;
  189. }
  190. current = componentUnderMouse = safeNewComp;
  191. if (current != nullptr)
  192. sendMouseEnter (*current, screenPos, time);
  193. revealCursor (false);
  194. setButtons (screenPos, time, originalButtonState);
  195. }
  196. }
  197. void setPeer (ComponentPeer& newPeer, Point<float> screenPos, Time time)
  198. {
  199. ModifierKeys::updateCurrentModifiers();
  200. if (&newPeer != lastPeer)
  201. {
  202. setComponentUnderMouse (nullptr, screenPos, time);
  203. lastPeer = &newPeer;
  204. setComponentUnderMouse (findComponentAt (screenPos), screenPos, time);
  205. }
  206. }
  207. void setScreenPos (Point<float> newScreenPos, Time time, const bool forceUpdate)
  208. {
  209. if (! isDragging())
  210. setComponentUnderMouse (findComponentAt (newScreenPos), newScreenPos, time);
  211. if (newScreenPos != lastScreenPos || forceUpdate)
  212. {
  213. cancelPendingUpdate();
  214. lastScreenPos = newScreenPos;
  215. if (Component* const current = getComponentUnderMouse())
  216. {
  217. if (isDragging())
  218. {
  219. registerMouseDrag (newScreenPos);
  220. sendMouseDrag (*current, newScreenPos + unboundedMouseOffset, time);
  221. if (isUnboundedMouseModeOn)
  222. handleUnboundedDrag (*current);
  223. }
  224. else
  225. {
  226. sendMouseMove (*current, newScreenPos, time);
  227. }
  228. }
  229. revealCursor (false);
  230. }
  231. }
  232. //==============================================================================
  233. void handleEvent (ComponentPeer& newPeer, Point<float> positionWithinPeer, Time time,
  234. const ModifierKeys newMods, float newPressure)
  235. {
  236. lastTime = time;
  237. const bool pressureChanged = (pressure != newPressure);
  238. pressure = newPressure;
  239. ++mouseEventCounter;
  240. const Point<float> screenPos (newPeer.localToGlobal (positionWithinPeer));
  241. if (isDragging() && newMods.isAnyMouseButtonDown())
  242. {
  243. setScreenPos (screenPos, time, pressureChanged);
  244. }
  245. else
  246. {
  247. setPeer (newPeer, screenPos, time);
  248. if (ComponentPeer* peer = getPeer())
  249. {
  250. if (setButtons (screenPos, time, newMods))
  251. return; // some modal events have been dispatched, so the current event is now out-of-date
  252. peer = getPeer();
  253. if (peer != nullptr)
  254. setScreenPos (screenPos, time, pressureChanged);
  255. }
  256. }
  257. }
  258. Component* getTargetForGesture (ComponentPeer& peer, Point<float> positionWithinPeer,
  259. Time time, Point<float>& screenPos)
  260. {
  261. lastTime = time;
  262. ++mouseEventCounter;
  263. screenPos = peer.localToGlobal (positionWithinPeer);
  264. setPeer (peer, screenPos, time);
  265. setScreenPos (screenPos, time, false);
  266. triggerFakeMove();
  267. return getComponentUnderMouse();
  268. }
  269. void handleWheel (ComponentPeer& peer, Point<float> positionWithinPeer,
  270. Time time, const MouseWheelDetails& wheel)
  271. {
  272. Desktop::getInstance().incrementMouseWheelCounter();
  273. Point<float> screenPos;
  274. // This will make sure that when the wheel spins in its inertial phase, any events
  275. // continue to be sent to the last component that the mouse was over when it was being
  276. // actively controlled by the user. This avoids confusion when scrolling through nested
  277. // scrollable components.
  278. if (lastNonInertialWheelTarget == nullptr || ! wheel.isInertial)
  279. lastNonInertialWheelTarget = getTargetForGesture (peer, positionWithinPeer, time, screenPos);
  280. if (Component* target = lastNonInertialWheelTarget)
  281. sendMouseWheel (*target, screenPos, time, wheel);
  282. }
  283. void handleMagnifyGesture (ComponentPeer& peer, Point<float> positionWithinPeer,
  284. Time time, const float scaleFactor)
  285. {
  286. Point<float> screenPos;
  287. if (Component* current = getTargetForGesture (peer, positionWithinPeer, time, screenPos))
  288. sendMagnifyGesture (*current, screenPos, time, scaleFactor);
  289. }
  290. //==============================================================================
  291. Time getLastMouseDownTime() const noexcept { return mouseDowns[0].time; }
  292. Point<float> getLastMouseDownPosition() const noexcept { return ScalingHelpers::unscaledScreenPosToScaled (mouseDowns[0].position); }
  293. int getNumberOfMultipleClicks() const noexcept
  294. {
  295. int numClicks = 1;
  296. if (! hasMouseMovedSignificantlySincePressed())
  297. {
  298. for (int i = 1; i < numElementsInArray (mouseDowns); ++i)
  299. {
  300. if (mouseDowns[0].canBePartOfMultipleClickWith (mouseDowns[i], MouseEvent::getDoubleClickTimeout() * jmin (i, 2)))
  301. ++numClicks;
  302. else
  303. break;
  304. }
  305. }
  306. return numClicks;
  307. }
  308. bool hasMouseMovedSignificantlySincePressed() const noexcept
  309. {
  310. return mouseMovedSignificantlySincePressed
  311. || lastTime > mouseDowns[0].time + RelativeTime::milliseconds (300);
  312. }
  313. //==============================================================================
  314. void triggerFakeMove()
  315. {
  316. triggerAsyncUpdate();
  317. }
  318. void handleAsyncUpdate() override
  319. {
  320. setScreenPos (lastScreenPos, jmax (lastTime, Time::getCurrentTime()), true);
  321. }
  322. //==============================================================================
  323. void enableUnboundedMouseMovement (bool enable, bool keepCursorVisibleUntilOffscreen)
  324. {
  325. enable = enable && isDragging();
  326. isCursorVisibleUntilOffscreen = keepCursorVisibleUntilOffscreen;
  327. if (enable != isUnboundedMouseModeOn)
  328. {
  329. if ((! enable) && ((! isCursorVisibleUntilOffscreen) || ! unboundedMouseOffset.isOrigin()))
  330. {
  331. // when released, return the mouse to within the component's bounds
  332. if (Component* current = getComponentUnderMouse())
  333. setScreenPosition (current->getScreenBounds().toFloat()
  334. .getConstrainedPoint (ScalingHelpers::unscaledScreenPosToScaled (lastScreenPos)));
  335. }
  336. isUnboundedMouseModeOn = enable;
  337. unboundedMouseOffset = Point<float>();
  338. revealCursor (true);
  339. }
  340. }
  341. void handleUnboundedDrag (Component& current)
  342. {
  343. const Rectangle<float> componentScreenBounds
  344. = ScalingHelpers::scaledScreenPosToUnscaled (current.getParentMonitorArea().reduced (2, 2).toFloat());
  345. if (! componentScreenBounds.contains (lastScreenPos))
  346. {
  347. const Point<float> componentCentre (current.getScreenBounds().toFloat().getCentre());
  348. unboundedMouseOffset += (lastScreenPos - ScalingHelpers::scaledScreenPosToUnscaled (componentCentre));
  349. setScreenPosition (componentCentre);
  350. }
  351. else if (isCursorVisibleUntilOffscreen
  352. && (! unboundedMouseOffset.isOrigin())
  353. && componentScreenBounds.contains (lastScreenPos + unboundedMouseOffset))
  354. {
  355. MouseInputSource::setRawMousePosition (lastScreenPos + unboundedMouseOffset);
  356. unboundedMouseOffset = Point<float>();
  357. }
  358. }
  359. //==============================================================================
  360. void showMouseCursor (MouseCursor cursor, bool forcedUpdate)
  361. {
  362. if (isUnboundedMouseModeOn && ((! unboundedMouseOffset.isOrigin()) || ! isCursorVisibleUntilOffscreen))
  363. {
  364. cursor = MouseCursor::NoCursor;
  365. forcedUpdate = true;
  366. }
  367. if (forcedUpdate || cursor.getHandle() != currentCursorHandle)
  368. {
  369. currentCursorHandle = cursor.getHandle();
  370. cursor.showInWindow (getPeer());
  371. }
  372. }
  373. void hideCursor()
  374. {
  375. showMouseCursor (MouseCursor::NoCursor, true);
  376. }
  377. void revealCursor (bool forcedUpdate)
  378. {
  379. MouseCursor mc (MouseCursor::NormalCursor);
  380. if (Component* current = getComponentUnderMouse())
  381. mc = current->getLookAndFeel().getMouseCursorFor (*current);
  382. showMouseCursor (mc, forcedUpdate);
  383. }
  384. //==============================================================================
  385. const int index;
  386. const bool isMouseDevice;
  387. Point<float> lastScreenPos, unboundedMouseOffset; // NB: these are unscaled coords
  388. ModifierKeys buttonState;
  389. float pressure;
  390. bool isUnboundedMouseModeOn, isCursorVisibleUntilOffscreen;
  391. private:
  392. WeakReference<Component> componentUnderMouse, lastNonInertialWheelTarget;
  393. ComponentPeer* lastPeer;
  394. void* currentCursorHandle;
  395. int mouseEventCounter;
  396. struct RecentMouseDown
  397. {
  398. RecentMouseDown() noexcept : peerID (0) {}
  399. Point<float> position;
  400. Time time;
  401. ModifierKeys buttons;
  402. uint32 peerID;
  403. bool canBePartOfMultipleClickWith (const RecentMouseDown& other, const int maxTimeBetweenMs) const
  404. {
  405. return time - other.time < RelativeTime::milliseconds (maxTimeBetweenMs)
  406. && std::abs (position.x - other.position.x) < 8
  407. && std::abs (position.y - other.position.y) < 8
  408. && buttons == other.buttons
  409. && peerID == other.peerID;
  410. }
  411. };
  412. RecentMouseDown mouseDowns[4];
  413. Time lastTime;
  414. bool mouseMovedSignificantlySincePressed;
  415. void registerMouseDown (Point<float> screenPos, Time time,
  416. Component& component, const ModifierKeys modifiers) noexcept
  417. {
  418. for (int i = numElementsInArray (mouseDowns); --i > 0;)
  419. mouseDowns[i] = mouseDowns[i - 1];
  420. mouseDowns[0].position = screenPos;
  421. mouseDowns[0].time = time;
  422. mouseDowns[0].buttons = modifiers.withOnlyMouseButtons();
  423. if (ComponentPeer* const peer = component.getPeer())
  424. mouseDowns[0].peerID = peer->getUniqueID();
  425. else
  426. mouseDowns[0].peerID = 0;
  427. mouseMovedSignificantlySincePressed = false;
  428. lastNonInertialWheelTarget = nullptr;
  429. }
  430. void registerMouseDrag (Point<float> screenPos) noexcept
  431. {
  432. mouseMovedSignificantlySincePressed = mouseMovedSignificantlySincePressed
  433. || mouseDowns[0].position.getDistanceFrom (screenPos) >= 4;
  434. }
  435. JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (MouseInputSourceInternal)
  436. };
  437. //==============================================================================
  438. MouseInputSource::MouseInputSource (MouseInputSourceInternal* s) noexcept : pimpl (s) {}
  439. MouseInputSource::MouseInputSource (const MouseInputSource& other) noexcept : pimpl (other.pimpl) {}
  440. MouseInputSource::~MouseInputSource() noexcept {}
  441. MouseInputSource& MouseInputSource::operator= (const MouseInputSource& other) noexcept
  442. {
  443. pimpl = other.pimpl;
  444. return *this;
  445. }
  446. bool MouseInputSource::isMouse() const noexcept { return pimpl->isMouseDevice; }
  447. bool MouseInputSource::isTouch() const noexcept { return ! isMouse(); }
  448. bool MouseInputSource::canHover() const noexcept { return isMouse(); }
  449. bool MouseInputSource::hasMouseWheel() const noexcept { return isMouse(); }
  450. int MouseInputSource::getIndex() const noexcept { return pimpl->index; }
  451. bool MouseInputSource::isDragging() const noexcept { return pimpl->isDragging(); }
  452. Point<float> MouseInputSource::getScreenPosition() const noexcept { return pimpl->getScreenPosition(); }
  453. ModifierKeys MouseInputSource::getCurrentModifiers() const noexcept { return pimpl->getCurrentModifiers(); }
  454. float MouseInputSource::getCurrentPressure() const noexcept { return pimpl->pressure; }
  455. bool MouseInputSource::isPressureValid() const noexcept { return pimpl->isPressureValid(); }
  456. Component* MouseInputSource::getComponentUnderMouse() const { return pimpl->getComponentUnderMouse(); }
  457. void MouseInputSource::triggerFakeMove() const { pimpl->triggerFakeMove(); }
  458. int MouseInputSource::getNumberOfMultipleClicks() const noexcept { return pimpl->getNumberOfMultipleClicks(); }
  459. Time MouseInputSource::getLastMouseDownTime() const noexcept { return pimpl->getLastMouseDownTime(); }
  460. Point<float> MouseInputSource::getLastMouseDownPosition() const noexcept { return pimpl->getLastMouseDownPosition(); }
  461. bool MouseInputSource::hasMouseMovedSignificantlySincePressed() const noexcept { return pimpl->hasMouseMovedSignificantlySincePressed(); }
  462. bool MouseInputSource::canDoUnboundedMovement() const noexcept { return isMouse(); }
  463. void MouseInputSource::enableUnboundedMouseMovement (bool isEnabled, bool keepCursorVisibleUntilOffscreen) const
  464. { pimpl->enableUnboundedMouseMovement (isEnabled, keepCursorVisibleUntilOffscreen); }
  465. bool MouseInputSource::isUnboundedMouseMovementEnabled() const { return pimpl->isUnboundedMouseModeOn; }
  466. bool MouseInputSource::hasMouseCursor() const noexcept { return isMouse(); }
  467. void MouseInputSource::showMouseCursor (const MouseCursor& cursor) { pimpl->showMouseCursor (cursor, false); }
  468. void MouseInputSource::hideCursor() { pimpl->hideCursor(); }
  469. void MouseInputSource::revealCursor() { pimpl->revealCursor (false); }
  470. void MouseInputSource::forceMouseCursorUpdate() { pimpl->revealCursor (true); }
  471. void MouseInputSource::setScreenPosition (Point<float> p) { pimpl->setScreenPosition (p); }
  472. void MouseInputSource::handleEvent (ComponentPeer& peer, Point<float> pos, int64 time, ModifierKeys mods, float pressure)
  473. {
  474. pimpl->handleEvent (peer, pos, Time (time), mods.withOnlyMouseButtons(), pressure);
  475. }
  476. void MouseInputSource::handleWheel (ComponentPeer& peer, Point<float> pos, int64 time, const MouseWheelDetails& wheel)
  477. {
  478. pimpl->handleWheel (peer, pos, Time (time), wheel);
  479. }
  480. void MouseInputSource::handleMagnifyGesture (ComponentPeer& peer, Point<float> pos, int64 time, float scaleFactor)
  481. {
  482. pimpl->handleMagnifyGesture (peer, pos, Time (time), scaleFactor);
  483. }
  484. const float MouseInputSource::invalidPressure = 0.0f;
  485. //==============================================================================
  486. struct MouseInputSource::SourceList : public Timer
  487. {
  488. SourceList()
  489. {
  490. addSource();
  491. }
  492. bool addSource();
  493. void addSource (int index, bool isMouse)
  494. {
  495. MouseInputSourceInternal* s = new MouseInputSourceInternal (index, isMouse);
  496. sources.add (s);
  497. sourceArray.add (MouseInputSource (s));
  498. }
  499. MouseInputSource* getMouseSource (int index) const noexcept
  500. {
  501. return isPositiveAndBelow (index, sourceArray.size()) ? &sourceArray.getReference (index)
  502. : nullptr;
  503. }
  504. MouseInputSource* getOrCreateMouseInputSource (int touchIndex)
  505. {
  506. jassert (touchIndex >= 0 && touchIndex < 100); // sanity-check on number of fingers
  507. for (;;)
  508. {
  509. if (MouseInputSource* mouse = getMouseSource (touchIndex))
  510. return mouse;
  511. if (! addSource())
  512. {
  513. jassertfalse; // not enough mouse sources!
  514. return nullptr;
  515. }
  516. }
  517. }
  518. int getNumDraggingMouseSources() const noexcept
  519. {
  520. int num = 0;
  521. for (int i = 0; i < sources.size(); ++i)
  522. if (sources.getUnchecked(i)->isDragging())
  523. ++num;
  524. return num;
  525. }
  526. MouseInputSource* getDraggingMouseSource (int index) const noexcept
  527. {
  528. int num = 0;
  529. for (int i = 0; i < sources.size(); ++i)
  530. {
  531. MouseInputSource* const mi = &(sourceArray.getReference(i));
  532. if (mi->isDragging())
  533. {
  534. if (index == num)
  535. return mi;
  536. ++num;
  537. }
  538. }
  539. return nullptr;
  540. }
  541. void beginDragAutoRepeat (const int interval)
  542. {
  543. if (interval > 0)
  544. {
  545. if (getTimerInterval() != interval)
  546. startTimer (interval);
  547. }
  548. else
  549. {
  550. stopTimer();
  551. }
  552. }
  553. void timerCallback() override
  554. {
  555. int numMiceDown = 0;
  556. for (int i = 0; i < sources.size(); ++i)
  557. {
  558. MouseInputSourceInternal* const mi = sources.getUnchecked(i);
  559. if (mi->isDragging())
  560. {
  561. mi->triggerFakeMove();
  562. ++numMiceDown;
  563. }
  564. }
  565. if (numMiceDown == 0)
  566. stopTimer();
  567. }
  568. OwnedArray<MouseInputSourceInternal> sources;
  569. Array<MouseInputSource> sourceArray;
  570. };