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.

300 lines
9.1KB

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