add region & range loudnless report
[ardour.git] / gtk2_ardour / editor.cc
index 44cde597940b2ae526d185e969a3e7430b18cf6f..a118d1d6420b334f89e3525865c875040b219b2b 100644 (file)
 #include "gtkmm2ext/bindings.h"
 #include "gtkmm2ext/grouped_buttons.h"
 #include "gtkmm2ext/gtk_ui.h"
+#include <gtkmm2ext/keyboard.h>
 #include "gtkmm2ext/utils.h"
 #include "gtkmm2ext/window_title.h"
 #include "gtkmm2ext/choice.h"
 #include "gtkmm2ext/cell_renderer_pixbuf_toggle.h"
 
+#include "ardour/analysis_graph.h"
 #include "ardour/audio_track.h"
 #include "ardour/audioengine.h"
 #include "ardour/audioregion.h"
 #include "editor_routes.h"
 #include "editor_snapshots.h"
 #include "editor_summary.h"
+#include "export_report.h"
 #include "global_port_matrix.h"
 #include "gui_object.h"
 #include "gui_thread.h"
 #include "keyboard.h"
+#include "luainstance.h"
 #include "marker.h"
 #include "midi_region_view.h"
 #include "midi_time_axis.h"
 #include "rgb_macros.h"
 #include "rhythm_ferret.h"
 #include "selection.h"
+#include "simple_progress_dialog.h"
 #include "sfdb_ui.h"
 #include "tempo_lines.h"
 #include "time_axis_view.h"
@@ -290,7 +295,6 @@ Editor::Editor ()
        , _track_canvas_viewport (0)
        , within_track_canvas (false)
        , _verbose_cursor (0)
-       , logo_item (0)
        , tempo_group (0)
        , meter_group (0)
        , marker_group (0)
@@ -461,6 +465,7 @@ Editor::Editor ()
        , _stepping_axis_view (0)
        , quantize_dialog (0)
        , _main_menu_disabler (0)
+       , myactions (X_("editor"))
 {
        /* we are a singleton */
 
@@ -471,8 +476,6 @@ Editor::Editor ()
        last_event_time.tv_sec = 0;
        last_event_time.tv_usec = 0;
 
-       global_hpacker.set_data ("ardour-bindings", &bindings);
-
        selection_op_history.clear();
        before.clear();
 
@@ -493,11 +496,11 @@ Editor::Editor ()
        build_snap_type_menu();
        build_edit_point_menu();
 
-       location_marker_color = ARDOUR_UI::config()->color ("location marker");
-       location_range_color = ARDOUR_UI::config()->color ("location range");
-       location_cd_marker_color = ARDOUR_UI::config()->color ("location cd marker");
-       location_loop_color = ARDOUR_UI::config()->color ("location loop");
-       location_punch_color = ARDOUR_UI::config()->color ("location punch");
+       location_marker_color = UIConfiguration::instance().color ("location marker");
+       location_range_color = UIConfiguration::instance().color ("location range");
+       location_cd_marker_color = UIConfiguration::instance().color ("location cd marker");
+       location_loop_color = UIConfiguration::instance().color ("location loop");
+       location_punch_color = UIConfiguration::instance().color ("location punch");
 
        timebar_height = std::max(12., ceil (15. * ARDOUR_UI::ui_scale));
 
@@ -746,7 +749,7 @@ Editor::Editor ()
        /* need to show the "contents" widget so that notebook will show if tab is switched to
         */
 
-       global_hpacker.show (); 
+       global_hpacker.show ();
 
        /* register actions now so that set_state() can find them and set toggles/checks etc */
 
@@ -816,12 +819,12 @@ Editor::Editor ()
 
         /* Button bindings */
 
-        button_bindings = new Bindings;
+       button_bindings = new Bindings ("editor-mouse");
 
        XMLNode* node = button_settings();
         if (node) {
                 for (XMLNodeList::const_iterator i = node->children().begin(); i != node->children().end(); ++i) {
-                        button_bindings->load (**i);
+                        button_bindings->load_operation (**i);
                 }
         }
 
@@ -833,6 +836,9 @@ Editor::Editor ()
 
        setup_fade_images ();
 
+       LuaInstance::instance(); // instantiate
+       LuaInstance::instance()->ActionChanged.connect (sigc::mem_fun (*this, &Editor::set_script_action_name));
+
        instant_save ();
 }
 
