13
0
livetrax/libs/pbd/controllable.cc
Robin Gareus 74c4ca3e52
Reduce reliance on boost - the hard part
the rest from `tools/convert_boost.sh`.

* replace boost::function, boost::bind with std::function and std::bind.

This required some manual fixes, notably std::placeholders,
some static_casts<>, and boost::function::clear -> = {}.
2024-10-19 03:47:21 +02:00

185 lines
5.0 KiB
C++
Raw Blame History

/*
* Copyright (C) 2006-2017 Paul Davis <paul@linuxaudiosystems.com>
* Copyright (C) 2009-2011 Carl Hetherington <carl@carlh.net>
* Copyright (C) 2009 David Robillard <d@drobilla.net>
* Copyright (C) 2016-2019 Robin Gareus <robin@gareus.org>
* Copyright (C) 2016 Tim Mayberry <mojofunk@gmail.com>
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program 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.
*
* You should have received a copy of the GNU General Public License along
* with this program; if not, write to the Free Software Foundation, Inc.,
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
*/
#include "pbd/controllable.h"
#include "pbd/enumwriter.h"
#include "pbd/xml++.h"
#include "pbd/error.h"
#include "pbd/types_convert.h"
#include "pbd/string_convert.h"
#include "pbd/i18n.h"
using namespace PBD;
using namespace std;
PBD::Signal<bool(std::weak_ptr<PBD::Controllable> )> Controllable::StartLearning;
PBD::Signal<void(std::weak_ptr<PBD::Controllable> )> Controllable::StopLearning;
PBD::Signal<void(std::weak_ptr<PBD::Controllable> )> Controllable::GUIFocusChanged;
PBD::Signal<void(std::weak_ptr<PBD::Controllable> )> Controllable::ControlTouched;
Glib::Threads::RWLock Controllable::registry_lock;
Controllable::Controllables Controllable::registry;
PBD::ScopedConnectionList Controllable::registry_connections;
const std::string Controllable::xml_node_name = X_("Controllable");
Controllable::Controllable (const string& name, Flag f)
: _name (name)
, _flags (f)
, _touching (false)
{
add (*this);
}
XMLNode&
Controllable::get_state () const
{
XMLNode* node = new XMLNode (xml_node_name);
/* Waves' "Pressure3" has a parameter called "µ-iness"
* which causes a parser error : Input is not proper UTF-8, indicate encoding !
* Bytes: 0xB5 0x2D 0x69 0x6E
* <Controllable name="<22>-iness" id="2391" flags="" value="0.000000000000" p
*/
// this is not reloaded from XML, but it must be present because it is
// used to find and identify XML nodes by various Controllable-derived objects
node->set_property (X_("name"), _name);
node->set_property (X_("id"), id());
node->set_property (X_("flags"), _flags);
node->set_property (X_("value"), get_save_value());
if (_extra_xml) {
node->add_child_copy (*_extra_xml);
}
return *node;
}
int
Controllable::set_state (const XMLNode& node, int /*version*/)
{
Stateful::save_extra_xml (node);
set_id (node);
if (node.get_property (X_("flags"), _flags)) {
_flags = Flag(_flags | (_flags & Controllable::RealTime));
}
float val;
if (node.get_property (X_("value"), val)) {
set_value (val, NoGroup);
}
return 0;
}
void
Controllable::set_flags (Flag f)
{
_flags = f;
}
void
Controllable::set_flag (Flag f)
{
_flags = Flag ((int)_flags | f);
}
void
Controllable::clear_flag (Flag f)
{
_flags = Flag ((int)_flags & ~f);
}
void
Controllable::add (Controllable& ctl)
{
Glib::Threads::RWLock::WriterLock lm (registry_lock);
registry.insert (&ctl);
ctl.DropReferences.connect_same_thread (registry_connections, std::bind (&Controllable::remove, &ctl));
ctl.Destroyed.connect_same_thread (registry_connections, std::bind (&Controllable::remove, &ctl));
}
void
Controllable::remove (Controllable* ctl)
{
Glib::Threads::RWLock::WriterLock lm (registry_lock);
Controllables::iterator i = std::find (registry.begin(), registry.end(), ctl);
if (i != registry.end()) {
registry.erase (i);
}
}
std::shared_ptr<Controllable>
Controllable::by_id (const ID& id)
{
Glib::Threads::RWLock::ReaderLock lm (registry_lock);
for (Controllables::iterator i = registry.begin(); i != registry.end(); ++i) {
if ((*i)->id() == id) {
return (*i)->shared_from_this ();
}
}
return std::shared_ptr<Controllable>();
}
ControllableSet
Controllable::registered_controllables ()
{
ControllableSet rv;
Glib::Threads::RWLock::ReaderLock lm (registry_lock);
for (auto const& i : registry) {
try {
rv.insert (i->shared_from_this ());
} catch (...) {
/* ignore boost::bad_weak_ptr */
// cout << "No shared ctrl: " << i->name() << "\n";
}
}
return rv;
}
void
Controllable::dump_registry ()
{
Glib::Threads::RWLock::ReaderLock lm (registry_lock);
if (registry.size() == 0) {
return;
}
unsigned int cnt = 0;
cout << "-- List Of Registered Controllables\n";
for (Controllables::iterator i = registry.begin(); i != registry.end(); ++i, ++cnt) {
cout << "CTRL: " << (*i)->name() << "\n";
}
cout << "Total number of registered controllables: " << cnt << "\n";
}
void
Controllable::set_interface (float fraction, bool rotary, GroupControlDisposition gcd)
{
fraction = std::min (std::max (0.0f, fraction), 1.0f);
set_value (interface_to_internal (fraction, rotary), gcd);
}