when calculating average slave/master delta, use absolute value.
[ardour.git] / gtk2_ardour / ardour_ui_dependents.cc
index 7f75c174eb66834a861d6ee9bb275a0fbc44dd64..87beab6b0e417faddad7fd0b2bafae52d61f638c 100644 (file)
 #include "ardour_ui.h"
 #include "public_editor.h"
 #include "meterbridge.h"
+#include "luainstance.h"
+#include "luawindow.h"
 #include "mixer_ui.h"
 #include "keyboard.h"
+#include "keyeditor.h"
 #include "splash.h"
+#include "rc_option_editor.h"
 #include "route_params_ui.h"
+#include "time_info_box.h"
 #include "opts.h"
 #include "utils.h"
 
-#include "i18n.h"
+#include "pbd/i18n.h"
 
 using namespace Gtk;
 using namespace PBD;
@@ -61,13 +66,13 @@ ARDOUR_UI::we_have_dependents ()
 {
        install_actions ();
        load_bindings ();
-       
+
        ProcessorBox::register_actions ();
 
        /* 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. 
+          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
@@ -80,31 +85,30 @@ ARDOUR_UI::we_have_dependents ()
           importantly they don't have menus showing the bindings, so it is
           less of an issue.
        */
-       
+
        Gtkmm2ext::Bindings::associate_all ();
-       
+
        editor->setup_tooltips ();
        editor->UpdateAllTransportClocks.connect (sigc::mem_fun (*this, &ARDOUR_UI::update_transport_clocks));
 
-       /* catch up on tabbable state */
+       /* catch up on tabbable state, in the right order to leave the editor
+        * selected by default
+        */
 
-       tabbable_state_change (*editor);
-       tabbable_state_change (*mixer);
        tabbable_state_change (*rc_option_editor);
-       
+       tabbable_state_change (*mixer);
+       tabbable_state_change (*editor);
+
        /* all actions are defined */
 
        ActionManager::load_menus (ARDOUR_COMMAND_LINE::menus_file);
 
-       editor->track_mixer_selection ();
-       mixer->track_editor_selection ();
-
        /* catch up on parameters */
-       
+
        boost::function<void (std::string)> pc (boost::bind (&ARDOUR_UI::parameter_changed, this, _1));
        Config->map_parameters (pc);
 
-       ARDOUR_UI_UTILS::reset_dpi ();
+       UIConfiguration::instance().reset_dpi ();
 }
 
 void
@@ -116,6 +120,7 @@ ARDOUR_UI::connect_dependents_to_session (ARDOUR::Session *s)
        BootMessage (_("Setup Mixer"));
        mixer->set_session (s);
        meterbridge->set_session (s);
+       luawindow->set_session (s);
 
        /* its safe to do this now */
 
@@ -148,7 +153,7 @@ ARDOUR_UI::tab_window_root_drop (GtkNotebook* src,
        using namespace std;
        Gtk::Notebook* nb = 0;
        Gtk::Window* win = 0;
-       Gtkmm2ext::Tabbable* tabbable = 0;
+       ArdourWidgets::Tabbable* tabbable = 0;
 
 
        if (w == GTK_WIDGET(editor->contents().gobj())) {
@@ -170,7 +175,7 @@ ARDOUR_UI::tab_window_root_drop (GtkNotebook* src,
                win->present ();
                return nb->gobj();
        }
-       
+
        return 0; /* what was that? */
 }
 
@@ -182,7 +187,7 @@ ARDOUR_UI::idle_ask_about_quit ()
        } else {
                /* no session or session not dirty, but still ask anyway */
 
-               Gtk::MessageDialog msg (string_compose ("Quit %1?", PROGRAM_NAME),
+               Gtk::MessageDialog msg (string_compose (_("Quit %1?"), PROGRAM_NAME),
                                        false, /* no markup */
                                        Gtk::MESSAGE_INFO,
                                        Gtk::BUTTONS_YES_NO,
@@ -206,18 +211,18 @@ ARDOUR_UI::main_window_delete_event (GdkEventAny* ev)
         * 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));   
-       
+
+       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)
+                      GtkWidget* w,
+                      gint x,
+                      gint y,
+                      gpointer user_data)
 {
        return ARDOUR_UI::instance()->tab_window_root_drop (src, w, x, y, user_data);
 }
@@ -232,15 +237,10 @@ ARDOUR_UI::setup_windows ()
 
        keyboard->setup_keybindings ();
 