@@ -1259,7 +1265,7 @@ Editor::update_title ()
        if (!own_window()) {
                return;
        }
-               
+
        if (_session) {
                bool dirty = _session->dirty();
 
@@ -1276,6 +1282,7 @@ Editor::update_title ()
                }
 
                WindowTitle title(session_name);
+               title += S_("Window|Editor");
                title += Glib::get_application_name();
                own_window()->set_title (title.get_string());
        } else {
@@ -1400,6 +1407,8 @@ Editor::set_session (Session *t)
 
        ActionManager::ui_manager->signal_pre_activate().connect (sigc::mem_fun (*this, &Editor::action_pre_activated));
 
+       LuaInstance::instance()->set_session(_session);
+
        start_updating_meters ();
 }
 
@@ -1708,7 +1717,103 @@ Editor::build_track_region_context_menu ()
 }
 
 void
-Editor::analyze_region_selection ()
+Editor::loudness_analyze_region_selection ()
+{
+       if (!_session) {
+               return;
+       }
+       Selection& s (PublicEditor::instance ().get_selection ());
+       RegionSelection ars = s.regions;
+       ARDOUR::AnalysisGraph ag (_session);
+       framecnt_t total_work = 0;
+
+       for (RegionSelection::iterator j = ars.begin (); j != ars.end (); ++j) {
+               AudioRegionView* arv = dynamic_cast<AudioRegionView*> (*j);
+               if (!arv) {
+                       continue;
+               }
+               if (!boost::dynamic_pointer_cast<AudioRegion> (arv->region ())) {
+                       continue;
+               }
+               assert (dynamic_cast<RouteTimeAxisView *> (&arv->get_time_axis_view ()));
+               total_work += arv->region ()->length ();
+       }
+
+       SimpleProgressDialog spd (_("Region Loudness Analysis"), sigc::mem_fun (ag, &AnalysisGraph::cancel));
+       ScopedConnection c;
+       ag.set_total_frames (total_work);
+       ag.Progress.connect_same_thread (c, boost::bind (&SimpleProgressDialog::update_progress, &spd, _1, _2));
+       spd.show();
+
+       for (RegionSelection::iterator j = ars.begin (); j != ars.end (); ++j) {
+               AudioRegionView* arv = dynamic_cast<AudioRegionView*> (*j);
+               if (!arv) {
+                       continue;
+               }
+               boost::shared_ptr<AudioRegion> ar = boost::dynamic_pointer_cast<AudioRegion> (arv->region ());
+               if (!ar) {
+                       continue;
+               }
+               ag.analyze_region (ar);
+       }
+       spd.hide();
+       if (!ag.canceled ()) {
+               ExportReport er (_("Audio Report/Analysis"), ag.results ());
+               er.run();
+       }
+}
+
+void
+Editor::loudness_analyze_range_selection ()
+{
+       if (!_session) {
+               return;
+       }
+       Selection& s (PublicEditor::instance ().get_selection ());
+       TimeSelection ts = s.time;
+       ARDOUR::AnalysisGraph ag (_session);
+       framecnt_t total_work = 0;
+
+       for (TrackSelection::iterator i = s.tracks.begin (); i != s.tracks.end (); ++i) {
+               boost::shared_ptr<AudioPlaylist> pl = boost::dynamic_pointer_cast<AudioPlaylist> ((*i)->playlist ());
+               if (!pl) {
+                       continue;
+               }
+               RouteUI *rui = dynamic_cast<RouteUI *> (*i);
+               if (!pl || !rui) {
+                       continue;
+               }
+               for (std::list<AudioRange>::iterator j = ts.begin (); j != ts.end (); ++j) {
+                       total_work += j->length ();
+               }
+       }
+
+       SimpleProgressDialog spd (_("Range Loudness Analysis"), sigc::mem_fun (ag, &AnalysisGraph::cancel));
+       ScopedConnection c;
+       ag.set_total_frames (total_work);
+       ag.Progress.connect_same_thread (c, boost::bind (&SimpleProgressDialog::update_progress, &spd, _1, _2));
+       spd.show();
+
+       for (TrackSelection::iterator i = s.tracks.begin (); i != s.tracks.end (); ++i) {
+               boost::shared_ptr<AudioPlaylist> pl = boost::dynamic_pointer_cast<AudioPlaylist> ((*i)->playlist ());
+               if (!pl) {
+                       continue;
+               }
+               RouteUI *rui = dynamic_cast<RouteUI *> (*i);
+               if (!pl || !rui) {
+                       continue;
+               }
+               ag.analyze_range (rui->route (), pl, ts);
+       }
+       spd.hide();
+       if (!ag.canceled ()) {
+               ExportReport er (_("Audio Report/Analysis"), ag.results ());
+               er.run();
+       }
+}
+
+void
+Editor::spectral_analyze_region_selection ()
 {
        if (analysis_window == 0) {
                analysis_window = new AnalysisWindow();
@@ -1726,7 +1831,7 @@ Editor::analyze_region_selection ()
 }
 
 void
-Editor::analyze_range_selection()
+Editor::spectral_analyze_range_selection()
 {
        if (analysis_window == 0) {
                analysis_window = new AnalysisWindow();
@@ -1820,7 +1925,8 @@ Editor::add_selection_context_items (Menu_Helpers::MenuList& edit_items)
        edit_items.push_back (MenuElem (_("Zoom to Range"), sigc::bind (sigc::mem_fun(*this, &Editor::temporal_zoom_selection), false)));
 
        edit_items.push_back (SeparatorElem());
-       edit_items.push_back (MenuElem (_("Spectral Analysis"), sigc::mem_fun(*this, &Editor::analyze_range_selection)));
+       edit_items.push_back (MenuElem (_("Loudness Analysis"), sigc::mem_fun(*this, &Editor::loudness_analyze_range_selection)));
+       edit_items.push_back (MenuElem (_("Spectral Analysis"), sigc::mem_fun(*this, &Editor::spectral_analyze_range_selection)));
 
        edit_items.push_back (SeparatorElem());
 
@@ -2184,7 +2290,7 @@ Editor::set_state (const XMLNode& node, int version)
        set_id (node);
 
        Tabbable::set_state (node, version);
-       
+
        if (_session && (prop = node.property ("playhead"))) {
                framepos_t pos;
                sscanf (prop->value().c_str(), "%" PRIi64, &pos);
@@ -2402,7 +2508,7 @@ Editor::set_state (const XMLNode& node, int version)
                }
        }
 
-       return 0;
+       return LuaInstance::instance()->set_state(node);
 }
 
 XMLNode&
@@ -2415,13 +2521,13 @@ Editor::get_state ()
        node->add_property ("id", buf);
 
        node->add_child_nocopy (Tabbable::get_state());
-       
+
        snprintf(buf,sizeof(buf), "%d",gtk_paned_get_position (static_cast<Paned*>(&edit_pane)->gobj()));
        node->add_property("edit-horizontal-pane-pos", string(buf));
        node->add_property("notebook-shrunk", _notebook_shrunk ? "1" : "0");
        snprintf(buf,sizeof(buf), "%d",gtk_paned_get_position (static_cast<Paned*>(&editor_summary_pane)->gobj()));
        node->add_property("edit-vertical-pane-pos", string(buf));
-       
+
        maybe_add_mixer_strip_width (*node);
 
        node->add_property ("zoom-focus", enum_2_string (zoom_focus));
@@ -2482,6 +2588,9 @@ Editor::get_state ()
        snprintf (buf, sizeof (buf), "%" PRId64, nudge_clock->current_duration());
        node->add_property ("nudge-clock-value", buf);
 
+       node->add_child_nocopy (LuaInstance::instance()->get_action_state());
+       node->add_child_nocopy (LuaInstance::instance()->get_hook_state());
+
        return *node;
 }
 
