The JUCE cross-platform C++ framework, with DISTRHO/KXStudio specific changes
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.

1220 lines
38KB

  1. /*
  2. ==============================================================================
  3. This file is part of the JUCE 7 technical preview.
  4. Copyright (c) 2022 - Raw Material Software Limited
  5. You may use this code under the terms of the GPL v3
  6. (see www.gnu.org/licenses).
  7. For the technical preview this file cannot be licensed commercially.
  8. JUCE IS PROVIDED "AS IS" WITHOUT ANY WARRANTY, AND ALL WARRANTIES, WHETHER
  9. EXPRESSED OR IMPLIED, INCLUDING MERCHANTABILITY AND FITNESS FOR PURPOSE, ARE
  10. DISCLAIMED.
  11. ==============================================================================
  12. */
  13. namespace juce
  14. {
  15. template <typename RowComponentType>
  16. static AccessibilityActions getListRowAccessibilityActions (RowComponentType& rowComponent)
  17. {
  18. auto onFocus = [&rowComponent]
  19. {
  20. rowComponent.owner.scrollToEnsureRowIsOnscreen (rowComponent.row);
  21. rowComponent.owner.selectRow (rowComponent.row);
  22. };
  23. auto onPress = [&rowComponent, onFocus]
  24. {
  25. onFocus();
  26. rowComponent.owner.keyPressed (KeyPress (KeyPress::returnKey));
  27. };
  28. auto onToggle = [&rowComponent]
  29. {
  30. rowComponent.owner.flipRowSelection (rowComponent.row);
  31. };
  32. return AccessibilityActions().addAction (AccessibilityActionType::focus, std::move (onFocus))
  33. .addAction (AccessibilityActionType::press, std::move (onPress))
  34. .addAction (AccessibilityActionType::toggle, std::move (onToggle));
  35. }
  36. void ListBox::checkModelPtrIsValid() const
  37. {
  38. #if ! JUCE_DISABLE_ASSERTIONS
  39. // If this is hit, the model was destroyed while the ListBox was still using it.
  40. // You should ensure that the model remains alive for as long as the ListBox holds a pointer to it.
  41. // If this assertion is hit in the destructor of a ListBox instance, do one of the following:
  42. // - Adjust the order in which your destructors run, so that the ListBox destructor runs
  43. // before the destructor of your ListBoxModel, or
  44. // - Call ListBox::setModel (nullptr) before destroying your ListBoxModel.
  45. jassert ((model == nullptr) == (weakModelPtr.lock() == nullptr));
  46. #endif
  47. }
  48. class ListBox::RowComponent : public Component,
  49. public TooltipClient
  50. {
  51. public:
  52. RowComponent (ListBox& lb) : owner (lb) {}
  53. void paint (Graphics& g) override
  54. {
  55. if (auto* m = owner.getModel())
  56. m->paintListBoxItem (row, g, getWidth(), getHeight(), isSelected);
  57. }
  58. void update (const int newRow, const bool nowSelected)
  59. {
  60. const auto rowHasChanged = (row != newRow);
  61. const auto selectionHasChanged = (isSelected != nowSelected);
  62. if (rowHasChanged || selectionHasChanged)
  63. {
  64. repaint();
  65. if (rowHasChanged)
  66. row = newRow;
  67. if (selectionHasChanged)
  68. isSelected = nowSelected;
  69. }
  70. if (auto* m = owner.getModel())
  71. {
  72. setMouseCursor (m->getMouseCursorForRow (row));
  73. customComponent.reset (m->refreshComponentForRow (newRow, nowSelected, customComponent.release()));
  74. if (customComponent != nullptr)
  75. {
  76. addAndMakeVisible (customComponent.get());
  77. customComponent->setBounds (getLocalBounds());
  78. setFocusContainerType (FocusContainerType::focusContainer);
  79. }
  80. else
  81. {
  82. setFocusContainerType (FocusContainerType::none);
  83. }
  84. }
  85. }
  86. void performSelection (const MouseEvent& e, bool isMouseUp)
  87. {
  88. owner.selectRowsBasedOnModifierKeys (row, e.mods, isMouseUp);
  89. if (auto* m = owner.getModel())
  90. m->listBoxItemClicked (row, e);
  91. }
  92. void mouseDown (const MouseEvent& e) override
  93. {
  94. isDragging = false;
  95. isDraggingToScroll = false;
  96. selectRowOnMouseUp = false;
  97. if (isEnabled())
  98. {
  99. if (owner.selectOnMouseDown && ! isSelected && ! viewportWouldScrollOnEvent (owner.getViewport(), e.source))
  100. performSelection (e, false);
  101. else
  102. selectRowOnMouseUp = true;
  103. }
  104. }
  105. void mouseUp (const MouseEvent& e) override
  106. {
  107. if (isEnabled() && selectRowOnMouseUp && ! (isDragging || isDraggingToScroll))
  108. performSelection (e, true);
  109. }
  110. void mouseDoubleClick (const MouseEvent& e) override
  111. {
  112. if (isEnabled())
  113. if (auto* m = owner.getModel())
  114. m->listBoxItemDoubleClicked (row, e);
  115. }
  116. void mouseDrag (const MouseEvent& e) override
  117. {
  118. if (auto* m = owner.getModel())
  119. {
  120. if (isEnabled() && e.mouseWasDraggedSinceMouseDown() && ! isDragging)
  121. {
  122. SparseSet<int> rowsToDrag;
  123. if (owner.selectOnMouseDown || owner.isRowSelected (row))
  124. rowsToDrag = owner.getSelectedRows();
  125. else
  126. rowsToDrag.addRange (Range<int>::withStartAndLength (row, 1));
  127. if (rowsToDrag.size() > 0)
  128. {
  129. auto dragDescription = m->getDragSourceDescription (rowsToDrag);
  130. if (! (dragDescription.isVoid() || (dragDescription.isString() && dragDescription.toString().isEmpty())))
  131. {
  132. isDragging = true;
  133. owner.startDragAndDrop (e, rowsToDrag, dragDescription, true);
  134. }
  135. }
  136. }
  137. }
  138. if (! isDraggingToScroll)
  139. if (auto* vp = owner.getViewport())
  140. isDraggingToScroll = vp->isCurrentlyScrollingOnDrag();
  141. }
  142. void resized() override
  143. {
  144. if (customComponent != nullptr)
  145. customComponent->setBounds (getLocalBounds());
  146. }
  147. String getTooltip() override
  148. {
  149. if (auto* m = owner.getModel())
  150. return m->getTooltipForRow (row);
  151. return {};
  152. }
  153. //==============================================================================
  154. class RowAccessibilityHandler : public AccessibilityHandler
  155. {
  156. public:
  157. explicit RowAccessibilityHandler (RowComponent& rowComponentToWrap)
  158. : AccessibilityHandler (rowComponentToWrap,
  159. AccessibilityRole::listItem,
  160. getListRowAccessibilityActions (rowComponentToWrap),
  161. { std::make_unique<RowCellInterface> (*this) }),
  162. rowComponent (rowComponentToWrap)
  163. {
  164. }
  165. String getTitle() const override
  166. {
  167. if (auto* m = rowComponent.owner.getModel())
  168. return m->getNameForRow (rowComponent.row);
  169. return {};
  170. }
  171. String getHelp() const override { return rowComponent.getTooltip(); }
  172. AccessibleState getCurrentState() const override
  173. {
  174. if (auto* m = rowComponent.owner.getModel())
  175. if (rowComponent.row >= m->getNumRows())
  176. return AccessibleState().withIgnored();
  177. auto state = AccessibilityHandler::getCurrentState().withAccessibleOffscreen();
  178. if (rowComponent.owner.multipleSelection)
  179. state = state.withMultiSelectable();
  180. else
  181. state = state.withSelectable();
  182. if (rowComponent.isSelected)
  183. state = state.withSelected();
  184. return state;
  185. }
  186. private:
  187. class RowCellInterface : public AccessibilityCellInterface
  188. {
  189. public:
  190. explicit RowCellInterface (RowAccessibilityHandler& h) : handler (h) {}
  191. int getColumnIndex() const override { return 0; }
  192. int getColumnSpan() const override { return 1; }
  193. int getRowIndex() const override
  194. {
  195. const auto index = handler.rowComponent.row;
  196. if (handler.rowComponent.owner.hasAccessibleHeaderComponent())
  197. return index + 1;
  198. return index;
  199. }
  200. int getRowSpan() const override { return 1; }
  201. int getDisclosureLevel() const override { return 0; }
  202. const AccessibilityHandler* getTableHandler() const override
  203. {
  204. return handler.rowComponent.owner.getAccessibilityHandler();
  205. }
  206. private:
  207. RowAccessibilityHandler& handler;
  208. };
  209. RowComponent& rowComponent;
  210. };
  211. std::unique_ptr<AccessibilityHandler> createAccessibilityHandler() override
  212. {
  213. return std::make_unique<RowAccessibilityHandler> (*this);
  214. }
  215. //==============================================================================
  216. ListBox& owner;
  217. std::unique_ptr<Component> customComponent;
  218. int row = -1;
  219. bool isSelected = false, isDragging = false, isDraggingToScroll = false, selectRowOnMouseUp = false;
  220. JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (RowComponent)
  221. };
  222. //==============================================================================
  223. class ListBox::ListViewport : public Viewport,
  224. private Timer
  225. {
  226. public:
  227. ListViewport (ListBox& lb) : owner (lb)
  228. {
  229. setWantsKeyboardFocus (false);
  230. auto content = std::make_unique<Component>();
  231. content->setWantsKeyboardFocus (false);
  232. setViewedComponent (content.release());
  233. }
  234. RowComponent* getComponentForRow (int row) const noexcept
  235. {
  236. if (isPositiveAndBelow (row, rows.size()))
  237. return rows[row];
  238. return nullptr;
  239. }
  240. RowComponent* getComponentForRowWrapped (int row) const noexcept
  241. {
  242. return rows[row % jmax (1, rows.size())];
  243. }
  244. RowComponent* getComponentForRowIfOnscreen (int row) const noexcept
  245. {
  246. return (row >= firstIndex && row < firstIndex + rows.size())
  247. ? getComponentForRowWrapped (row) : nullptr;
  248. }
  249. int getRowNumberOfComponent (Component* const rowComponent) const noexcept
  250. {
  251. const int index = getViewedComponent()->getIndexOfChildComponent (rowComponent);
  252. const int num = rows.size();
  253. for (int i = num; --i >= 0;)
  254. if (((firstIndex + i) % jmax (1, num)) == index)
  255. return firstIndex + i;
  256. return -1;
  257. }
  258. void visibleAreaChanged (const Rectangle<int>&) override
  259. {
  260. updateVisibleArea (true);
  261. if (auto* m = owner.getModel())
  262. m->listWasScrolled();
  263. startTimer (50);
  264. }
  265. void updateVisibleArea (const bool makeSureItUpdatesContent)
  266. {
  267. hasUpdated = false;
  268. auto& content = *getViewedComponent();
  269. auto newX = content.getX();
  270. auto newY = content.getY();
  271. auto newW = jmax (owner.minimumRowWidth, getMaximumVisibleWidth());
  272. auto newH = owner.totalItems * owner.getRowHeight();
  273. if (newY + newH < getMaximumVisibleHeight() && newH > getMaximumVisibleHeight())
  274. newY = getMaximumVisibleHeight() - newH;
  275. content.setBounds (newX, newY, newW, newH);
  276. if (makeSureItUpdatesContent && ! hasUpdated)
  277. updateContents();
  278. }
  279. void updateContents()
  280. {
  281. hasUpdated = true;
  282. auto rowH = owner.getRowHeight();
  283. auto& content = *getViewedComponent();
  284. if (rowH > 0)
  285. {
  286. auto y = getViewPositionY();
  287. auto w = content.getWidth();
  288. const int numNeeded = 4 + getMaximumVisibleHeight() / rowH;
  289. rows.removeRange (numNeeded, rows.size());
  290. while (numNeeded > rows.size())
  291. {
  292. auto* newRow = rows.add (new RowComponent (owner));
  293. content.addAndMakeVisible (newRow);
  294. }
  295. firstIndex = y / rowH;
  296. firstWholeIndex = (y + rowH - 1) / rowH;
  297. lastWholeIndex = (y + getMaximumVisibleHeight() - 1) / rowH;
  298. auto startIndex = jmax (0, firstIndex - 1);
  299. for (int i = 0; i < numNeeded; ++i)
  300. {
  301. const int row = i + startIndex;
  302. if (auto* rowComp = getComponentForRowWrapped (row))
  303. {
  304. rowComp->setBounds (0, row * rowH, w, rowH);
  305. rowComp->update (row, owner.isRowSelected (row));
  306. }
  307. }
  308. }
  309. if (owner.headerComponent != nullptr)
  310. owner.headerComponent->setBounds (owner.outlineThickness + content.getX(),
  311. owner.outlineThickness,
  312. jmax (owner.getWidth() - owner.outlineThickness * 2,
  313. content.getWidth()),
  314. owner.headerComponent->getHeight());
  315. }
  316. void selectRow (const int row, const int rowH, const bool dontScroll,
  317. const int lastSelectedRow, const int totalRows, const bool isMouseClick)
  318. {
  319. hasUpdated = false;
  320. if (row < firstWholeIndex && ! dontScroll)
  321. {
  322. setViewPosition (getViewPositionX(), row * rowH);
  323. }
  324. else if (row >= lastWholeIndex && ! dontScroll)
  325. {
  326. const int rowsOnScreen = lastWholeIndex - firstWholeIndex;
  327. if (row >= lastSelectedRow + rowsOnScreen
  328. && rowsOnScreen < totalRows - 1
  329. && ! isMouseClick)
  330. {
  331. setViewPosition (getViewPositionX(),
  332. jlimit (0, jmax (0, totalRows - rowsOnScreen), row) * rowH);
  333. }
  334. else
  335. {
  336. setViewPosition (getViewPositionX(),
  337. jmax (0, (row + 1) * rowH - getMaximumVisibleHeight()));
  338. }
  339. }
  340. if (! hasUpdated)
  341. updateContents();
  342. }
  343. void scrollToEnsureRowIsOnscreen (const int row, const int rowH)
  344. {
  345. if (row < firstWholeIndex)
  346. {
  347. setViewPosition (getViewPositionX(), row * rowH);
  348. }
  349. else if (row >= lastWholeIndex)
  350. {
  351. setViewPosition (getViewPositionX(),
  352. jmax (0, (row + 1) * rowH - getMaximumVisibleHeight()));
  353. }
  354. }
  355. void paint (Graphics& g) override
  356. {
  357. if (isOpaque())
  358. g.fillAll (owner.findColour (ListBox::backgroundColourId));
  359. }
  360. bool keyPressed (const KeyPress& key) override
  361. {
  362. if (Viewport::respondsToKey (key))
  363. {
  364. const int allowableMods = owner.multipleSelection ? ModifierKeys::shiftModifier : 0;
  365. if ((key.getModifiers().getRawFlags() & ~allowableMods) == 0)
  366. {
  367. // we want to avoid these keypresses going to the viewport, and instead allow
  368. // them to pass up to our listbox..
  369. return false;
  370. }
  371. }
  372. return Viewport::keyPressed (key);
  373. }
  374. private:
  375. std::unique_ptr<AccessibilityHandler> createAccessibilityHandler() override
  376. {
  377. return createIgnoredAccessibilityHandler (*this);
  378. }
  379. void timerCallback() override
  380. {
  381. stopTimer();
  382. if (auto* handler = owner.getAccessibilityHandler())
  383. handler->notifyAccessibilityEvent (AccessibilityEvent::structureChanged);
  384. }
  385. ListBox& owner;
  386. OwnedArray<RowComponent> rows;
  387. int firstIndex = 0, firstWholeIndex = 0, lastWholeIndex = 0;
  388. bool hasUpdated = false;
  389. JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (ListViewport)
  390. };
  391. //==============================================================================
  392. struct ListBoxMouseMoveSelector : public MouseListener
  393. {
  394. ListBoxMouseMoveSelector (ListBox& lb) : owner (lb)
  395. {
  396. owner.addMouseListener (this, true);
  397. }
  398. ~ListBoxMouseMoveSelector() override
  399. {
  400. owner.removeMouseListener (this);
  401. }
  402. void mouseMove (const MouseEvent& e) override
  403. {
  404. auto pos = e.getEventRelativeTo (&owner).position.toInt();
  405. owner.selectRow (owner.getRowContainingPosition (pos.x, pos.y), true);
  406. }
  407. void mouseExit (const MouseEvent& e) override
  408. {
  409. mouseMove (e);
  410. }
  411. ListBox& owner;
  412. JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (ListBoxMouseMoveSelector)
  413. };
  414. //==============================================================================
  415. ListBox::ListBox (const String& name, ListBoxModel* const m)
  416. : Component (name)
  417. {
  418. viewport.reset (new ListViewport (*this));
  419. addAndMakeVisible (viewport.get());
  420. setWantsKeyboardFocus (true);
  421. setFocusContainerType (FocusContainerType::focusContainer);
  422. colourChanged();
  423. setModel (m);
  424. }
  425. ListBox::~ListBox()
  426. {
  427. headerComponent.reset();
  428. viewport.reset();
  429. }
  430. void ListBox::setModel (ListBoxModel* const newModel)
  431. {
  432. if (model != newModel)
  433. {
  434. model = newModel;
  435. #if ! JUCE_DISABLE_ASSERTIONS
  436. weakModelPtr = model != nullptr ? model->sharedState : nullptr;
  437. #endif
  438. repaint();
  439. updateContent();
  440. }
  441. }
  442. void ListBox::setMultipleSelectionEnabled (bool b) noexcept { multipleSelection = b; }
  443. void ListBox::setClickingTogglesRowSelection (bool b) noexcept { alwaysFlipSelection = b; }
  444. void ListBox::setRowSelectedOnMouseDown (bool b) noexcept { selectOnMouseDown = b; }
  445. void ListBox::setMouseMoveSelectsRows (bool b)
  446. {
  447. if (b)
  448. {
  449. if (mouseMoveSelector == nullptr)
  450. mouseMoveSelector.reset (new ListBoxMouseMoveSelector (*this));
  451. }
  452. else
  453. {
  454. mouseMoveSelector.reset();
  455. }
  456. }
  457. //==============================================================================
  458. void ListBox::paint (Graphics& g)
  459. {
  460. if (! hasDoneInitialUpdate)
  461. updateContent();
  462. g.fillAll (findColour (backgroundColourId));
  463. }
  464. void ListBox::paintOverChildren (Graphics& g)
  465. {
  466. if (outlineThickness > 0)
  467. {
  468. g.setColour (findColour (outlineColourId));
  469. g.drawRect (getLocalBounds(), outlineThickness);
  470. }
  471. }
  472. void ListBox::resized()
  473. {
  474. viewport->setBoundsInset (BorderSize<int> (outlineThickness + (headerComponent != nullptr ? headerComponent->getHeight() : 0),
  475. outlineThickness, outlineThickness, outlineThickness));
  476. viewport->setSingleStepSizes (20, getRowHeight());
  477. viewport->updateVisibleArea (false);
  478. }
  479. void ListBox::visibilityChanged()
  480. {
  481. viewport->updateVisibleArea (true);
  482. }
  483. Viewport* ListBox::getViewport() const noexcept
  484. {
  485. return viewport.get();
  486. }
  487. //==============================================================================
  488. void ListBox::updateContent()
  489. {
  490. checkModelPtrIsValid();
  491. hasDoneInitialUpdate = true;
  492. totalItems = (model != nullptr) ? model->getNumRows() : 0;
  493. bool selectionChanged = false;
  494. if (selected.size() > 0 && selected [selected.size() - 1] >= totalItems)
  495. {
  496. selected.removeRange ({ totalItems, std::numeric_limits<int>::max() });
  497. lastRowSelected = getSelectedRow (0);
  498. selectionChanged = true;
  499. }
  500. viewport->updateVisibleArea (isVisible());
  501. viewport->resized();
  502. if (selectionChanged)
  503. {
  504. if (model != nullptr)
  505. model->selectedRowsChanged (lastRowSelected);
  506. if (auto* handler = getAccessibilityHandler())
  507. handler->notifyAccessibilityEvent (AccessibilityEvent::rowSelectionChanged);
  508. }
  509. }
  510. //==============================================================================
  511. void ListBox::selectRow (int row, bool dontScroll, bool deselectOthersFirst)
  512. {
  513. selectRowInternal (row, dontScroll, deselectOthersFirst, false);
  514. }
  515. void ListBox::selectRowInternal (const int row,
  516. bool dontScroll,
  517. bool deselectOthersFirst,
  518. bool isMouseClick)
  519. {
  520. checkModelPtrIsValid();
  521. if (! multipleSelection)
  522. deselectOthersFirst = true;
  523. if ((! isRowSelected (row))
  524. || (deselectOthersFirst && getNumSelectedRows() > 1))
  525. {
  526. if (isPositiveAndBelow (row, totalItems))
  527. {
  528. if (deselectOthersFirst)
  529. selected.clear();
  530. selected.addRange ({ row, row + 1 });
  531. if (getHeight() == 0 || getWidth() == 0)
  532. dontScroll = true;
  533. viewport->selectRow (row, getRowHeight(), dontScroll,
  534. lastRowSelected, totalItems, isMouseClick);
  535. lastRowSelected = row;
  536. model->selectedRowsChanged (row);
  537. if (auto* handler = getAccessibilityHandler())
  538. handler->notifyAccessibilityEvent (AccessibilityEvent::rowSelectionChanged);
  539. }
  540. else
  541. {
  542. if (deselectOthersFirst)
  543. deselectAllRows();
  544. }
  545. }
  546. }
  547. void ListBox::deselectRow (const int row)
  548. {
  549. checkModelPtrIsValid();
  550. if (selected.contains (row))
  551. {
  552. selected.removeRange ({ row, row + 1 });
  553. if (row == lastRowSelected)
  554. lastRowSelected = getSelectedRow (0);
  555. viewport->updateContents();
  556. model->selectedRowsChanged (lastRowSelected);
  557. if (auto* handler = getAccessibilityHandler())
  558. handler->notifyAccessibilityEvent (AccessibilityEvent::rowSelectionChanged);
  559. }
  560. }
  561. void ListBox::setSelectedRows (const SparseSet<int>& setOfRowsToBeSelected,
  562. const NotificationType sendNotificationEventToModel)
  563. {
  564. checkModelPtrIsValid();
  565. selected = setOfRowsToBeSelected;
  566. selected.removeRange ({ totalItems, std::numeric_limits<int>::max() });
  567. if (! isRowSelected (lastRowSelected))
  568. lastRowSelected = getSelectedRow (0);
  569. viewport->updateContents();
  570. if (model != nullptr && sendNotificationEventToModel == sendNotification)
  571. model->selectedRowsChanged (lastRowSelected);
  572. if (auto* handler = getAccessibilityHandler())
  573. handler->notifyAccessibilityEvent (AccessibilityEvent::rowSelectionChanged);
  574. }
  575. SparseSet<int> ListBox::getSelectedRows() const
  576. {
  577. return selected;
  578. }
  579. void ListBox::selectRangeOfRows (int firstRow, int lastRow, bool dontScrollToShowThisRange)
  580. {
  581. if (multipleSelection && (firstRow != lastRow))
  582. {
  583. const int numRows = totalItems - 1;
  584. firstRow = jlimit (0, jmax (0, numRows), firstRow);
  585. lastRow = jlimit (0, jmax (0, numRows), lastRow);
  586. selected.addRange ({ jmin (firstRow, lastRow),
  587. jmax (firstRow, lastRow) + 1 });
  588. selected.removeRange ({ lastRow, lastRow + 1 });
  589. }
  590. selectRowInternal (lastRow, dontScrollToShowThisRange, false, true);
  591. }
  592. void ListBox::flipRowSelection (const int row)
  593. {
  594. if (isRowSelected (row))
  595. deselectRow (row);
  596. else
  597. selectRowInternal (row, false, false, true);
  598. }
  599. void ListBox::deselectAllRows()
  600. {
  601. checkModelPtrIsValid();
  602. if (! selected.isEmpty())
  603. {
  604. selected.clear();
  605. lastRowSelected = -1;
  606. viewport->updateContents();
  607. if (model != nullptr)
  608. model->selectedRowsChanged (lastRowSelected);
  609. if (auto* handler = getAccessibilityHandler())
  610. handler->notifyAccessibilityEvent (AccessibilityEvent::rowSelectionChanged);
  611. }
  612. }
  613. void ListBox::selectRowsBasedOnModifierKeys (const int row,
  614. ModifierKeys mods,
  615. const bool isMouseUpEvent)
  616. {
  617. if (multipleSelection && (mods.isCommandDown() || alwaysFlipSelection))
  618. {
  619. flipRowSelection (row);
  620. }
  621. else if (multipleSelection && mods.isShiftDown() && lastRowSelected >= 0)
  622. {
  623. selectRangeOfRows (lastRowSelected, row);
  624. }
  625. else if ((! mods.isPopupMenu()) || ! isRowSelected (row))
  626. {
  627. selectRowInternal (row, false, ! (multipleSelection && (! isMouseUpEvent) && isRowSelected (row)), true);
  628. }
  629. }
  630. int ListBox::getNumSelectedRows() const
  631. {
  632. return selected.size();
  633. }
  634. int ListBox::getSelectedRow (const int index) const
  635. {
  636. return (isPositiveAndBelow (index, selected.size()))
  637. ? selected [index] : -1;
  638. }
  639. bool ListBox::isRowSelected (const int row) const
  640. {
  641. return selected.contains (row);
  642. }
  643. int ListBox::getLastRowSelected() const
  644. {
  645. return isRowSelected (lastRowSelected) ? lastRowSelected : -1;
  646. }
  647. //==============================================================================
  648. int ListBox::getRowContainingPosition (const int x, const int y) const noexcept
  649. {
  650. if (isPositiveAndBelow (x, getWidth()))
  651. {
  652. const int row = (viewport->getViewPositionY() + y - viewport->getY()) / rowHeight;
  653. if (isPositiveAndBelow (row, totalItems))
  654. return row;
  655. }
  656. return -1;
  657. }
  658. int ListBox::getInsertionIndexForPosition (const int x, const int y) const noexcept
  659. {
  660. if (isPositiveAndBelow (x, getWidth()))
  661. return jlimit (0, totalItems, (viewport->getViewPositionY() + y + rowHeight / 2 - viewport->getY()) / rowHeight);
  662. return -1;
  663. }
  664. Component* ListBox::getComponentForRowNumber (const int row) const noexcept
  665. {
  666. if (auto* listRowComp = viewport->getComponentForRowIfOnscreen (row))
  667. return listRowComp->customComponent.get();
  668. return nullptr;
  669. }
  670. int ListBox::getRowNumberOfComponent (Component* const rowComponent) const noexcept
  671. {
  672. return viewport->getRowNumberOfComponent (rowComponent);
  673. }
  674. Rectangle<int> ListBox::getRowPosition (int rowNumber, bool relativeToComponentTopLeft) const noexcept
  675. {
  676. auto y = viewport->getY() + rowHeight * rowNumber;
  677. if (relativeToComponentTopLeft)
  678. y -= viewport->getViewPositionY();
  679. return { viewport->getX(), y,
  680. viewport->getViewedComponent()->getWidth(), rowHeight };
  681. }
  682. void ListBox::setVerticalPosition (const double proportion)
  683. {
  684. auto offscreen = viewport->getViewedComponent()->getHeight() - viewport->getHeight();
  685. viewport->setViewPosition (viewport->getViewPositionX(),
  686. jmax (0, roundToInt (proportion * offscreen)));
  687. }
  688. double ListBox::getVerticalPosition() const
  689. {
  690. auto offscreen = viewport->getViewedComponent()->getHeight() - viewport->getHeight();
  691. return offscreen > 0 ? viewport->getViewPositionY() / (double) offscreen
  692. : 0;
  693. }
  694. int ListBox::getVisibleRowWidth() const noexcept
  695. {
  696. return viewport->getViewWidth();
  697. }
  698. void ListBox::scrollToEnsureRowIsOnscreen (const int row)
  699. {
  700. viewport->scrollToEnsureRowIsOnscreen (row, getRowHeight());
  701. }
  702. //==============================================================================
  703. bool ListBox::keyPressed (const KeyPress& key)
  704. {
  705. checkModelPtrIsValid();
  706. const int numVisibleRows = viewport->getHeight() / getRowHeight();
  707. const bool multiple = multipleSelection
  708. && lastRowSelected >= 0
  709. && key.getModifiers().isShiftDown();
  710. if (key.isKeyCode (KeyPress::upKey))
  711. {
  712. if (multiple)
  713. selectRangeOfRows (lastRowSelected, lastRowSelected - 1);
  714. else
  715. selectRow (jmax (0, lastRowSelected - 1));
  716. }
  717. else if (key.isKeyCode (KeyPress::downKey))
  718. {
  719. if (multiple)
  720. selectRangeOfRows (lastRowSelected, lastRowSelected + 1);
  721. else
  722. selectRow (jmin (totalItems - 1, jmax (0, lastRowSelected + 1)));
  723. }
  724. else if (key.isKeyCode (KeyPress::pageUpKey))
  725. {
  726. if (multiple)
  727. selectRangeOfRows (lastRowSelected, lastRowSelected - numVisibleRows);
  728. else
  729. selectRow (jmax (0, jmax (0, lastRowSelected) - numVisibleRows));
  730. }
  731. else if (key.isKeyCode (KeyPress::pageDownKey))
  732. {
  733. if (multiple)
  734. selectRangeOfRows (lastRowSelected, lastRowSelected + numVisibleRows);
  735. else
  736. selectRow (jmin (totalItems - 1, jmax (0, lastRowSelected) + numVisibleRows));
  737. }
  738. else if (key.isKeyCode (KeyPress::homeKey))
  739. {
  740. if (multiple)
  741. selectRangeOfRows (lastRowSelected, 0);
  742. else
  743. selectRow (0);
  744. }
  745. else if (key.isKeyCode (KeyPress::endKey))
  746. {
  747. if (multiple)
  748. selectRangeOfRows (lastRowSelected, totalItems - 1);
  749. else
  750. selectRow (totalItems - 1);
  751. }
  752. else if (key.isKeyCode (KeyPress::returnKey) && isRowSelected (lastRowSelected))
  753. {
  754. if (model != nullptr)
  755. model->returnKeyPressed (lastRowSelected);
  756. }
  757. else if ((key.isKeyCode (KeyPress::deleteKey) || key.isKeyCode (KeyPress::backspaceKey))
  758. && isRowSelected (lastRowSelected))
  759. {
  760. if (model != nullptr)
  761. model->deleteKeyPressed (lastRowSelected);
  762. }
  763. else if (multipleSelection && key == KeyPress ('a', ModifierKeys::commandModifier, 0))
  764. {
  765. selectRangeOfRows (0, std::numeric_limits<int>::max());
  766. }
  767. else
  768. {
  769. return false;
  770. }
  771. return true;
  772. }
  773. bool ListBox::keyStateChanged (const bool isKeyDown)
  774. {
  775. return isKeyDown
  776. && (KeyPress::isKeyCurrentlyDown (KeyPress::upKey)
  777. || KeyPress::isKeyCurrentlyDown (KeyPress::pageUpKey)
  778. || KeyPress::isKeyCurrentlyDown (KeyPress::downKey)
  779. || KeyPress::isKeyCurrentlyDown (KeyPress::pageDownKey)
  780. || KeyPress::isKeyCurrentlyDown (KeyPress::homeKey)
  781. || KeyPress::isKeyCurrentlyDown (KeyPress::endKey)
  782. || KeyPress::isKeyCurrentlyDown (KeyPress::returnKey));
  783. }
  784. void ListBox::mouseWheelMove (const MouseEvent& e, const MouseWheelDetails& wheel)
  785. {
  786. bool eventWasUsed = false;
  787. if (wheel.deltaX != 0.0f && getHorizontalScrollBar().isVisible())
  788. {
  789. eventWasUsed = true;
  790. getHorizontalScrollBar().mouseWheelMove (e, wheel);
  791. }
  792. if (wheel.deltaY != 0.0f && getVerticalScrollBar().isVisible())
  793. {
  794. eventWasUsed = true;
  795. getVerticalScrollBar().mouseWheelMove (e, wheel);
  796. }
  797. if (! eventWasUsed)
  798. Component::mouseWheelMove (e, wheel);
  799. }
  800. void ListBox::mouseUp (const MouseEvent& e)
  801. {
  802. checkModelPtrIsValid();
  803. if (e.mouseWasClicked() && model != nullptr)
  804. model->backgroundClicked (e);
  805. }
  806. //==============================================================================
  807. void ListBox::setRowHeight (const int newHeight)
  808. {
  809. rowHeight = jmax (1, newHeight);
  810. viewport->setSingleStepSizes (20, rowHeight);
  811. updateContent();
  812. }
  813. int ListBox::getNumRowsOnScreen() const noexcept
  814. {
  815. return viewport->getMaximumVisibleHeight() / rowHeight;
  816. }
  817. void ListBox::setMinimumContentWidth (const int newMinimumWidth)
  818. {
  819. minimumRowWidth = newMinimumWidth;
  820. updateContent();
  821. }
  822. int ListBox::getVisibleContentWidth() const noexcept { return viewport->getMaximumVisibleWidth(); }
  823. ScrollBar& ListBox::getVerticalScrollBar() const noexcept { return viewport->getVerticalScrollBar(); }
  824. ScrollBar& ListBox::getHorizontalScrollBar() const noexcept { return viewport->getHorizontalScrollBar(); }
  825. void ListBox::colourChanged()
  826. {
  827. setOpaque (findColour (backgroundColourId).isOpaque());
  828. viewport->setOpaque (isOpaque());
  829. repaint();
  830. }
  831. void ListBox::parentHierarchyChanged()
  832. {
  833. colourChanged();
  834. }
  835. void ListBox::setOutlineThickness (int newThickness)
  836. {
  837. outlineThickness = newThickness;
  838. resized();
  839. }
  840. void ListBox::setHeaderComponent (std::unique_ptr<Component> newHeaderComponent)
  841. {
  842. headerComponent = std::move (newHeaderComponent);
  843. addAndMakeVisible (headerComponent.get());
  844. ListBox::resized();
  845. invalidateAccessibilityHandler();
  846. }
  847. bool ListBox::hasAccessibleHeaderComponent() const
  848. {
  849. return headerComponent != nullptr
  850. && headerComponent->getAccessibilityHandler() != nullptr;
  851. }
  852. void ListBox::repaintRow (const int rowNumber) noexcept
  853. {
  854. repaint (getRowPosition (rowNumber, true));
  855. }
  856. ScaledImage ListBox::createSnapshotOfRows (const SparseSet<int>& rows, int& imageX, int& imageY)
  857. {
  858. Rectangle<int> imageArea;
  859. auto firstRow = getRowContainingPosition (0, viewport->getY());
  860. for (int i = getNumRowsOnScreen() + 2; --i >= 0;)
  861. {
  862. if (rows.contains (firstRow + i))
  863. {
  864. if (auto* rowComp = viewport->getComponentForRowIfOnscreen (firstRow + i))
  865. {
  866. auto pos = getLocalPoint (rowComp, Point<int>());
  867. imageArea = imageArea.getUnion ({ pos.x, pos.y, rowComp->getWidth(), rowComp->getHeight() });
  868. }
  869. }
  870. }
  871. imageArea = imageArea.getIntersection (getLocalBounds());
  872. imageX = imageArea.getX();
  873. imageY = imageArea.getY();
  874. const auto additionalScale = 2.0f;
  875. const auto listScale = Component::getApproximateScaleFactorForComponent (this) * additionalScale;
  876. Image snapshot (Image::ARGB,
  877. roundToInt ((float) imageArea.getWidth() * listScale),
  878. roundToInt ((float) imageArea.getHeight() * listScale),
  879. true);
  880. for (int i = getNumRowsOnScreen() + 2; --i >= 0;)
  881. {
  882. if (rows.contains (firstRow + i))
  883. {
  884. if (auto* rowComp = viewport->getComponentForRowIfOnscreen (firstRow + i))
  885. {
  886. Graphics g (snapshot);
  887. g.setOrigin ((getLocalPoint (rowComp, Point<int>()) - imageArea.getPosition()) * additionalScale);
  888. const auto rowScale = Component::getApproximateScaleFactorForComponent (rowComp) * additionalScale;
  889. if (g.reduceClipRegion (rowComp->getLocalBounds() * rowScale))
  890. {
  891. g.beginTransparencyLayer (0.6f);
  892. g.addTransform (AffineTransform::scale (rowScale));
  893. rowComp->paintEntireComponent (g, false);
  894. g.endTransparencyLayer();
  895. }
  896. }
  897. }
  898. }
  899. return { snapshot, additionalScale };
  900. }
  901. void ListBox::startDragAndDrop (const MouseEvent& e, const SparseSet<int>& rowsToDrag, const var& dragDescription, bool allowDraggingToOtherWindows)
  902. {
  903. if (auto* dragContainer = DragAndDropContainer::findParentDragContainerFor (this))
  904. {
  905. int x, y;
  906. auto dragImage = createSnapshotOfRows (rowsToDrag, x, y);
  907. auto p = Point<int> (x, y) - e.getEventRelativeTo (this).position.toInt();
  908. dragContainer->startDragging (dragDescription, this, dragImage, allowDraggingToOtherWindows, &p, &e.source);
  909. }
  910. else
  911. {
  912. // to be able to do a drag-and-drop operation, the listbox needs to
  913. // be inside a component which is also a DragAndDropContainer.
  914. jassertfalse;
  915. }
  916. }
  917. std::unique_ptr<AccessibilityHandler> ListBox::createAccessibilityHandler()
  918. {
  919. class TableInterface : public AccessibilityTableInterface
  920. {
  921. public:
  922. explicit TableInterface (ListBox& listBoxToWrap)
  923. : listBox (listBoxToWrap)
  924. {
  925. }
  926. int getNumRows() const override
  927. {
  928. listBox.checkModelPtrIsValid();
  929. if (listBox.model == nullptr)
  930. return 0;
  931. const auto numRows = listBox.model->getNumRows();
  932. if (listBox.hasAccessibleHeaderComponent())
  933. return numRows + 1;
  934. return numRows;
  935. }
  936. int getNumColumns() const override
  937. {
  938. return 1;
  939. }
  940. const AccessibilityHandler* getCellHandler (int row, int) const override
  941. {
  942. if (auto* headerHandler = getHeaderHandler())
  943. {
  944. if (row == 0)
  945. return headerHandler;
  946. --row;
  947. }
  948. if (auto* rowComponent = listBox.viewport->getComponentForRow (row))
  949. return rowComponent->getAccessibilityHandler();
  950. return nullptr;
  951. }
  952. private:
  953. const AccessibilityHandler* getHeaderHandler() const
  954. {
  955. if (listBox.hasAccessibleHeaderComponent())
  956. return listBox.headerComponent->getAccessibilityHandler();
  957. return nullptr;
  958. }
  959. ListBox& listBox;
  960. JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (TableInterface)
  961. };
  962. return std::make_unique<AccessibilityHandler> (*this,
  963. AccessibilityRole::list,
  964. AccessibilityActions{},
  965. AccessibilityHandler::Interfaces { std::make_unique<TableInterface> (*this) });
  966. }
  967. //==============================================================================
  968. Component* ListBoxModel::refreshComponentForRow (int, bool, Component* existingComponentToUpdate)
  969. {
  970. ignoreUnused (existingComponentToUpdate);
  971. jassert (existingComponentToUpdate == nullptr); // indicates a failure in the code that recycles the components
  972. return nullptr;
  973. }
  974. String ListBoxModel::getNameForRow (int rowNumber) { return "Row " + String (rowNumber + 1); }
  975. void ListBoxModel::listBoxItemClicked (int, const MouseEvent&) {}
  976. void ListBoxModel::listBoxItemDoubleClicked (int, const MouseEvent&) {}
  977. void ListBoxModel::backgroundClicked (const MouseEvent&) {}
  978. void ListBoxModel::selectedRowsChanged (int) {}
  979. void ListBoxModel::deleteKeyPressed (int) {}
  980. void ListBoxModel::returnKeyPressed (int) {}
  981. void ListBoxModel::listWasScrolled() {}
  982. var ListBoxModel::getDragSourceDescription (const SparseSet<int>&) { return {}; }
  983. String ListBoxModel::getTooltipForRow (int) { return {}; }
  984. MouseCursor ListBoxModel::getMouseCursorForRow (int) { return MouseCursor::NormalCursor; }
  985. } // namespace juce