mirror of
https://github.com/juce-framework/JUCE.git
synced 2026-01-10 23:44:24 +00:00
290 lines
6.7 KiB
C++
290 lines
6.7 KiB
C++
/*
|
|
==============================================================================
|
|
|
|
This file is part of the JUCE library.
|
|
Copyright (c) 2013 - Raw Material Software Ltd.
|
|
|
|
Permission is granted to use this software under the terms of either:
|
|
a) the GPL v2 (or any later version)
|
|
b) the Affero GPL v3
|
|
|
|
Details of these licenses can be found at: www.gnu.org/licenses
|
|
|
|
JUCE is distributed in the hope that it will be useful, but WITHOUT ANY
|
|
WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR
|
|
A PARTICULAR PURPOSE. See the GNU General Public License for more details.
|
|
|
|
------------------------------------------------------------------------------
|
|
|
|
To release a closed-source product which uses JUCE, commercial licenses are
|
|
available: visit www.juce.com for more information.
|
|
|
|
==============================================================================
|
|
*/
|
|
|
|
struct SharedValueSourceUpdater : private AsyncUpdater
|
|
{
|
|
public:
|
|
SharedValueSourceUpdater() {}
|
|
|
|
void update (Value::ValueSource* const source)
|
|
{
|
|
{
|
|
const ScopedLock sl (lock);
|
|
sourcesNeedingUpdate.addIfNotAlreadyThere (source);
|
|
}
|
|
|
|
triggerAsyncUpdate();
|
|
}
|
|
|
|
void valueDeleted (Value::ValueSource* const source)
|
|
{
|
|
const ScopedLock sl (lock);
|
|
sourcesNeedingUpdate.removeFirstMatchingValue (source);
|
|
}
|
|
|
|
private:
|
|
Array<Value::ValueSource*> sourcesNeedingUpdate;
|
|
CriticalSection lock;
|
|
|
|
void handleAsyncUpdate() override
|
|
{
|
|
SharedResourcePointer<SharedValueSourceUpdater> localRef;
|
|
|
|
int maxCallbacks = sourcesNeedingUpdate.size();
|
|
|
|
for (;;)
|
|
{
|
|
ReferenceCountedObjectPtr<Value::ValueSource> toUpdate;
|
|
|
|
{
|
|
const ScopedLock sl (lock);
|
|
toUpdate = sourcesNeedingUpdate.remove (0);
|
|
}
|
|
|
|
if (toUpdate == nullptr)
|
|
break;
|
|
|
|
toUpdate->sendChangeMessage (true);
|
|
|
|
if (--maxCallbacks <= 0)
|
|
{
|
|
triggerAsyncUpdate();
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (SharedValueSourceUpdater)
|
|
};
|
|
|
|
struct Value::ValueSource::Pimpl
|
|
{
|
|
SharedResourcePointer<SharedValueSourceUpdater> updater;
|
|
};
|
|
|
|
Value::ValueSource::ValueSource()
|
|
{
|
|
}
|
|
|
|
Value::ValueSource::~ValueSource()
|
|
{
|
|
if (pimpl != nullptr)
|
|
pimpl->updater->valueDeleted (this);
|
|
}
|
|
|
|
void Value::ValueSource::sendChangeMessage (const bool synchronous)
|
|
{
|
|
const int numListeners = valuesWithListeners.size();
|
|
|
|
if (numListeners > 0)
|
|
{
|
|
if (synchronous)
|
|
{
|
|
const ReferenceCountedObjectPtr<ValueSource> localRef (this);
|
|
|
|
for (int i = numListeners; --i >= 0;)
|
|
if (Value* const v = valuesWithListeners[i])
|
|
v->callListeners();
|
|
}
|
|
else
|
|
{
|
|
if (pimpl == nullptr)
|
|
pimpl = new Pimpl();
|
|
|
|
pimpl->updater->update (this);
|
|
}
|
|
}
|
|
}
|
|
|
|
//==============================================================================
|
|
class SimpleValueSource : public Value::ValueSource
|
|
{
|
|
public:
|
|
SimpleValueSource()
|
|
{
|
|
}
|
|
|
|
SimpleValueSource (const var& initialValue)
|
|
: value (initialValue)
|
|
{
|
|
}
|
|
|
|
var getValue() const
|
|
{
|
|
return value;
|
|
}
|
|
|
|
void setValue (const var& newValue)
|
|
{
|
|
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::operator= (const Value& other)
|
|
{
|
|
value = other.value;
|
|
return *this;
|
|
}
|
|
|
|
#if JUCE_COMPILER_SUPPORTS_MOVE_SEMANTICS
|
|
Value::Value (Value&& other) noexcept
|
|
: value (static_cast<ReferenceCountedObjectPtr<ValueSource>&&> (other.value))
|
|
{
|
|
}
|
|
|
|
Value& Value::operator= (Value&& other) noexcept
|
|
{
|
|
value = static_cast<ReferenceCountedObjectPtr<ValueSource>&&> (other.value);
|
|
return *this;
|
|
}
|
|
#endif
|
|
|
|
Value::~Value()
|
|
{
|
|
if (listeners.size() > 0)
|
|
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 (ValueListener* const listener)
|
|
{
|
|
if (listener != nullptr)
|
|
{
|
|
if (listeners.size() == 0)
|
|
value->valuesWithListeners.add (this);
|
|
|
|
listeners.add (listener);
|
|
}
|
|
}
|
|
|
|
void Value::removeListener (ValueListener* const 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 (&ValueListener::valueChanged, v);
|
|
}
|
|
}
|
|
|
|
OutputStream& JUCE_CALLTYPE operator<< (OutputStream& stream, const Value& value)
|
|
{
|
|
return stream << value.toString();
|
|
}
|