Audio plugin host https://kx.studio/carla
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.

328 lines
11KB

  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. //==============================================================================
  16. /**
  17. This class acts as a wrapper around a property inside a ValueTree.
  18. If the property inside the ValueTree is missing it will return a default value,
  19. which can be specified in the constructor or by calling setDefault().
  20. @tags{DataStructures}
  21. */
  22. class JUCE_API ValueTreePropertyWithDefault : private Value::Listener
  23. {
  24. public:
  25. //==============================================================================
  26. /** Creates an uninitialised ValueTreePropertyWithDefault object.
  27. Initialise it using one of the referTo() methods.
  28. */
  29. ValueTreePropertyWithDefault() = default;
  30. /** Creates a ValueTreePropertyWithDefault object for the specified property.
  31. The default value will be an empty var.
  32. */
  33. ValueTreePropertyWithDefault (ValueTree& tree,
  34. const Identifier& propertyID,
  35. UndoManager* um)
  36. {
  37. referTo (tree, propertyID, um);
  38. }
  39. /** Creates an ValueTreePropertyWithDefault object for the specified property.
  40. The default value will be defaultToUse.
  41. */
  42. ValueTreePropertyWithDefault (ValueTree& tree,
  43. const Identifier& propertyID,
  44. UndoManager* um,
  45. var defaultToUse)
  46. {
  47. referTo (tree, propertyID, um, defaultToUse);
  48. }
  49. /** Creates a ValueTreePropertyWithDefault object for the specified property.
  50. The default value will be defaultToUse.
  51. Use this constructor if the underlying var object being controlled is an array and
  52. it will handle the conversion to/from a delimited String that can be written to
  53. XML format.
  54. */
  55. ValueTreePropertyWithDefault (ValueTree& tree,
  56. const Identifier& propertyID,
  57. UndoManager* um,
  58. var defaultToUse,
  59. StringRef arrayDelimiter)
  60. {
  61. referTo (tree, propertyID, um, defaultToUse, arrayDelimiter);
  62. }
  63. /** Creates a ValueTreePropertyWithDefault object from another ValueTreePropertyWithDefault object. */
  64. ValueTreePropertyWithDefault (const ValueTreePropertyWithDefault& other)
  65. {
  66. referToWithDefault (other.targetTree,
  67. other.targetProperty,
  68. other.undoManager,
  69. other.defaultValue,
  70. other.delimiter);
  71. }
  72. /** Destructor. */
  73. ~ValueTreePropertyWithDefault() override
  74. {
  75. defaultValue.removeListener (this);
  76. }
  77. //==============================================================================
  78. /** Returns the current value of the property.
  79. If the property does not exist this returns the default value.
  80. */
  81. var get() const noexcept
  82. {
  83. if (isUsingDefault())
  84. return defaultValue;
  85. if (delimiter.isNotEmpty())
  86. return delimitedStringToVarArray (targetTree[targetProperty].toString(), delimiter);
  87. return targetTree[targetProperty];
  88. }
  89. /** Returns the current property as a Value object. */
  90. Value getPropertyAsValue() { return targetTree.getPropertyAsValue (targetProperty, undoManager); }
  91. /** Returns the current default value. */
  92. var getDefault() const { return defaultValue; }
  93. /** Sets the default value to a new var. */
  94. void setDefault (const var& newDefault) { defaultValue = newDefault; }
  95. /** Returns true if the property does not exist in the referenced ValueTree. */
  96. bool isUsingDefault() const { return ! targetTree.hasProperty (targetProperty); }
  97. /** Removes the property from the referenced ValueTree. */
  98. void resetToDefault() noexcept { targetTree.removeProperty (targetProperty, nullptr); }
  99. /** You can assign a lambda to this callback and it will called when the default
  100. value is changed.
  101. @see setDefault
  102. */
  103. std::function<void()> onDefaultChange;
  104. //==============================================================================
  105. /** Sets the property and returns the new ValueTreePropertyWithDefault.
  106. This will modify the property in the referenced ValueTree.
  107. */
  108. ValueTreePropertyWithDefault& operator= (const var& newValue)
  109. {
  110. setValue (newValue, undoManager);
  111. return *this;
  112. }
  113. /** Sets the property.
  114. This will modify the property in the referenced ValueTree.
  115. */
  116. void setValue (const var& newValue, UndoManager* undoManagerToUse)
  117. {
  118. if (auto* array = newValue.getArray())
  119. targetTree.setProperty (targetProperty, varArrayToDelimitedString (*array, delimiter), undoManagerToUse);
  120. else
  121. targetTree.setProperty (targetProperty, newValue, undoManagerToUse);
  122. }
  123. //==============================================================================
  124. /** Makes the ValueTreePropertyWithDefault refer to the specified property inside
  125. the given ValueTree.
  126. The default value will be an empty var.
  127. */
  128. void referTo (ValueTree tree,
  129. const Identifier& property,
  130. UndoManager* um)
  131. {
  132. referToWithDefault (tree,
  133. property,
  134. um,
  135. Value (new SynchronousValueSource (var())),
  136. {});
  137. }
  138. /** Makes the ValueTreePropertyWithDefault refer to the specified property inside
  139. the given ValueTree.
  140. The default value will be defaultVal.
  141. */
  142. void referTo (ValueTree tree,
  143. const Identifier& property,
  144. UndoManager* um,
  145. var defaultVal)
  146. {
  147. referToWithDefault (tree,
  148. property,
  149. um,
  150. Value (new SynchronousValueSource (defaultVal)),
  151. {});
  152. }
  153. /** Makes the ValueTreePropertyWithDefault refer to the specified property inside
  154. the given ValueTree.
  155. The default value will be defaultVal.
  156. */
  157. void referTo (ValueTree tree,
  158. const Identifier& property,
  159. UndoManager* um,
  160. var defaultVal,
  161. StringRef arrayDelimiter)
  162. {
  163. referToWithDefault (tree,
  164. property,
  165. um,
  166. Value (new SynchronousValueSource (defaultVal)),
  167. arrayDelimiter);
  168. }
  169. //==============================================================================
  170. /** Returns a reference to the ValueTree containing the referenced property. */
  171. ValueTree& getValueTree() noexcept { return targetTree; }
  172. /** Returns the property ID of the referenced property. */
  173. Identifier& getPropertyID() noexcept { return targetProperty; }
  174. /** Returns the UndoManager that is being used. */
  175. UndoManager* getUndoManager() noexcept { return undoManager; }
  176. //==============================================================================
  177. ValueTreePropertyWithDefault& operator= (const ValueTreePropertyWithDefault& other)
  178. {
  179. referToWithDefault (other.targetTree,
  180. other.targetProperty,
  181. other.undoManager,
  182. other.defaultValue,
  183. other.delimiter);
  184. return *this;
  185. }
  186. private:
  187. //==============================================================================
  188. class SynchronousValueSource : public Value::ValueSource
  189. {
  190. public:
  191. explicit SynchronousValueSource (const var& initialValue)
  192. : value (initialValue)
  193. {
  194. }
  195. var getValue() const override
  196. {
  197. return value;
  198. }
  199. void setValue (const var& newValue) override
  200. {
  201. if (! newValue.equalsWithSameType (value))
  202. {
  203. value = newValue;
  204. sendChangeMessage (true);
  205. }
  206. }
  207. private:
  208. var value;
  209. JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (SynchronousValueSource)
  210. };
  211. //==============================================================================
  212. static String varArrayToDelimitedString (const Array<var>& input, StringRef delim)
  213. {
  214. // if you are trying to control a var that is an array then you need to
  215. // set a delimiter string that will be used when writing to XML!
  216. jassert (delim.isNotEmpty());
  217. StringArray elements;
  218. for (auto& v : input)
  219. elements.add (v.toString());
  220. return elements.joinIntoString (delim);
  221. }
  222. static Array<var> delimitedStringToVarArray (StringRef input, StringRef delim)
  223. {
  224. Array<var> arr;
  225. for (auto t : StringArray::fromTokens (input, delim, {}))
  226. arr.add (t);
  227. return arr;
  228. }
  229. void valueChanged (Value&) override
  230. {
  231. if (onDefaultChange != nullptr)
  232. onDefaultChange();
  233. }
  234. void referToWithDefault (ValueTree v,
  235. const Identifier& i,
  236. UndoManager* um,
  237. const Value& defaultVal,
  238. StringRef del)
  239. {
  240. targetTree = v;
  241. targetProperty = i;
  242. undoManager = um;
  243. defaultValue.referTo (defaultVal);
  244. delimiter = del;
  245. defaultValue.addListener (this);
  246. }
  247. //==============================================================================
  248. ValueTree targetTree;
  249. Identifier targetProperty;
  250. UndoManager* undoManager = nullptr;
  251. Value defaultValue;
  252. String delimiter;
  253. //==============================================================================
  254. JUCE_LEAK_DETECTOR (ValueTreePropertyWithDefault)
  255. };
  256. //==============================================================================
  257. #ifndef DOXYGEN
  258. using ValueWithDefault [[deprecated ("This class has been renamed to better describe what is does. "
  259. "This declaration is here for backwards compatibility and new "
  260. "code should use the new class name.")]]
  261. = ValueTreePropertyWithDefault;
  262. #endif
  263. } // namespace juce