-       /* we don't use a widget with its own window for the tab close button,
-          which makes it impossible to rely on GTK+ to generate signals for
-          events occuring "in" this widget. Instead, we pre-connect a
-          handler to the relevant events on the notebook and then check
-          to see if the event coordinates tell us that it occured "in"
-          the close button.
-       */
-       _tabs.signal_button_press_event().connect (sigc::mem_fun (*this, &ARDOUR_UI::tabs_button_event), false);
-       _tabs.signal_button_release_event().connect (sigc::mem_fun (*this, &ARDOUR_UI::tabs_button_event), false);
+       _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));
@@ -260,12 +260,18 @@ ARDOUR_UI::setup_windows ()
                return -1;
        }
 
+       if (create_luawindow ()) {
+               error << _("UI: cannot setup luawindow") << endmsg;
+               return -1;
+       }
+
        /* order of addition affects order seen in initial window display */
-       
+
        rc_option_editor->add_to_notebook (_tabs, _("Preferences"));
        mixer->add_to_notebook (_tabs, _("Mixer"));
        editor->add_to_notebook (_tabs, _("Editor"));
 
+       time_info_box = new TimeInfoBox ("ToolbarTimeInfo", false);
        /* all other dialogs are created conditionally */
 
        we_have_dependents ();
@@ -291,7 +297,7 @@ ARDOUR_UI::setup_windows ()
        main_vpacker.pack_start (top_packer, false, false);
 
        /* now add the transport frame to the top of main window */
-       
+
        main_vpacker.pack_start (transport_frame, false, false);
        main_vpacker.pack_start (_tabs, true, true);
 
@@ -299,12 +305,33 @@ ARDOUR_UI::setup_windows ()
        main_vpacker.pack_start (status_bar_hpacker, false, false);
 #endif
 
+       LuaInstance::instance()->ActionChanged.connect (sigc::mem_fun (*this, &ARDOUR_UI::update_action_script_btn));
+
+       for (int i = 0; i < 9; ++i) {
+               std::string const a = string_compose (X_("script-action-%1"), i + 1);
+               Glib::RefPtr<Action> act = ActionManager::get_action(X_("Editor"), a.c_str());
+               assert (act);
+               action_script_call_btn[i].set_text (string_compose ("%1", 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));
+               }
+               const int row = i % 2;
+               const int col = i / 2;
+               action_script_table.attach (action_script_call_btn[i], col, col + 1, row, row + 1, EXPAND, EXPAND, 1, 0);
+               action_script_call_btn[i].set_no_show_all ();
+       }
+       action_script_table.show ();
+
        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
         */
 
@@ -314,7 +341,7 @@ ARDOUR_UI::setup_windows ()
        const XMLNode* mnode = main_window_settings ();
 
        if (mnode) {
-               const XMLProperty* prop;
+               XMLProperty const * prop;
                gint x = -1;
                gint y = -1;
                gint w = -1;
@@ -326,12 +353,12 @@ ARDOUR_UI::setup_windows ()
 
                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());
                }
@@ -339,17 +366,17 @@ ARDOUR_UI::setup_windows ()
                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 {
@@ -363,13 +390,11 @@ ARDOUR_UI::setup_windows ()
                        _tabs.set_current_page (_tabs.page_num (editor->contents()));
                }
        }
-       
-       _main_window.show_all ();
+
        setup_toplevel_window (_main_window, "", this);
-       
-       _tabs.signal_switch_page().connect (sigc::mem_fun (*this, &ARDOUR_UI::tabs_switch));
-       _tabs.signal_page_removed().connect (sigc::mem_fun (*this, &ARDOUR_UI::tabs_page_removed));
-       _tabs.signal_page_added().connect (sigc::mem_fun (*this, &ARDOUR_UI::tabs_page_added));
+       _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...
@@ -378,3 +403,43 @@ ARDOUR_UI::setup_windows ()
 
        return 0;
 }
+
+bool
+ARDOUR_UI::bind_lua_action_script (GdkEventButton*ev, int i)
+{
+       if (ev->button != 3) {
+               return false;
+       }
+       LuaInstance *li = LuaInstance::instance();
+       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::update_action_script_btn (int i, const std::string& n)
+{
+       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);
+       }
+
+       std::string const a = string_compose (X_("script-action-%1"), i + 1);
+       Glib::RefPtr<Action> act = ActionManager::get_action(X_("Editor"), 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 ();
+}