X-Git-Url: https://main.carlh.net/gitweb/?a=blobdiff_plain;f=libs%2Fardour%2Fautomation_control.cc;h=bfa16f9e4aec743c9109721b6ac7293b34f4efce;hb=1fdb3560e85e3f16944568005365b4482643ac02;hp=ef54705e03b499757d2ef92931d7c76a2f07ebcb;hpb=e0aaed6d65f160c328cb8b56d7c6552ee15d65e2;p=ardour.git diff --git a/libs/ardour/automation_control.cc b/libs/ardour/automation_control.cc index ef54705e03..bfa16f9e4a 100644 --- a/libs/ardour/automation_control.cc +++ b/libs/ardour/automation_control.cc @@ -1,6 +1,6 @@ /* - Copyright (C) 2007 Paul Davis - Author: Dave Robillard + Copyright (C) 2007 Paul Davis + Author: David Robillard 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 @@ -19,8 +19,9 @@ */ #include -#include "ardour/automatable.h" + #include "ardour/automation_control.h" +#include "ardour/automation_watch.h" #include "ardour/event_type_map.h" #include "ardour/session.h" @@ -28,43 +29,113 @@ using namespace std; using namespace ARDOUR; using namespace PBD; - -AutomationControl::AutomationControl( - ARDOUR::Session& session, - const Evoral::Parameter& parameter, - boost::shared_ptr list, - const string& name) - : Controllable((name != "") ? name : EventTypeMap::instance().to_symbol(parameter)) - , Evoral::Control(parameter, list) +AutomationControl::AutomationControl(ARDOUR::Session& session, + const Evoral::Parameter& parameter, + const ParameterDescriptor& desc, + boost::shared_ptr list, + const string& name) + : Controllable (name.empty() ? EventTypeMap::instance().to_symbol(parameter) : name) + , Evoral::Control(parameter, desc, list) , _session(session) + , _desc(desc) { } +AutomationControl::~AutomationControl () +{ +} -float +/** Get the current effective `user' value based on automation state */ +double AutomationControl::get_value() const { bool from_list = _list && ((AutomationList*)_list.get())->automation_playback(); - return Control::get_float(from_list, _session.transport_frame()); + return Control::get_double (from_list, _session.transport_frame()); } - +/** Set the value and do the right thing based on automation state + * (e.g. record if necessary, etc.) + * @param value `user' value + */ void -AutomationControl::set_value(float value) +AutomationControl::set_value (double value) { - bool to_list = _list && _session.transport_stopped() - && ((AutomationList*)_list.get())->automation_write(); - - Control::set_float(value, to_list, _session.transport_frame()); + bool to_list = _list && ((AutomationList*)_list.get())->automation_write(); + + Control::set_double (value, _session.transport_frame(), to_list); Changed(); /* EMIT SIGNAL */ } - void -AutomationControl::set_list(boost::shared_ptr list) +AutomationControl::set_list (boost::shared_ptr list) { - Control::set_list(list); + Control::set_list (list); Changed(); /* EMIT SIGNAL */ } +void +AutomationControl::set_automation_state (AutoState as) +{ + if (_list && as != alist()->automation_state()) { + + alist()->set_automation_state (as); + if (_desc.toggled) { + return; // No watch for boolean automation + } + + if (as == Write) { + AutomationWatch::instance().add_automation_watch (shared_from_this()); + } else if (as == Touch) { + if (!touching()) { + AutomationWatch::instance().remove_automation_watch (shared_from_this()); + } else { + /* this seems unlikely, but the combination of + * a control surface and the mouse could make + * it possible to put the control into Touch + * mode *while* touching it. + */ + AutomationWatch::instance().add_automation_watch (shared_from_this()); + } + } else { + AutomationWatch::instance().remove_automation_watch (shared_from_this()); + } + } +} + +void +AutomationControl::set_automation_style (AutoStyle as) +{ + if (!_list) return; + alist()->set_automation_style (as); +} + +void +AutomationControl::start_touch(double when) +{ + if (!_list) return; + if (!touching()) { + if (alist()->automation_state() == Touch) { + alist()->start_touch (when); + if (!_desc.toggled) { + AutomationWatch::instance().add_automation_watch (shared_from_this()); + } + } + set_touching (true); + } +} + +void +AutomationControl::stop_touch(bool mark, double when) +{ + if (!_list) return; + if (touching()) { + set_touching (false); + if (alist()->automation_state() == Touch) { + alist()->stop_touch (mark, when); + if (!_desc.toggled) { + AutomationWatch::instance().remove_automation_watch (shared_from_this()); + } + } + } +}