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.

297 lines
9.0KB

  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. #include "../JuceDemoHeader.h"
  18. //==============================================================================
  19. class ValueTreeItem : public TreeViewItem,
  20. private ValueTree::Listener
  21. {
  22. public:
  23. ValueTreeItem (const ValueTree& v, UndoManager& um)
  24. : tree (v), undoManager (um)
  25. {
  26. tree.addListener (this);
  27. }
  28. String getUniqueName() const override
  29. {
  30. return tree["name"].toString();
  31. }
  32. bool mightContainSubItems() override
  33. {
  34. return tree.getNumChildren() > 0;
  35. }
  36. void paintItem (Graphics& g, int width, int height) override
  37. {
  38. g.setColour (Colours::black);
  39. g.setFont (15.0f);
  40. g.drawText (tree["name"].toString(),
  41. 4, 0, width - 4, height,
  42. Justification::centredLeft, true);
  43. }
  44. void itemOpennessChanged (bool isNowOpen) override
  45. {
  46. if (isNowOpen && getNumSubItems() == 0)
  47. refreshSubItems();
  48. else
  49. clearSubItems();
  50. }
  51. var getDragSourceDescription() override
  52. {
  53. return "Drag Demo";
  54. }
  55. bool isInterestedInDragSource (const DragAndDropTarget::SourceDetails& dragSourceDetails) override
  56. {
  57. return dragSourceDetails.description == "Drag Demo";
  58. }
  59. void itemDropped (const DragAndDropTarget::SourceDetails&, int insertIndex) override
  60. {
  61. OwnedArray<ValueTree> selectedTrees;
  62. getSelectedTreeViewItems (*getOwnerView(), selectedTrees);
  63. moveItems (*getOwnerView(), selectedTrees, tree, insertIndex, undoManager);
  64. }
  65. static void moveItems (TreeView& treeView, const OwnedArray<ValueTree>& items,
  66. ValueTree newParent, int insertIndex, UndoManager& undoManager)
  67. {
  68. if (items.size() > 0)
  69. {
  70. ScopedPointer<XmlElement> oldOpenness (treeView.getOpennessState (false));
  71. for (int i = items.size(); --i >= 0;)
  72. {
  73. ValueTree& v = *items.getUnchecked(i);
  74. if (v.getParent().isValid() && newParent != v && ! newParent.isAChildOf (v))
  75. {
  76. if (v.getParent() == newParent && newParent.indexOf(v) < insertIndex)
  77. --insertIndex;
  78. v.getParent().removeChild (v, &undoManager);
  79. newParent.addChild (v, insertIndex, &undoManager);
  80. }
  81. }
  82. if (oldOpenness != nullptr)
  83. treeView.restoreOpennessState (*oldOpenness, false);
  84. }
  85. }
  86. static void getSelectedTreeViewItems (TreeView& treeView, OwnedArray<ValueTree>& items)
  87. {
  88. const int numSelected = treeView.getNumSelectedItems();
  89. for (int i = 0; i < numSelected; ++i)
  90. if (const ValueTreeItem* vti = dynamic_cast<ValueTreeItem*> (treeView.getSelectedItem (i)))
  91. items.add (new ValueTree (vti->tree));
  92. }
  93. private:
  94. ValueTree tree;
  95. UndoManager& undoManager;
  96. void refreshSubItems()
  97. {
  98. clearSubItems();
  99. for (int i = 0; i < tree.getNumChildren(); ++i)
  100. addSubItem (new ValueTreeItem (tree.getChild (i), undoManager));
  101. }
  102. void valueTreePropertyChanged (ValueTree&, const Identifier&) override
  103. {
  104. repaintItem();
  105. }
  106. void valueTreeChildAdded (ValueTree& parentTree, ValueTree&) override { treeChildrenChanged (parentTree); }
  107. void valueTreeChildRemoved (ValueTree& parentTree, ValueTree&, int) override { treeChildrenChanged (parentTree); }
  108. void valueTreeChildOrderChanged (ValueTree& parentTree, int, int) override { treeChildrenChanged (parentTree); }
  109. void valueTreeParentChanged (ValueTree&) override {}
  110. void treeChildrenChanged (const ValueTree& parentTree)
  111. {
  112. if (parentTree == tree)
  113. {
  114. refreshSubItems();
  115. treeHasChanged();
  116. setOpen (true);
  117. }
  118. }
  119. JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (ValueTreeItem)
  120. };
  121. //==============================================================================
  122. class ValueTreesDemo : public Component,
  123. public DragAndDropContainer,
  124. private ButtonListener,
  125. private Timer
  126. {
  127. public:
  128. ValueTreesDemo()
  129. : undoButton ("Undo"),
  130. redoButton ("Redo")
  131. {
  132. addAndMakeVisible (tree);
  133. tree.setDefaultOpenness (true);
  134. tree.setMultiSelectEnabled (true);
  135. tree.setRootItem (rootItem = new ValueTreeItem (createRootValueTree(), undoManager));
  136. tree.setColour (TreeView::backgroundColourId, Colours::white);
  137. addAndMakeVisible (undoButton);
  138. addAndMakeVisible (redoButton);
  139. undoButton.addListener (this);
  140. redoButton.addListener (this);
  141. startTimer (500);
  142. }
  143. ~ValueTreesDemo()
  144. {
  145. tree.setRootItem (nullptr);
  146. }
  147. void paint (Graphics& g) override
  148. {
  149. fillStandardDemoBackground (g);
  150. }
  151. void resized() override
  152. {
  153. Rectangle<int> r (getLocalBounds().reduced (8));
  154. Rectangle<int> buttons (r.removeFromBottom (22));
  155. undoButton.setBounds (buttons.removeFromLeft (100));
  156. buttons.removeFromLeft (6);
  157. redoButton.setBounds (buttons.removeFromLeft (100));
  158. r.removeFromBottom (4);
  159. tree.setBounds (r);
  160. }
  161. static ValueTree createTree (const String& desc)
  162. {
  163. ValueTree t ("Item");
  164. t.setProperty ("name", desc, nullptr);
  165. return t;
  166. }
  167. static ValueTree createRootValueTree()
  168. {
  169. ValueTree vt = createTree ("This demo displays a ValueTree as a treeview.");
  170. vt.addChild (createTree ("You can drag around the nodes to rearrange them"), -1, nullptr);
  171. vt.addChild (createTree ("..and press 'delete' to delete them"), -1, nullptr);
  172. vt.addChild (createTree ("Then, you can use the undo/redo buttons to undo these changes"), -1, nullptr);
  173. int n = 1;
  174. vt.addChild (createRandomTree (n, 0), -1, nullptr);
  175. return vt;
  176. }
  177. static ValueTree createRandomTree (int& counter, int depth)
  178. {
  179. ValueTree t = createTree ("Item " + String (counter++));
  180. if (depth < 3)
  181. for (int i = 1 + Random::getSystemRandom().nextInt (7); --i >= 0;)
  182. t.addChild (createRandomTree (counter, depth + 1), -1, nullptr);
  183. return t;
  184. }
  185. void deleteSelectedItems()
  186. {
  187. OwnedArray<ValueTree> selectedItems;
  188. ValueTreeItem::getSelectedTreeViewItems (tree, selectedItems);
  189. for (int i = selectedItems.size(); --i >= 0;)
  190. {
  191. ValueTree& v = *selectedItems.getUnchecked(i);
  192. if (v.getParent().isValid())
  193. v.getParent().removeChild (v, &undoManager);
  194. }
  195. }
  196. bool keyPressed (const KeyPress& key) override
  197. {
  198. if (key == KeyPress::deleteKey)
  199. {
  200. deleteSelectedItems();
  201. return true;
  202. }
  203. if (key == KeyPress ('z', ModifierKeys::commandModifier, 0))
  204. {
  205. undoManager.undo();
  206. return true;
  207. }
  208. if (key == KeyPress ('z', ModifierKeys::commandModifier | ModifierKeys::shiftModifier, 0))
  209. {
  210. undoManager.redo();
  211. return true;
  212. }
  213. return Component::keyPressed (key);
  214. }
  215. void buttonClicked (Button* b) override
  216. {
  217. if (b == &undoButton)
  218. undoManager.undo();
  219. else if (b == &redoButton)
  220. undoManager.redo();
  221. }
  222. private:
  223. TreeView tree;
  224. TextButton undoButton, redoButton;
  225. ScopedPointer<ValueTreeItem> rootItem;
  226. UndoManager undoManager;
  227. void timerCallback() override
  228. {
  229. undoManager.beginNewTransaction();
  230. }
  231. JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (ValueTreesDemo)
  232. };
  233. // This static object will register this demo type in a global list of demos..
  234. static JuceDemoType<ValueTreesDemo> demo ("40 ValueTrees");