|
- /*
- ==============================================================================
-
- This file is part of the JUCE library.
- Copyright (c) 2020 - Raw Material Software Limited
-
- JUCE is an open source library subject to commercial or open-source
- licensing.
-
- By using JUCE, you agree to the terms of both the JUCE 6 End-User License
- Agreement and JUCE Privacy Policy (both effective as of the 16th June 2020).
-
- End User License Agreement: www.juce.com/juce-6-licence
- Privacy Policy: www.juce.com/juce-privacy-policy
-
- Or: You may also use this code under the terms of the GPL v3 (see
- www.gnu.org/licenses).
-
- JUCE IS PROVIDED "AS IS" WITHOUT ANY WARRANTY, AND ALL WARRANTIES, WHETHER
- EXPRESSED OR IMPLIED, INCLUDING MERCHANTABILITY AND FITNESS FOR PURPOSE, ARE
- DISCLAIMED.
-
- ==============================================================================
- */
-
- namespace juce
- {
-
- Value::ValueSource::ValueSource()
- {
- }
-
- Value::ValueSource::~ValueSource()
- {
- cancelPendingUpdate();
- }
-
- void Value::ValueSource::handleAsyncUpdate()
- {
- sendChangeMessage (true);
- }
-
- void Value::ValueSource::sendChangeMessage (const bool synchronous)
- {
- const int numListeners = valuesWithListeners.size();
-
- if (numListeners > 0)
- {
- if (synchronous)
- {
- const ReferenceCountedObjectPtr<ValueSource> localRef (this);
-
- cancelPendingUpdate();
-
- for (int i = numListeners; --i >= 0;)
- if (Value* const v = valuesWithListeners[i])
- v->callListeners();
- }
- else
- {
- triggerAsyncUpdate();
- }
- }
- }
-
- //==============================================================================
- class SimpleValueSource : public Value::ValueSource
- {
- public:
- SimpleValueSource()
- {
- }
-
- SimpleValueSource (const var& initialValue)
- : value (initialValue)
- {
- }
-
- var getValue() const override
- {
- return value;
- }
-
- void setValue (const var& newValue) override
- {
- if (! newValue.equalsWithSameType (value))
- {
- value = newValue;
- sendChangeMessage (false);
- }
- }
-
- private:
- var value;
-
- JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (SimpleValueSource)
- };
-
-
- //==============================================================================
- Value::Value() : value (new SimpleValueSource())
- {
- }
-
- Value::Value (ValueSource* const v) : value (v)
- {
- jassert (v != nullptr);
- }
-
- Value::Value (const var& initialValue) : value (new SimpleValueSource (initialValue))
- {
- }
-
- Value::Value (const Value& other) : value (other.value)
- {
- }
-
- Value::Value (Value&& other) noexcept
- {
- // moving a Value with listeners will lose those listeners, which
- // probably isn't what you wanted to happen!
- jassert (other.listeners.size() == 0);
-
- other.removeFromListenerList();
- value = std::move (other.value);
- }
-
- Value& Value::operator= (Value&& other) noexcept
- {
- // moving a Value with listeners will lose those listeners, which
- // probably isn't what you wanted to happen!
- jassert (other.listeners.size() == 0);
-
- other.removeFromListenerList();
- value = std::move (other.value);
- return *this;
- }
-
- Value::~Value()
- {
- removeFromListenerList();
- }
-
- void Value::removeFromListenerList()
- {
- if (listeners.size() > 0 && value != nullptr) // may be nullptr after a move operation
- value->valuesWithListeners.removeValue (this);
- }
-
- //==============================================================================
- var Value::getValue() const
- {
- return value->getValue();
- }
-
- Value::operator var() const
- {
- return value->getValue();
- }
-
- void Value::setValue (const var& newValue)
- {
- value->setValue (newValue);
- }
-
- String Value::toString() const
- {
- return value->getValue().toString();
- }
-
- Value& Value::operator= (const var& newValue)
- {
- value->setValue (newValue);
- return *this;
- }
-
- void Value::referTo (const Value& valueToReferTo)
- {
- if (valueToReferTo.value != value)
- {
- if (listeners.size() > 0)
- {
- value->valuesWithListeners.removeValue (this);
- valueToReferTo.value->valuesWithListeners.add (this);
- }
-
- value = valueToReferTo.value;
- callListeners();
- }
- }
-
- bool Value::refersToSameSourceAs (const Value& other) const
- {
- return value == other.value;
- }
-
- bool Value::operator== (const Value& other) const
- {
- return value == other.value || value->getValue() == other.getValue();
- }
-
- bool Value::operator!= (const Value& other) const
- {
- return value != other.value && value->getValue() != other.getValue();
- }
-
- //==============================================================================
- void Value::addListener (Value::Listener* listener)
- {
- if (listener != nullptr)
- {
- if (listeners.size() == 0)
- value->valuesWithListeners.add (this);
-
- listeners.add (listener);
- }
- }
-
- void Value::removeListener (Value::Listener* listener)
- {
- listeners.remove (listener);
-
- if (listeners.size() == 0)
- value->valuesWithListeners.removeValue (this);
- }
-
- void Value::callListeners()
- {
- if (listeners.size() > 0)
- {
- Value v (*this); // (create a copy in case this gets deleted by a callback)
- listeners.call ([&] (Value::Listener& l) { l.valueChanged (v); });
- }
- }
-
- OutputStream& JUCE_CALLTYPE operator<< (OutputStream& stream, const Value& value)
- {
- return stream << value.toString();
- }
-
- } // namespace juce
|