13
0
livetrax/gtk2_ardour/timectl_gui.cc

228 lines
6.0 KiB
C++

/*
* Copyright (C) 2007-2017 Paul Davis <paul@linuxaudiosystems.com>
* Copyright (C) 2008-2011 David Robillard <d@drobilla.net>
* Copyright (C) 2009-2011 Carl Hetherington <carl@carlh.net>
* Copyright (C) 2013-2019 Robin Gareus <robin@gareus.org>
*
* 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 <inttypes.h>
#include <iomanip>
#include <gtkmm/stock.h>
#include "pbd/convert.h"
#include "pbd/error.h"
#include "pbd/unwind.h"
#include "ardour/latent.h"
#include "ardour/rc_configuration.h"
#include "ardour/tailtime.h"
#include "gtkmm2ext/utils.h"
#include "timectl_gui.h"
#include "utils.h"
#include "pbd/i18n.h"
using namespace PBD;
using namespace Gtk;
using namespace Gtkmm2ext;
using namespace ARDOUR;
static const gchar *_unit_strings[] = {
N_("sample"),
N_("msec"),
N_("period"),
0
};
std::vector<std::string> TimeCtlGUI::unit_strings;
std::string
TimeCtlBarController::get_label (double&)
{
return ARDOUR_UI_UTILS::samples_as_time_string (
_timectl_gui->adjustment.get_value(), _timectl_gui->sample_rate, true);
}
void
TimeCtlGUIControllable::set_value (double v, PBD::Controllable::GroupControlDisposition group_override)
{
_timectl_gui->adjustment.set_value (v);
}
double
TimeCtlGUIControllable::get_value () const
{
return _timectl_gui->adjustment.get_value ();
}
double
TimeCtlGUIControllable::lower() const
{
return _timectl_gui->adjustment.get_lower ();
}
double
TimeCtlGUIControllable::upper() const
{
return _timectl_gui->adjustment.get_upper ();
}
TimeCtlGUI::TimeCtlGUI (Latent& l, samplepos_t sr, samplepos_t psz)
: _latent (&l)
, _tailtime (0)
, sample_rate (sr)
, period_size (psz)
, _ignore_change (false)
, adjustment (0, 0.0, sample_rate, 1.0, sample_rate / 1000.0f) /* max 1 second, step by samples, page by msecs */
, bc (adjustment, this)
, reset_button (_("Reset"))
{
init ();
}
TimeCtlGUI::TimeCtlGUI (TailTime& t, samplepos_t sr, samplepos_t psz)
: _latent (0)
, _tailtime (&t)
, sample_rate (sr)
, period_size (psz)
, _ignore_change (false)
, adjustment (0, 0.0, 20 * sample_rate, sample_rate / 1000.f, 1.0, sample_rate / 2.0f) /* max 20 second, step by msec, page by 0.5 sec */
, bc (adjustment, this)
, reset_button (_("Reset"))
{
init ();
}
void
TimeCtlGUI::init ()
{
Widget* w;
if (unit_strings.empty()) {
unit_strings = I18N (_unit_strings);
}
set_popdown_strings (units_combo, unit_strings);
units_combo.set_active_text (unit_strings.front());
w = manage (new Image (Stock::ADD, ICON_SIZE_BUTTON));
w->show ();
plus_button.add (*w);
w = manage (new Image (Stock::REMOVE, ICON_SIZE_BUTTON));
w->show ();
minus_button.add (*w);
hbox1.pack_start (bc, true, true);
hbox2.set_homogeneous (false);
hbox2.set_spacing (12);
hbox2.pack_start (reset_button);
hbox2.pack_start (minus_button);
hbox2.pack_start (plus_button);
hbox2.pack_start (units_combo, true, true);
minus_button.signal_clicked().connect (sigc::bind (sigc::mem_fun (*this, &TimeCtlGUI::change_from_button), -1));
plus_button.signal_clicked().connect (sigc::bind (sigc::mem_fun (*this, &TimeCtlGUI::change_from_button), 1));
reset_button.signal_clicked().connect (sigc::mem_fun (*this, &TimeCtlGUI::reset));
/* Limit value to adjustment range (max = sample_rate).
* Otherwise if the signal_latency() is larger than the adjustment's max,
* TimeCtlGUI::finish() would set the adjustment's max value as custom-latency.
*/
if (_latent) {
adjustment.set_value (std::min (sample_rate, _latent->signal_latency ()));
} else if (_tailtime) {
adjustment.set_value (std::min (sample_rate, _tailtime->signal_tailtime ()));
}
adjustment.signal_value_changed().connect (sigc::mem_fun (*this, &TimeCtlGUI::finish));
bc.set_size_request (-1, 25);
bc.set_name (X_("ProcessorControlSlider"));
set_spacing (12);
pack_start (hbox1, true, true);
pack_start (hbox2, true, true);
}
void
TimeCtlGUI::finish ()
{
if (_ignore_change) {
return;
}
samplepos_t new_value = (samplepos_t) adjustment.get_value();
if (_latent) {
_latent->set_user_latency (new_value);
} else if (_tailtime) {
_tailtime->set_user_tailtime (new_value);
}
}
void
TimeCtlGUI::reset ()
{
if (_latent) {
_latent->unset_user_latency ();
PBD::Unwinder<bool> uw (_ignore_change, true);
adjustment.set_value (std::min (sample_rate, _latent->signal_latency ()));
} else if (_tailtime) {
_tailtime->unset_user_tailtime ();
PBD::Unwinder<bool> uw (_ignore_change, true);
adjustment.set_value (std::min<samplecnt_t> (Config->get_max_tail_samples (), _tailtime->signal_tailtime ()));
}
}
void
TimeCtlGUI::refresh ()
{
PBD::Unwinder<bool> uw (_ignore_change, true);
if (_latent) {
adjustment.set_value (std::min (sample_rate, _latent->effective_latency ()));
} else if (_tailtime) {
adjustment.set_value (std::min<samplecnt_t> (Config->get_max_tail_samples (),_tailtime->effective_tailtime ()));
}
}
void
TimeCtlGUI::change_from_button (int dir)
{
std::string unitstr = units_combo.get_active_text();
double shift = 0.0;
if (unitstr == unit_strings[0]) {
shift = 1;
} else if (unitstr == unit_strings[1]) {
shift = (sample_rate / 1000.0);
} else if (unitstr == unit_strings[2]) {
shift = period_size;
} else {
fatal << string_compose (_("programming error: %1 (%2)"), X_("illegal string in latency GUI units combo"), unitstr)
<< endmsg;
abort(); /*NOTREACHED*/
}
if (dir > 0) {
adjustment.set_value (adjustment.get_value() + shift);
} else {
adjustment.set_value (adjustment.get_value() - shift);
}
}