@@ -2886,9 +2995,9 @@ Editor::setup_toolbar ()
        if (!ARDOUR::Profile->get_trx()) {
                mode_box->pack_start (edit_mode_selector, false, false);
        }
-       
+
        mode_box->pack_start (*mouse_mode_box, false, false);
-       
+
        /* Zoom */
 
        _zoom_box.set_spacing (2);
@@ -3487,11 +3596,7 @@ Editor::cycle_edit_mode ()
 {
        switch (Config->get_edit_mode()) {
        case Slide:
-               if (Profile->get_sae()) {
-                       Config->set_edit_mode (Lock);
-               } else {
-                       Config->set_edit_mode (Ripple);
-               }
+               Config->set_edit_mode (Ripple);
                break;
        case Splice:
        case Ripple:
@@ -5608,18 +5713,42 @@ Editor::session_going_away ()
        stop_step_editing ();
 
        if (own_window()) {
-       
+
                /* get rid of any existing editor mixer strip */
-               
+
                WindowTitle title(Glib::get_application_name());
                title += _("Editor");
-               
+
                own_window()->set_title (title.get_string());
        }
 
        SessionHandlePtr::session_going_away ();
 }
 
+void
+Editor::manage_action_scripts ()
+{
+       ARDOUR_UI::instance()->lua_script_manager();
+}
+
+void
+Editor::trigger_script (int i)
+{
+       LuaInstance::instance()-> call_action (i);
+}
+
+void
+Editor::set_script_action_name (int i, const std::string& n)
+{
+       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));
+       } else {
+               act->set_label (n);
+       }
+}
 
 void
 Editor::show_editor_list (bool yn)
