13
0
livetrax/gtk2_ardour/main_clock.cc

240 lines
7.8 KiB
C++

/*
* Copyright (C) 2012-2017 Robin Gareus <robin@gareus.org>
* Copyright (C) 2013-2017 Paul Davis <paul@linuxaudiosystems.com>
* Copyright (C) 2015 Colin Fletcher <colin.m.fletcher@googlemail.com>
* Copyright (C) 2015 Tim Mayberry <mojofunk@gmail.com>
* Copyright (C) 2016 Nick Mainsbridge <mainsbridge@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/unwind.h"
#include "ardour/audioengine.h"
#include "ardour/session.h"
#include "ardour/tempo.h"
#include "actions.h"
#include "main_clock.h"
#include "public_editor.h"
#include "ardour_ui.h"
#include "pbd/i18n.h"
using namespace Gtk;
using namespace ARDOUR;
using namespace std;
MainClock::MainClock (
const std::string& clock_name,
const std::string& widget_name,
ClockDisposition d
)
: AudioClock (clock_name, false, widget_name, true, true, false, true)
, _disposition(d)
, _suspend_delta_mode_signal (false)
, _widget_name(widget_name)
{
ValueChanged.connect (sigc::mem_fun(*this, &MainClock::clock_value_changed));
UIConfiguration::instance().ParameterChanged.connect (sigc::mem_fun (*this, &MainClock::parameter_changed));
std::function<void (std::string)> pc (std::bind (&MainClock::parameter_changed, this, _1));
UIConfiguration::instance().map_parameters (pc);
}
void
MainClock::set_session (ARDOUR::Session *s)
{
AudioClock::set_session (s);
_left_btn.set_related_action (ActionManager::get_action (X_("Editor"), X_("edit-current-tempo")));
_right_btn.set_related_action (ActionManager::get_action (X_("Editor"), X_("edit-current-meter")));
}
void
MainClock::parameter_changed (std::string p)
{
if (p == "primary-clock-delta-mode" && _disposition == PrimaryClock) {
set_display_delta_mode(UIConfiguration::instance().get_primary_clock_delta_mode());
} else if (p == "secondary-clock-delta-mode" && _disposition == SecondaryClock) {
set_display_delta_mode(UIConfiguration::instance().get_secondary_clock_delta_mode());
}
}
void
MainClock::clock_value_changed ()
{
if (_session) {
_session->request_locate (last_when ().samples());
}
}
void
MainClock::build_ops_menu ()
{
using namespace Menu_Helpers;
AudioClock::build_ops_menu ();
MenuList& ops_items = ops_menu->items();
ops_items.push_back (SeparatorElem ());
RadioMenuItem::Group group;
PBD::Unwinder<bool> uw (_suspend_delta_mode_signal, true);
ops_items.push_back (RadioMenuElem (group, _("Display absolute time"), sigc::bind (sigc::mem_fun (*this, &MainClock::change_display_delta_mode), NoDelta)));
if (_delta_mode == NoDelta) {
RadioMenuItem* i = dynamic_cast<RadioMenuItem *> (&ops_items.back ());
i->set_active (true);
}
ops_items.push_back (RadioMenuElem (group, _("Display delta to edit cursor"), sigc::bind (sigc::mem_fun (*this, &MainClock::change_display_delta_mode), DeltaEditPoint)));
if (_delta_mode == DeltaEditPoint) {
RadioMenuItem* i = dynamic_cast<RadioMenuItem *> (&ops_items.back ());
i->set_active (true);
}
ops_items.push_back (RadioMenuElem (group, _("Display delta to origin marker"), sigc::bind (sigc::mem_fun (*this, &MainClock::change_display_delta_mode), DeltaOriginMarker)));
if (_delta_mode == DeltaOriginMarker) {
RadioMenuItem* i = dynamic_cast<RadioMenuItem *> (&ops_items.back ());
i->set_active (true);
}
ops_items.push_back (SeparatorElem());
ops_items.push_back (MenuElem (_("Edit Tempo"), sigc::mem_fun(*this, &MainClock::edit_current_tempo)));
ops_items.push_back (MenuElem (_("Edit Time Signature"), sigc::mem_fun(*this, &MainClock::edit_current_meter)));
ops_items.push_back (MenuElem (_("Insert Tempo Change"), sigc::mem_fun(*this, &MainClock::insert_new_tempo)));
ops_items.push_back (MenuElem (_("Insert Time Signature Change"), sigc::mem_fun(*this, &MainClock::insert_new_meter)));
}
void
MainClock::set (timepos_t const & when, bool force)
{
if (!AudioEngine::instance()->session()) {
_delta_mode = NoDelta;
}
switch (_delta_mode) {
case NoDelta:
AudioClock::set (when, force);
break;
case DeltaEditPoint:
set_duration (when.distance (PublicEditor::instance().get_preferred_edit_position (Editing::EDIT_IGNORE_PHEAD)), force);
break;
case DeltaOriginMarker:
{
Location* loc = AudioEngine::instance()->session()->locations()->clock_origin_location ();
set_duration (-when.distance (loc ? loc->start() : timepos_t (when.time_domain())), force);
}
break;
}
CanonicalClockChanged (); //signal
}
void
MainClock::change_display_delta_mode (ClockDeltaMode m)
{
if (_suspend_delta_mode_signal) {
return;
}
change_display_delta_mode_signal (m);
}
void
MainClock::set_display_delta_mode (ClockDeltaMode m)
{
_delta_mode = m;
if (_delta_mode != NoDelta) {
set_editable (false);
set_widget_name (_widget_name + " delta");
} else {
set_editable (true);
set_widget_name (_widget_name);
}
}
void
MainClock::edit_current_tempo ()
{
if (!PublicEditor::instance().session()) return;
PublicEditor::instance().edit_tempo_section (Temporal::TempoMap::use()->metric_at (last_when()).get_editable_tempo());
}
void
MainClock::edit_current_meter ()
{
if (!PublicEditor::instance().session()) return;
PublicEditor::instance().edit_meter_section (Temporal::TempoMap::use()->metric_at (last_when()).get_editable_meter());
}
void
MainClock::insert_new_tempo ()
{
PublicEditor::instance().mouse_add_new_tempo_event (last_when ());
}
void
MainClock::insert_new_meter ()
{
PublicEditor::instance().mouse_add_new_meter_event (last_when ());
}
TransportClock::TransportClock (
const std::string& clock_name,
const std::string& widget_name,
ClockDisposition d
)
: MainClock (clock_name, widget_name, d)
{
if (_disposition==PrimaryClock) {
ARDOUR_UI::instance()->primary_clock->CanonicalClockChanged.connect (sigc::mem_fun(*this, &TransportClock::follow_canonical_clock));
ARDOUR_UI::instance()->primary_clock->mode_changed.connect (sigc::mem_fun(*this, &TransportClock::follow_canonical_clock));
change_display_delta_mode_signal.connect (sigc::mem_fun(UIConfiguration::instance(), &UIConfiguration::set_primary_clock_delta_mode));
} else {
ARDOUR_UI::instance()->secondary_clock->CanonicalClockChanged.connect (sigc::mem_fun(*this, &TransportClock::follow_canonical_clock));
ARDOUR_UI::instance()->secondary_clock->mode_changed.connect (sigc::mem_fun(*this, &TransportClock::follow_canonical_clock));
change_display_delta_mode_signal.connect (sigc::mem_fun(UIConfiguration::instance(), &UIConfiguration::set_secondary_clock_delta_mode));
}
}
void
TransportClock::set_mode(Mode m)
{
//we don't set the mode for ourselves; instead we set the canonical clock and then 'follow' it
if (_disposition==PrimaryClock) {
ARDOUR_UI::instance()->primary_clock->set_mode(m);
} else {
ARDOUR_UI::instance()->secondary_clock->set_mode(m);
}
}
void
TransportClock::follow_canonical_clock ()
{
if (!_session) {
return;
}
set(timepos_t(_session->audible_sample()));
if (_disposition==PrimaryClock) {
AudioClock::set_mode(ARDOUR_UI::instance()->primary_clock->mode());
set_display_delta_mode(ARDOUR_UI::instance()->primary_clock->display_delta_mode());
} else {
AudioClock::set_mode(ARDOUR_UI::instance()->secondary_clock->mode());
set_display_delta_mode(ARDOUR_UI::instance()->secondary_clock->display_delta_mode());
}
}