using namespace ARDOUR;
using namespace PBD;
+/* used for templates (previously: !full_state) */
+bool Automatable::skip_saving_automation = false;
+
const string Automatable::xml_node_name = X_("Automation");
Automatable::Automatable(Session& session)
Automatable::Automatable (const Automatable& other)
: ControlSet (other)
+ , Slavable ()
, _a_session (other._a_session)
{
Glib::Threads::Mutex::Lock lm (other._control_lock);
Automatable::~Automatable ()
{
- {
- Glib::Threads::Mutex::Lock lm (_control_lock);
-
- for (Controls::const_iterator li = _controls.begin(); li != _controls.end(); ++li) {
- boost::dynamic_pointer_cast<AutomationControl>(li->second)->drop_references ();
- }
+ Glib::Threads::Mutex::Lock lm (_control_lock);
+ for (Controls::const_iterator li = _controls.begin(); li != _controls.end(); ++li) {
+ boost::dynamic_pointer_cast<AutomationControl>(li->second)->drop_references ();
}
}
continue;
}
- if (_can_automate_list.find (param) == _can_automate_list.end ()) {
- warning << "Ignored automation data for non-automatable parameter" << endl;
- continue;
- }
-
if (!id_prop) {
warning << "AutomationList node without automation-id property, "
<< "using default: " << EventTypeMap::instance().to_symbol(legacy_param) << endmsg;
}
+ if (_can_automate_list.find (param) == _can_automate_list.end ()) {
+ boost::shared_ptr<AutomationControl> actl = automation_control (param);
+ if (actl && (*niter)->children().size() > 0 && Config->get_limit_n_automatables () > 0) {
+ actl->set_flags (Controllable::Flag ((int)actl->flags() & ~Controllable::NotAutomatable));
+ can_automate (param);
+ info << "Marked parmater as automatable" << endl;
+ } else {
+ warning << "Ignored automation data for non-automatable parameter" << endl;
+ continue;
+ }
+ }
+
+
boost::shared_ptr<AutomationControl> existing = automation_control (param);
if (existing) {
return result;
}
-void
-Automatable::set_parameter_automation_style (Evoral::Parameter param, AutoStyle s)
-{
- Glib::Threads::Mutex::Lock lm (control_lock());
-
- boost::shared_ptr<AutomationControl> c = automation_control(param, true);
-
- if (c && (s != c->automation_style())) {
- c->set_automation_style (s);
- _a_session.set_dirty ();
- }
-}
-
-AutoStyle
-Automatable::get_parameter_automation_style (Evoral::Parameter param)
-{
- Glib::Threads::Mutex::Lock lm (control_lock());
-
- boost::shared_ptr<Evoral::Control> c = control(param);
- boost::shared_ptr<AutomationList> l = boost::dynamic_pointer_cast<AutomationList>(c->list());
-
- if (c) {
- return l->automation_style();
- } else {
- return Absolute; // whatever
- }
-}
-
void
Automatable::protect_automation ()
{
case Write:
l->set_automation_state (Off);
break;
+ case Latch:
+ /* fall through */
case Touch:
l->set_automation_state (Play);
break;
}
void
-Automatable::transport_located (framepos_t now)
+Automatable::non_realtime_locate (samplepos_t now)
{
+ bool rolling = _a_session.transport_rolling ();
+
for (Controls::iterator li = controls().begin(); li != controls().end(); ++li) {
boost::shared_ptr<AutomationControl> c
= boost::dynamic_pointer_cast<AutomationControl>(li->second);
if (c) {
- boost::shared_ptr<AutomationList> l
+ boost::shared_ptr<AutomationList> l
= boost::dynamic_pointer_cast<AutomationList>(c->list());
- if (l) {
- l->start_write_pass (now);
+ if (!l) {
+ continue;
+ }
+
+ bool am_touching = c->touching ();
+ if (rolling && am_touching) {
+ /* when locating while rolling, and writing automation,
+ * start a new write pass.
+ * compare to compare to non_realtime_transport_stop()
+ */
+ const bool list_did_write = !l->in_new_write_pass ();
+ c->stop_touch (-1); // time is irrelevant
+ l->stop_touch (-1);
+ c->commit_transaction (list_did_write);
+ l->write_pass_finished (now, Config->get_automation_thinning_factor ());
+
+ if (l->automation_state () == Write) {
+ l->set_automation_state (Touch);
+ }
+ if (l->automation_playback ()) {
+ c->set_value_unchecked (c->list ()->eval (now));
+ }
+ }
+
+ l->start_write_pass (now);
+
+ if (rolling && am_touching) {
+ c->start_touch (now);
}
}
}
}
void
-Automatable::transport_stopped (framepos_t now)
+Automatable::non_realtime_transport_stop (samplepos_t now, bool /*flush_processors*/)
{
for (Controls::iterator li = controls().begin(); li != controls().end(); ++li) {
boost::shared_ptr<AutomationControl> c =
*/
const bool list_did_write = !l->in_new_write_pass ();
- l->stop_touch (true, now);
+ c->stop_touch (now);
+ l->stop_touch (now);
c->commit_transaction (list_did_write);
}
}
+void
+Automatable::automation_run (samplepos_t start, pframes_t nframes)
+{
+ for (Controls::iterator li = controls().begin(); li != controls().end(); ++li) {
+ boost::shared_ptr<AutomationControl> c =
+ boost::dynamic_pointer_cast<AutomationControl>(li->second);
+ if (!c) {
+ continue;
+ }
+ c->automation_run (start, nframes);
+ }
+}
+
boost::shared_ptr<Evoral::Control>
Automatable::control_factory(const Evoral::Parameter& param)
{
return boost::shared_ptr<Evoral::Control>(control);
}
+boost::shared_ptr<AutomationControl>
+Automatable::automation_control (PBD::ID const & id) const
+{
+ Controls::const_iterator li;
+
+ for (li = _controls.begin(); li != _controls.end(); ++li) {
+ boost::shared_ptr<AutomationControl> ac = boost::dynamic_pointer_cast<AutomationControl> (li->second);
+ if (ac && (ac->id() == id)) {
+ return ac;
+ }
+ }
+
+ return boost::shared_ptr<AutomationControl>();
+}
+
boost::shared_ptr<AutomationControl>
Automatable::automation_control (const Evoral::Parameter& id, bool create)
{
ControlSet::clear_controls ();
}
-string
-Automatable::value_as_string (boost::shared_ptr<const AutomationControl> ac) const
-{
- return ARDOUR::value_as_string(ac->desc(), ac->get_value());
-}
-
bool
Automatable::find_next_event (double now, double end, Evoral::ControlEvent& next_event, bool only_active) const
{
continue;
}
+ boost::shared_ptr<SlavableAutomationControl> sc
+ = boost::dynamic_pointer_cast<SlavableAutomationControl>(li->second);
+
+ if (sc) {
+ sc->find_next_event (now, end, next_event);
+ }
+
Evoral::ControlList::const_iterator i;
boost::shared_ptr<const Evoral::ControlList> alist (li->second->list());
Evoral::ControlEvent cp (now, 0.0f);