487 lines
13 KiB
C++
487 lines
13 KiB
C++
/*
|
|
* Copyright (C) 2005-2019 Paul Davis <paul@linuxaudiosystems.com>
|
|
* Copyright (C) 2005 Taybin Rutkin <taybin@taybin.com>
|
|
* Copyright (C) 2006-2015 Tim Mayberry <mojofunk@gmail.com>
|
|
* Copyright (C) 2007-2011 David Robillard <d@drobilla.net>
|
|
* Copyright (C) 2009-2012 Carl Hetherington <carl@carlh.net>
|
|
* Copyright (C) 2013-2018 Robin Gareus <robin@gareus.org>
|
|
* Copyright (C) 2016-2018 Ben Loftis <ben@harrisonconsoles.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.
|
|
*/
|
|
|
|
#ifdef WAF_BUILD
|
|
#include "gtk2ardour-config.h"
|
|
#endif
|
|
|
|
/* this file exists solely to break compilation dependencies that
|
|
would connect changes to the mixer or editor objects.
|
|
*/
|
|
|
|
#include <cstdio>
|
|
|
|
#include "pbd/error.h"
|
|
|
|
#include "ardour/session.h"
|
|
#include "ardour/lv2_plugin.h"
|
|
|
|
#include "gtkmm2ext/bindings.h"
|
|
|
|
#include "actions.h"
|
|
#include "ardour_message.h"
|
|
#include "ardour_ui.h"
|
|
#include "audio_clip_editor.h"
|
|
#include "public_editor.h"
|
|
#include "meterbridge.h"
|
|
#include "luainstance.h"
|
|
#include "luawindow.h"
|
|
#include "mixer_ui.h"
|
|
#include "recorder_ui.h"
|
|
#include "trigger_page.h"
|
|
#include "keyboard.h"
|
|
#include "keyeditor.h"
|
|
#include "rc_option_editor.h"
|
|
#include "route_params_ui.h"
|
|
#include "time_info_box.h"
|
|
#include "trigger_ui.h"
|
|
#include "step_entry.h"
|
|
#include "opts.h"
|
|
|
|
#ifdef GDK_WINDOWING_X11
|
|
#include <gdk/gdkx.h>
|
|
#endif
|
|
|
|
#include "pbd/i18n.h"
|
|
|
|
using namespace Gtk;
|
|
using namespace PBD;
|
|
|
|
namespace ARDOUR {
|
|
class Session;
|
|
class Route;
|
|
}
|
|
|
|
using namespace ARDOUR;
|
|
using namespace Gtkmm2ext;
|
|
|
|
void
|
|
ARDOUR_UI::we_have_dependents ()
|
|
{
|
|
install_dependent_actions ();
|
|
|
|
/* The monitor section relies on at least 1 action defined by us. Since that
|
|
* action now exists, give it a chance to use it.
|
|
*/
|
|
mixer->monitor_section().use_others_actions ();
|
|
|
|
/* Create "static" actions that apply to all ProcessorBoxes
|
|
*/
|
|
ProcessorBox::register_actions ();
|
|
StepEntry::setup_actions_and_bindings ();
|
|
ClipEditorBox::init ();
|
|
|
|
/* Global, editor, mixer, processor box actions are defined now. Link
|
|
them with any bindings, so that GTK does not get a chance to define
|
|
the GTK accel map entries first when we ask the GtkUIManager to
|
|
create menus/widgets.
|
|
|
|
If GTK adds the actions to its accel map before we do, we lose our
|
|
freedom to use any keys. More precisely, we can use any keys, but
|
|
ones that GTK considers illegal as accelerators will not show up in
|
|
menus.
|
|
|
|
There are other dynamic actions that can be created by a monitor
|
|
section, by step entry dialogs. These need to be handled
|
|
separately. They don't tend to use GTK-illegal bindings and more
|
|
importantly they don't have menus showing the bindings, so it is
|
|
less of an issue.
|
|
*/
|
|
|
|
Gtkmm2ext::Bindings::associate_all ();
|
|
|
|
editor->UpdateAllTransportClocks.connect (sigc::mem_fun (*this, &ARDOUR_UI::update_transport_clocks));
|
|
|
|
/* all actions are defined */
|
|
|
|
ActionManager::load_menus (ARDOUR_COMMAND_LINE::menus_file);
|
|
|
|
/* catch up on parameters */
|
|
|
|
boost::function<void (std::string)> pc (boost::bind (&ARDOUR_UI::parameter_changed, this, _1));
|
|
Config->map_parameters (pc);
|
|
|
|
UIConfiguration::instance().reset_dpi ();
|
|
}
|
|
|
|
void
|
|
ARDOUR_UI::connect_dependents_to_session (ARDOUR::Session *s)
|
|
{
|
|
DisplaySuspender ds;
|
|
BootMessage (_("Setup Editor"));
|
|
editor->set_session (s);
|
|
BootMessage (_("Setup Mixer"));
|
|
mixer->set_session (s);
|
|
recorder->set_session (s);
|
|
trigger_page->set_session (s);
|
|
meterbridge->set_session (s);
|
|
|
|
/* its safe to do this now */
|
|
|
|
BootMessage (_("Reload Session History"));
|
|
s->restore_history ("");
|
|
}
|
|
|
|
/** The main editor window has been closed */
|
|
gint
|
|
ARDOUR_UI::exit_on_main_window_close (GdkEventAny * /*ev*/)
|
|
{
|
|
#ifdef __APPLE__
|
|
/* just hide the window, and return - the top menu stays up */
|
|
editor->hide ();
|
|
return TRUE;
|
|
#else
|
|
/* time to get out of here */
|
|
finish();
|
|
return TRUE;
|
|
#endif
|
|
}
|
|
|
|
GtkNotebook*
|
|
ARDOUR_UI::tab_window_root_drop (GtkNotebook* src,
|
|
GtkWidget* w,
|
|
gint x,
|
|
gint y,
|
|
gpointer)
|
|
{
|
|
using namespace std;
|
|
Gtk::Notebook* nb = 0;
|
|
Gtk::Window* win = 0;
|
|
ArdourWidgets::Tabbable* tabbable = 0;
|
|
|
|
|
|
if (w == GTK_WIDGET(editor->contents().gobj())) {
|
|
tabbable = editor;
|
|
} else if (w == GTK_WIDGET(mixer->contents().gobj())) {
|
|
tabbable = mixer;
|
|
} else if (w == GTK_WIDGET(rc_option_editor->contents().gobj())) {
|
|
tabbable = rc_option_editor;
|
|
} else if (w == GTK_WIDGET(recorder->contents().gobj())) {
|
|
tabbable = recorder;
|
|
} else if (w == GTK_WIDGET(trigger_page->contents().gobj())) {
|
|
tabbable = trigger_page;
|
|
} else {
|
|
return 0;
|
|
}
|
|
|
|
nb = tabbable->tab_root_drop ();
|
|
win = tabbable->own_window ();
|
|
|
|
if (nb) {
|
|
win->move (x, y);
|
|
win->show_all ();
|
|
win->present ();
|
|
return nb->gobj();
|
|
}
|
|
|
|
return 0; /* what was that? */
|
|
}
|
|
|
|
bool
|
|
ARDOUR_UI::idle_ask_about_quit ()
|
|
{
|
|
const auto ask_before_closing = UIConfiguration::instance ().get_ask_before_closing_last_window ();
|
|
|
|
if ((_session && _session->dirty ()) || !ask_before_closing) {
|
|
finish ();
|
|
} else {
|
|
/* no session or session not dirty, but still ask anyway */
|
|
|
|
ArdourMessageDialog msg (string_compose (_("Quit %1?"), PROGRAM_NAME),
|
|
false, /* no markup */
|
|
Gtk::MESSAGE_INFO,
|
|
Gtk::BUTTONS_YES_NO,
|
|
true); /* modal */
|
|
msg.set_default_response (Gtk::RESPONSE_YES);
|
|
|
|
if (msg.run () == Gtk::RESPONSE_YES) {
|
|
finish ();
|
|
}
|
|
}
|
|
|
|
/* not reached but keep the compiler happy */
|
|
|
|
return false;
|
|
}
|
|
|
|
bool
|
|
ARDOUR_UI::main_window_delete_event (GdkEventAny* ev)
|
|
{
|
|
/* quit the application as soon as we go idle. If we call this here,
|
|
* the window manager/desktop can think we're taking too longer to
|
|
* handle the "delete" event
|
|
*/
|
|
|
|
Glib::signal_idle().connect (sigc::mem_fun (*this, &ARDOUR_UI::idle_ask_about_quit));
|
|
|
|
return true;
|
|
}
|
|
|
|
static GtkNotebook*
|
|
tab_window_root_drop (GtkNotebook* src,
|
|
GtkWidget* w,
|
|
gint x,
|
|
gint y,
|
|
gpointer user_data)
|
|
{
|
|
return ARDOUR_UI::instance()->tab_window_root_drop (src, w, x, y, user_data);
|
|
}
|
|
|
|
int
|
|
ARDOUR_UI::setup_windows ()
|
|
{
|
|
_tabs.set_show_border(false);
|
|
_tabs.signal_switch_page().connect (sigc::mem_fun (*this, &ARDOUR_UI::tabs_switch));
|
|
_tabs.signal_page_added().connect (sigc::mem_fun (*this, &ARDOUR_UI::tabs_page_added));
|
|
_tabs.signal_page_removed().connect (sigc::mem_fun (*this, &ARDOUR_UI::tabs_page_removed));
|
|
|
|
rc_option_editor = new RCOptionEditor;
|
|
rc_option_editor->StateChange.connect (sigc::mem_fun (*this, &ARDOUR_UI::tabbable_state_change));
|
|
|
|
if (create_editor ()) {
|
|
error << _("UI: cannot setup editor") << endmsg;
|
|
return -1;
|
|
}
|
|
|
|
if (create_mixer ()) {
|
|
error << _("UI: cannot setup mixer") << endmsg;
|
|
return -1;
|
|
}
|
|
|
|
if (create_recorder ()) {
|
|
error << _("UI: cannot setup recorder") << endmsg;
|
|
return -1;
|
|
}
|
|
|
|
if (create_trigger_page ()) {
|
|
error << _("UI: cannot setup trigger") << endmsg;
|
|
return -1;
|
|
}
|
|
|
|
if (create_meterbridge ()) {
|
|
error << _("UI: cannot setup meterbridge") << endmsg;
|
|
return -1;
|
|
}
|
|
|
|
time_info_box = new TimeInfoBox ("ToolbarTimeInfo", false);
|
|
/* all other dialogs are created conditionally */
|
|
|
|
we_have_dependents ();
|
|
|
|
/* order of addition affects order seen in initial window display */
|
|
|
|
rc_option_editor->add_to_notebook (_tabs);
|
|
mixer->add_to_notebook (_tabs);
|
|
editor->add_to_notebook (_tabs);
|
|
recorder->add_to_notebook (_tabs);
|
|
trigger_page->add_to_notebook (_tabs);
|
|
|
|
top_packer.pack_start (menu_bar_base, false, false);
|
|
|
|
main_vpacker.pack_start (top_packer, false, false);
|
|
|
|
ArdourWidgets::ArdourDropShadow *spacer = manage (new (ArdourWidgets::ArdourDropShadow));
|
|
spacer->set_size_request( -1, 4 );
|
|
spacer->show();
|
|
|
|
/* now add the transport sample to the top of main window */
|
|
|
|
main_vpacker.pack_start ( *spacer, false, false);
|
|
main_vpacker.pack_start (transport_frame, false, false);
|
|
main_vpacker.pack_start (_tabs, true, true);
|
|
|
|
LuaInstance::instance()->ActionChanged.connect (sigc::mem_fun (*this, &ARDOUR_UI::action_script_changed));
|
|
|
|
for (int i = 0; i < MAX_LUA_ACTION_BUTTONS; ++i) {
|
|
std::string const a = string_compose (X_("script-%1"), i + 1);
|
|
Glib::RefPtr<Action> act = ActionManager::get_action(X_("LuaAction"), a.c_str());
|
|
assert (act);
|
|
action_script_call_btn[i].set_name ("lua action button");
|
|
action_script_call_btn[i].set_text (string_compose ("%1%2", std::hex, i+1));
|
|
action_script_call_btn[i].set_related_action (act);
|
|
action_script_call_btn[i].signal_button_press_event().connect (sigc::bind (sigc::mem_fun(*this, &ARDOUR_UI::bind_lua_action_script), i), false);
|
|
if (act->get_sensitive ()) {
|
|
action_script_call_btn[i].set_visual_state (Gtkmm2ext::VisualState (action_script_call_btn[i].visual_state() & ~Gtkmm2ext::Insensitive));
|
|
} else {
|
|
action_script_call_btn[i].set_visual_state (Gtkmm2ext::VisualState (action_script_call_btn[i].visual_state() | Gtkmm2ext::Insensitive));
|
|
}
|
|
action_script_call_btn[i].set_sizing_text ("88");
|
|
action_script_call_btn[i].set_no_show_all ();
|
|
}
|
|
|
|
setup_transport();
|
|
build_menu_bar ();
|
|
setup_tooltips ();
|
|
|
|
_main_window.signal_delete_event().connect (sigc::mem_fun (*this, &ARDOUR_UI::main_window_delete_event));
|
|
|
|
/* pack the main vpacker into the main window and show everything
|
|
*/
|
|
|
|
_main_window.add (main_vpacker);
|
|
transport_frame.show_all ();
|
|
|
|
apply_window_settings (true);
|
|
|
|
setup_toplevel_window (_main_window, "", this);
|
|
_main_window.show_all ();
|
|
|
|
_tabs.set_show_tabs (false);
|
|
|
|
/* It would be nice if Gtkmm had wrapped this rather than just
|
|
* deprecating the old set_window_creation_hook() method, but oh well...
|
|
*/
|
|
g_signal_connect (_tabs.gobj(), "create-window", (GCallback) ::tab_window_root_drop, this);
|
|
|
|
#ifdef GDK_WINDOWING_X11
|
|
/* allow externalUIs to be transient, on top of the main window */
|
|
LV2Plugin::set_main_window_id (GDK_DRAWABLE_XID(_main_window.get_window()->gobj()));
|
|
#endif
|
|
|
|
return 0;
|
|
}
|
|
|
|
void
|
|
ARDOUR_UI::apply_window_settings (bool with_size)
|
|
{
|
|
const XMLNode* mnode = main_window_settings ();
|
|
|
|
if (!mnode) {
|
|
return;
|
|
}
|
|
|
|
XMLProperty const* prop;
|
|
|
|
if (with_size) {
|
|
gint x = -1;
|
|
gint y = -1;
|
|
gint w = -1;
|
|
gint h = -1;
|
|
|
|
if ((prop = mnode->property (X_("x"))) != 0) {
|
|
x = atoi (prop->value());
|
|
}
|
|
|
|
if ((prop = mnode->property (X_("y"))) != 0) {
|
|
y = atoi (prop->value());
|
|
}
|
|
|
|
if ((prop = mnode->property (X_("w"))) != 0) {
|
|
w = atoi (prop->value());
|
|
}
|
|
|
|
if ((prop = mnode->property (X_("h"))) != 0) {
|
|
h = atoi (prop->value());
|
|
}
|
|
|
|
if (x >= 0 && y >= 0 && w >= 0 && h >= 0) {
|
|
_main_window.set_position (Gtk::WIN_POS_NONE);
|
|
}
|
|
|
|
if (x >= 0 && y >= 0) {
|
|
_main_window.move (x, y);
|
|
}
|
|
|
|
if (w > 0 && h > 0) {
|
|
_main_window.set_default_size (w, h);
|
|
}
|
|
}
|
|
|
|
std::string current_tab;
|
|
|
|
if ((prop = mnode->property (X_("current-tab"))) != 0) {
|
|
current_tab = prop->value();
|
|
} else {
|
|
current_tab = "editor";
|
|
}
|
|
|
|
if (mixer && current_tab == "mixer") {
|
|
_tabs.set_current_page (_tabs.page_num (mixer->contents()));
|
|
} else if (rc_option_editor && current_tab == "preferences") {
|
|
_tabs.set_current_page (_tabs.page_num (rc_option_editor->contents()));
|
|
} else if (recorder && current_tab == "recorder") {
|
|
_tabs.set_current_page (_tabs.page_num (recorder->contents()));
|
|
} else if (trigger_page && current_tab == "trigger") {
|
|
_tabs.set_current_page (_tabs.page_num (trigger_page->contents()));
|
|
} else if (editor) {
|
|
_tabs.set_current_page (_tabs.page_num (editor->contents()));
|
|
}
|
|
return;
|
|
}
|
|
|
|
bool
|
|
ARDOUR_UI::bind_lua_action_script (GdkEventButton*ev, int i)
|
|
{
|
|
if (!_session) {
|
|
return false;
|
|
}
|
|
LuaInstance *li = LuaInstance::instance();
|
|
std::string name;
|
|
if (ev->button != 3 && !(ev->button == 1 && !li->lua_action_name (i, name))) {
|
|
return false;
|
|
}
|
|
if (Gtkmm2ext::Keyboard::modifier_state_equals (ev->state, Gtkmm2ext::Keyboard::TertiaryModifier)) {
|
|
li->remove_lua_action (i);
|
|
} else {
|
|
li->interactive_add (LuaScriptInfo::EditorAction, i);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
void
|
|
ARDOUR_UI::action_script_changed (int i, const std::string& n)
|
|
{
|
|
if (i < 0 || i >= MAX_LUA_ACTION_SCRIPTS) {
|
|
return;
|
|
}
|
|
|
|
if (i < MAX_LUA_ACTION_BUTTONS) {
|
|
if (LuaInstance::instance()->lua_action_has_icon (i)) {
|
|
uintptr_t ii = i;
|
|
action_script_call_btn[i].set_icon (&LuaInstance::render_action_icon, (void*)ii);
|
|
} else {
|
|
action_script_call_btn[i].set_icon (0, 0);
|
|
}
|
|
if (n.empty ()) {
|
|
action_script_call_btn[i].set_text (string_compose ("%1%2", std::hex, i+1));
|
|
} else {
|
|
action_script_call_btn[i].set_text (n.substr(0,1));
|
|
}
|
|
}
|
|
|
|
std::string const a = string_compose (X_("script-%1"), i + 1);
|
|
Glib::RefPtr<Action> act = ActionManager::get_action(X_("LuaAction"), a.c_str());
|
|
assert (act);
|
|
if (n.empty ()) {
|
|
act->set_label (string_compose (_("Unset #%1"), i + 1));
|
|
act->set_tooltip (_("No action bound\nRight-click to assign"));
|
|
act->set_sensitive (false);
|
|
} else {
|
|
act->set_label (n);
|
|
act->set_tooltip (string_compose (_("%1\n\nClick to run\nRight-click to re-assign\nShift+right-click to unassign"), n));
|
|
act->set_sensitive (true);
|
|
}
|
|
KeyEditor::UpdateBindings ();
|
|
}
|