/* Copyright (C) 2007 Paul Davis Author: Dave 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 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., 675 Mass Ave, Cambridge, MA 02139, USA. */ #include #include "pbd/error.h" #include "ardour/automation_list.h" #include "ardour/automation_control.h" #include "ardour/event_type_map.h" #include "ardour/automatable.h" #include "ardour/panner.h" #include "ardour_ui.h" #include "utils.h" #include "automation_controller.h" #include "gui_thread.h" #include "i18n.h" using namespace ARDOUR; using namespace Gtk; AutomationController::AutomationController(boost::shared_ptr ac, Adjustment* adj) : BarController (*adj, ac) , _ignore_change(false) , _controllable(ac) , _adjustment(adj) { set_name (X_("PluginSlider")); // FIXME: get yer own name! set_style (BarController::LeftToRight); set_use_parent (true); StartGesture.connect (sigc::mem_fun(*this, &AutomationController::start_touch)); StopGesture.connect (sigc::mem_fun(*this, &AutomationController::end_touch)); _adjustment->signal_value_changed().connect ( sigc::mem_fun(*this, &AutomationController::value_adjusted)); _screen_update_connection = ARDOUR_UI::RapidScreenUpdate.connect ( sigc::mem_fun (*this, &AutomationController::display_effective_value)); ac->Changed.connect (_changed_connection, invalidator (*this), boost::bind (&AutomationController::value_changed, this), gui_context()); } AutomationController::~AutomationController() { } boost::shared_ptr AutomationController::create( boost::shared_ptr parent, const Evoral::Parameter& param, boost::shared_ptr ac) { Gtk::Adjustment* adjustment = manage(new Gtk::Adjustment(param.normal(), param.min(), param.max())); if (!ac) { PBD::warning << "Creating AutomationController for " << EventTypeMap::instance().to_symbol(param) << endmsg; ac = boost::dynamic_pointer_cast(parent->control_factory(param)); } else { assert(ac->parameter() == param); } return boost::shared_ptr(new AutomationController(ac, adjustment)); } std::string AutomationController::get_label (int&) { std::stringstream s; // Hack to display CC rounded to int if (_controllable->parameter().type() == MidiCCAutomation) { s << (int)_controllable->get_value(); } else if (_controllable->parameter().type() == PanAutomation) { s << Panner::value_as_string (_controllable->get_value ()); } else { s << std::fixed << std::setprecision(3) << _controllable->get_value(); } return s.str (); } void AutomationController::display_effective_value() { //if ( ! _controllable->list()->automation_playback()) // return; float value = _controllable->get_value(); if (_adjustment->get_value() != value) { _ignore_change = true; _adjustment->set_value (value); _ignore_change = false; } } void AutomationController::value_adjusted() { if (!_ignore_change) { _controllable->set_value(_adjustment->get_value()); } } void AutomationController::start_touch() { _controllable->start_touch(); } void AutomationController::end_touch() { _controllable->stop_touch(); } void AutomationController::automation_state_changed () { ENSURE_GUI_THREAD (*this, &AutomationController::automation_state_changed) bool x = (_controllable->automation_state() != Off); /* start watching automation so that things move */ _screen_update_connection.disconnect(); if (x) { _screen_update_connection = ARDOUR_UI::RapidScreenUpdate.connect ( sigc::mem_fun (*this, &AutomationController::display_effective_value)); } } void AutomationController::value_changed () { Gtkmm2ext::UI::instance()->call_slot (invalidator (*this), boost::bind (&AutomationController::display_effective_value, this)); } /** Stop updating our value from our controllable */ void AutomationController::stop_updating () { _screen_update_connection.disconnect (); }