@@ -5852,7 +5981,7 @@ Gtk::Window*
 Editor::use_own_window (bool and_fill_it)
 {
        bool new_window = !own_window();
-       
+
        Gtk::Window* win = Tabbable::use_own_window (and_fill_it);
 
        if (win && new_window) {
@@ -5861,41 +5990,42 @@ Editor::use_own_window (bool and_fill_it)
                ARDOUR_UI::instance()->setup_toplevel_window (*win, _("Editor"), this);
 
                // win->signal_realize().connect (*this, &Editor::on_realize);
+               win->signal_event().connect (sigc::bind (sigc::ptr_fun (&Keyboard::catch_user_event_for_pre_dialog_focus), win));
                win->signal_event().connect (sigc::mem_fun (*this, &Editor::generic_event_handler));
-               win->set_data ("ardour-bindings", &bindings);
-               
+               win->set_data ("ardour-bindings", bindings);
+
                update_title ();
        }
 
        DisplaySuspender ds;
        contents().show_all ();
-       
+
        /* XXX: this is a bit unfortunate; it would probably
           be nicer if we could just call show () above rather
           than needing the show_all ()
        */
-       
+
        /* re-hide stuff if necessary */
        editor_list_button_toggled ();
        parameter_changed ("show-summary");
        parameter_changed ("show-group-tabs");
        parameter_changed ("show-zoom-tools");
-       
+
        /* now reset all audio_time_axis heights, because widgets might need
           to be re-hidden
        */
-       
+
        TimeAxisView *tv;
-       
+
        for (TrackViewList::iterator i = track_views.begin(); i != track_views.end(); ++i) {
                tv = (static_cast<TimeAxisView*>(*i));
                tv->reset_height ();
        }
-       
+
        if (current_mixer_strip) {
                current_mixer_strip->hide_things ();
                current_mixer_strip->parameter_changed ("mixer-element-visibility");
        }
-       
+
        return win;
 }