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.

220 lines
6.9KB

  1. /*
  2. ==============================================================================
  3. This file is part of the JUCE library.
  4. Copyright (c) 2017 - ROLI Ltd.
  5. JUCE is an open source library subject to commercial or open-source
  6. licensing.
  7. By using JUCE, you agree to the terms of both the JUCE 5 End-User License
  8. Agreement and JUCE 5 Privacy Policy (both updated and effective as of the
  9. 27th April 2017).
  10. End User License Agreement: www.juce.com/juce-5-licence
  11. Privacy Policy: www.juce.com/juce-5-privacy-policy
  12. Or: You may also use this code under the terms of the GPL v3 (see
  13. www.gnu.org/licenses).
  14. JUCE IS PROVIDED "AS IS" WITHOUT ANY WARRANTY, AND ALL WARRANTIES, WHETHER
  15. EXPRESSED OR IMPLIED, INCLUDING MERCHANTABILITY AND FITNESS FOR PURPOSE, ARE
  16. DISCLAIMED.
  17. ==============================================================================
  18. */
  19. namespace ValueTreeSynchroniserHelpers
  20. {
  21. enum ChangeType
  22. {
  23. propertyChanged = 1,
  24. fullSync = 2,
  25. childAdded = 3,
  26. childRemoved = 4,
  27. childMoved = 5
  28. };
  29. static void getValueTreePath (ValueTree v, const ValueTree& topLevelTree, Array<int>& path)
  30. {
  31. while (v != topLevelTree)
  32. {
  33. ValueTree parent (v.getParent());
  34. if (! parent.isValid())
  35. break;
  36. path.add (parent.indexOf (v));
  37. v = parent;
  38. }
  39. }
  40. static void writeHeader (MemoryOutputStream& stream, ChangeType type)
  41. {
  42. stream.writeByte ((char) type);
  43. }
  44. static void writeHeader (ValueTreeSynchroniser& target, MemoryOutputStream& stream,
  45. ChangeType type, ValueTree v)
  46. {
  47. writeHeader (stream, type);
  48. Array<int> path;
  49. getValueTreePath (v, target.getRoot(), path);
  50. stream.writeCompressedInt (path.size());
  51. for (int i = path.size(); --i >= 0;)
  52. stream.writeCompressedInt (path.getUnchecked(i));
  53. }
  54. static ValueTree readSubTreeLocation (MemoryInputStream& input, ValueTree v)
  55. {
  56. const int numLevels = input.readCompressedInt();
  57. if (! isPositiveAndBelow (numLevels, 65536)) // sanity-check
  58. return ValueTree();
  59. for (int i = numLevels; --i >= 0;)
  60. {
  61. const int index = input.readCompressedInt();
  62. if (! isPositiveAndBelow (index, v.getNumChildren()))
  63. return ValueTree();
  64. v = v.getChild (index);
  65. }
  66. return v;
  67. }
  68. }
  69. ValueTreeSynchroniser::ValueTreeSynchroniser (const ValueTree& tree) : valueTree (tree)
  70. {
  71. valueTree.addListener (this);
  72. }
  73. ValueTreeSynchroniser::~ValueTreeSynchroniser()
  74. {
  75. valueTree.removeListener (this);
  76. }
  77. void ValueTreeSynchroniser::sendFullSyncCallback()
  78. {
  79. MemoryOutputStream m;
  80. writeHeader (m, ValueTreeSynchroniserHelpers::fullSync);
  81. valueTree.writeToStream (m);
  82. stateChanged (m.getData(), m.getDataSize());
  83. }
  84. void ValueTreeSynchroniser::valueTreePropertyChanged (ValueTree& vt, const Identifier& property)
  85. {
  86. MemoryOutputStream m;
  87. ValueTreeSynchroniserHelpers::writeHeader (*this, m, ValueTreeSynchroniserHelpers::propertyChanged, vt);
  88. m.writeString (property.toString());
  89. vt.getProperty (property).writeToStream (m);
  90. stateChanged (m.getData(), m.getDataSize());
  91. }
  92. void ValueTreeSynchroniser::valueTreeChildAdded (ValueTree& parentTree, ValueTree& childTree)
  93. {
  94. const int index = parentTree.indexOf (childTree);
  95. jassert (index >= 0);
  96. MemoryOutputStream m;
  97. ValueTreeSynchroniserHelpers::writeHeader (*this, m, ValueTreeSynchroniserHelpers::childAdded, parentTree);
  98. m.writeCompressedInt (index);
  99. childTree.writeToStream (m);
  100. stateChanged (m.getData(), m.getDataSize());
  101. }
  102. void ValueTreeSynchroniser::valueTreeChildRemoved (ValueTree& parentTree, ValueTree&, int oldIndex)
  103. {
  104. MemoryOutputStream m;
  105. ValueTreeSynchroniserHelpers::writeHeader (*this, m, ValueTreeSynchroniserHelpers::childRemoved, parentTree);
  106. m.writeCompressedInt (oldIndex);
  107. stateChanged (m.getData(), m.getDataSize());
  108. }
  109. void ValueTreeSynchroniser::valueTreeChildOrderChanged (ValueTree& parent, int oldIndex, int newIndex)
  110. {
  111. MemoryOutputStream m;
  112. ValueTreeSynchroniserHelpers::writeHeader (*this, m, ValueTreeSynchroniserHelpers::childMoved, parent);
  113. m.writeCompressedInt (oldIndex);
  114. m.writeCompressedInt (newIndex);
  115. stateChanged (m.getData(), m.getDataSize());
  116. }
  117. void ValueTreeSynchroniser::valueTreeParentChanged (ValueTree&) {} // (No action needed here)
  118. bool ValueTreeSynchroniser::applyChange (ValueTree& root, const void* data, size_t dataSize, UndoManager* undoManager)
  119. {
  120. MemoryInputStream input (data, dataSize, false);
  121. const ValueTreeSynchroniserHelpers::ChangeType type = (ValueTreeSynchroniserHelpers::ChangeType) input.readByte();
  122. if (type == ValueTreeSynchroniserHelpers::fullSync)
  123. {
  124. root = ValueTree::readFromStream (input);
  125. return true;
  126. }
  127. ValueTree v (ValueTreeSynchroniserHelpers::readSubTreeLocation (input, root));
  128. if (! v.isValid())
  129. return false;
  130. switch (type)
  131. {
  132. case ValueTreeSynchroniserHelpers::propertyChanged:
  133. {
  134. Identifier property (input.readString());
  135. v.setProperty (property, var::readFromStream (input), undoManager);
  136. return true;
  137. }
  138. case ValueTreeSynchroniserHelpers::childAdded:
  139. {
  140. const int index = input.readCompressedInt();
  141. v.addChild (ValueTree::readFromStream (input), index, undoManager);
  142. return true;
  143. }
  144. case ValueTreeSynchroniserHelpers::childRemoved:
  145. {
  146. const int index = input.readCompressedInt();
  147. if (isPositiveAndBelow (index, v.getNumChildren()))
  148. {
  149. v.removeChild (index, undoManager);
  150. return true;
  151. }
  152. jassertfalse; // Either received some corrupt data, or the trees have drifted out of sync
  153. break;
  154. }
  155. case ValueTreeSynchroniserHelpers::childMoved:
  156. {
  157. const int oldIndex = input.readCompressedInt();
  158. const int newIndex = input.readCompressedInt();
  159. if (isPositiveAndBelow (oldIndex, v.getNumChildren())
  160. && isPositiveAndBelow (newIndex, v.getNumChildren()))
  161. {
  162. v.moveChild (oldIndex, newIndex, undoManager);
  163. return true;
  164. }
  165. jassertfalse; // Either received some corrupt data, or the trees have drifted out of sync
  166. break;
  167. }
  168. default:
  169. jassertfalse; // Seem to have received some corrupt data?
  170. break;
  171. }
  172. return false;
  173. }