2 Copyright (C) 1999-2007 Paul Davis
4 This program is free software; you can redistribute it and/or modify
5 it under the terms of the GNU General Public License as published by
6 the Free Software Foundation; either version 2 of the License, or
7 (at your option) any later version.
9 This program is distributed in the hope that it will be useful,
10 but WITHOUT ANY WARRANTY; without even the implied warranty of
11 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 GNU General Public License for more details.
14 You should have received a copy of the GNU General Public License
15 along with this program; if not, write to the Free Software
16 Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
21 #include "gtk2ardour-config.h"
37 #include <sys/resource.h>
39 #include <gtkmm/messagedialog.h>
40 #include <gtkmm/accelmap.h>
42 #include "pbd/error.h"
43 #include "pbd/basename.h"
44 #include "pbd/compose.h"
45 #include "pbd/failed_constructor.h"
46 #include "pbd/enumwriter.h"
47 #include "pbd/memento_command.h"
48 #include "pbd/openuri.h"
49 #include "pbd/file_utils.h"
51 #include "gtkmm2ext/application.h"
52 #include "gtkmm2ext/bindings.h"
53 #include "gtkmm2ext/gtk_ui.h"
54 #include "gtkmm2ext/utils.h"
55 #include "gtkmm2ext/click_box.h"
56 #include "gtkmm2ext/fastmeter.h"
57 #include "gtkmm2ext/popup.h"
58 #include "gtkmm2ext/window_title.h"
60 #include "midi++/manager.h"
62 #include "ardour/ardour.h"
63 #include "ardour/callback.h"
64 #include "ardour/profile.h"
65 #include "ardour/session_directory.h"
66 #include "ardour/session_route.h"
67 #include "ardour/session_state_utils.h"
68 #include "ardour/session_utils.h"
69 #include "ardour/port.h"
70 #include "ardour/audioengine.h"
71 #include "ardour/playlist.h"
72 #include "ardour/utils.h"
73 #include "ardour/audio_diskstream.h"
74 #include "ardour/audiofilesource.h"
75 #include "ardour/recent_sessions.h"
76 #include "ardour/port.h"
77 #include "ardour/audio_track.h"
78 #include "ardour/midi_track.h"
79 #include "ardour/filesystem_paths.h"
80 #include "ardour/filename_extensions.h"
82 typedef uint64_t microseconds_t;
86 #include "add_route_dialog.h"
87 #include "ambiguous_file_dialog.h"
88 #include "ardour_ui.h"
89 #include "audio_clock.h"
90 #include "bundle_manager.h"
91 #include "engine_dialog.h"
92 #include "gain_meter.h"
93 #include "global_port_matrix.h"
94 #include "gui_thread.h"
96 #include "location_ui.h"
97 #include "missing_file_dialog.h"
98 #include "missing_plugin_dialog.h"
101 #include "processor_box.h"
102 #include "prompter.h"
103 #include "public_editor.h"
104 #include "route_time_axis.h"
105 #include "session_metadata_dialog.h"
106 #include "speaker_dialog.h"
109 #include "theme_manager.h"
110 #include "time_axis_view_item.h"
112 #include "window_proxy.h"
116 using namespace ARDOUR;
118 using namespace Gtkmm2ext;
121 ARDOUR_UI *ARDOUR_UI::theArdourUI = 0;
122 UIConfiguration *ARDOUR_UI::ui_config = 0;
124 sigc::signal<void,bool> ARDOUR_UI::Blink;
125 sigc::signal<void> ARDOUR_UI::RapidScreenUpdate;
126 sigc::signal<void> ARDOUR_UI::SuperRapidScreenUpdate;
127 sigc::signal<void, framepos_t, bool, framepos_t> ARDOUR_UI::Clock;
129 bool could_be_a_valid_path (const string& path);
131 ARDOUR_UI::ARDOUR_UI (int *argcp, char **argvp[])
133 : Gtkmm2ext::UI (PROGRAM_NAME, argcp, argvp),
135 primary_clock (X_("primary"), false, X_("TransportClockDisplay"), true, true, false, true),
136 secondary_clock (X_("secondary"), false, X_("SecondaryClockDisplay"), true, true, false, true),
137 preroll_clock (X_("preroll"), false, X_("PreRollClock"), true, false, true),
138 postroll_clock (X_("postroll"), false, X_("PostRollClock"), true, false, true),
142 preroll_button (_("pre\nroll")),
143 postroll_button (_("post\nroll")),
147 big_clock (X_("bigclock"), false, "BigClockNonRecording", true, true, false, false),
151 roll_controllable (new TransportControllable ("transport roll", *this, TransportControllable::Roll)),
152 stop_controllable (new TransportControllable ("transport stop", *this, TransportControllable::Stop)),
153 goto_start_controllable (new TransportControllable ("transport goto start", *this, TransportControllable::GotoStart)),
154 goto_end_controllable (new TransportControllable ("transport goto end", *this, TransportControllable::GotoEnd)),
155 auto_loop_controllable (new TransportControllable ("transport auto loop", *this, TransportControllable::AutoLoop)),
156 play_selection_controllable (new TransportControllable ("transport play selection", *this, TransportControllable::PlaySelection)),
157 rec_controllable (new TransportControllable ("transport rec-enable", *this, TransportControllable::RecordEnable)),
158 shuttle_controllable (new TransportControllable ("shuttle", *this, TransportControllable::ShuttleControl)),
159 shuttle_controller_binding_proxy (shuttle_controllable),
161 roll_button (roll_controllable),
162 stop_button (stop_controllable),
163 goto_start_button (goto_start_controllable),
164 goto_end_button (goto_end_controllable),
165 auto_loop_button (auto_loop_controllable),
166 play_selection_button (play_selection_controllable),
167 rec_button (rec_controllable),
169 shuttle_units_button (_("% ")),
171 punch_in_button (_("Punch In")),
172 punch_out_button (_("Punch Out")),
173 auto_return_button (_("Auto Return")),
174 auto_play_button (_("Auto Play")),
175 auto_input_button (_("Auto Input")),
176 click_button (_("Click")),
177 time_master_button (_("time\nmaster")),
179 auditioning_alert_button (_("AUDITION")),
180 solo_alert_button (_("SOLO")),
182 error_log_button (_("Errors"))
185 using namespace Gtk::Menu_Helpers;
191 // _auto_display_errors = false;
193 * This was commented out as it wasn't defined
194 * in A3 IIRC. If this is not needed it should
195 * be completely removed.
203 if (theArdourUI == 0) {
207 ui_config = new UIConfiguration();
208 theme_manager = new ThemeManager();
214 _session_is_new = false;
215 big_clock_window = 0;
216 big_clock_height = 0;
217 big_clock_resize_in_progress = false;
218 session_selector_window = 0;
219 last_key_press_time = 0;
220 _will_create_new_session_automatically = false;
221 add_route_dialog = 0;
223 rc_option_editor = 0;
224 session_option_editor = 0;
226 open_session_selector = 0;
227 have_configure_timeout = false;
228 have_disk_speed_dialog_displayed = false;
229 session_loaded = false;
230 last_speed_displayed = -1.0f;
231 ignore_dual_punch = false;
232 original_big_clock_width = -1;
233 original_big_clock_height = -1;
234 original_big_clock_font_size = 0;
236 roll_button.unset_flags (Gtk::CAN_FOCUS);
237 stop_button.unset_flags (Gtk::CAN_FOCUS);
238 goto_start_button.unset_flags (Gtk::CAN_FOCUS);
239 goto_end_button.unset_flags (Gtk::CAN_FOCUS);
240 auto_loop_button.unset_flags (Gtk::CAN_FOCUS);
241 play_selection_button.unset_flags (Gtk::CAN_FOCUS);
242 rec_button.unset_flags (Gtk::CAN_FOCUS);
244 last_configure_time= 0;
246 shuttle_grabbed = false;
248 shuttle_max_speed = 8.0f;
250 shuttle_style_menu = 0;
251 shuttle_unit_menu = 0;
253 // We do not have jack linked in yet so;
255 last_shuttle_request = last_peak_grab = 0; // get_microseconds();
257 ARDOUR::Diskstream::DiskOverrun.connect (forever_connections, MISSING_INVALIDATOR, boost::bind (&ARDOUR_UI::disk_overrun_handler, this), gui_context());
258 ARDOUR::Diskstream::DiskUnderrun.connect (forever_connections, MISSING_INVALIDATOR, boost::bind (&ARDOUR_UI::disk_underrun_handler, this), gui_context());
260 /* handle dialog requests */
262 ARDOUR::Session::Dialog.connect (forever_connections, MISSING_INVALIDATOR, ui_bind (&ARDOUR_UI::session_dialog, this, _1), gui_context());
264 /* handle pending state with a dialog (PROBLEM: needs to return a value and thus cannot be x-thread) */
266 ARDOUR::Session::AskAboutPendingState.connect_same_thread (forever_connections, boost::bind (&ARDOUR_UI::pending_state_dialog, this));
268 /* handle sr mismatch with a dialog (PROBLEM: needs to return a value and thus cannot be x-thread) */
270 ARDOUR::Session::AskAboutSampleRateMismatch.connect_same_thread (forever_connections, boost::bind (&ARDOUR_UI::sr_mismatch_dialog, this, _1, _2));
272 /* handle requests to quit (coming from JACK session) */
274 ARDOUR::Session::Quit.connect (forever_connections, MISSING_INVALIDATOR, ui_bind (&ARDOUR_UI::finish, this), gui_context ());
276 /* handle requests to deal with missing files */
278 ARDOUR::Session::MissingFile.connect_same_thread (forever_connections, boost::bind (&ARDOUR_UI::missing_file, this, _1, _2, _3));
280 /* and ambiguous files */
282 ARDOUR::FileSource::AmbiguousFileName.connect_same_thread (forever_connections, boost::bind (&ARDOUR_UI::ambiguous_file, this, _1, _2, _3));
284 /* lets get this party started */
287 if (ARDOUR::init (ARDOUR_COMMAND_LINE::use_vst, ARDOUR_COMMAND_LINE::try_hw_optimization)) {
288 throw failed_constructor ();
291 setup_gtk_ardour_enums ();
294 GainMeter::setup_slider_pix ();
295 RouteTimeAxisView::setup_slider_pix ();
296 SendProcessorEntry::setup_slider_pix ();
297 SessionEvent::create_per_thread_pool ("GUI", 512);
299 } catch (failed_constructor& err) {
300 error << string_compose (_("could not initialize %1."), PROGRAM_NAME) << endmsg;
305 /* we like keyboards */
307 keyboard = new ArdourKeyboard(*this);
310 XMLNode* node = ARDOUR_UI::instance()->keyboard_settings();
312 keyboard->set_state (*node, Stateful::loading_state_version);
315 /* we don't like certain modifiers */
316 Bindings::set_ignored_state (GDK_LOCK_MASK|GDK_MOD2_MASK|GDK_MOD3_MASK);
320 TimeAxisViewItem::set_constant_heights ();
322 /* The following must happen after ARDOUR::init() so that Config is set up */
324 location_ui = new ActionWindowProxy<LocationUIWindow> (X_("locations"), Config->extra_xml (X_("UI")), X_("ToggleLocations"));
325 big_clock_window = new ActionWindowProxy<Gtk::Window> (X_("bigclock"), Config->extra_xml (X_("UI")), X_("ToggleBigClock"));
326 speaker_config_window = new ActionWindowProxy<SpeakerDialog> (X_("speakerconf"), Config->extra_xml (X_("UI")), X_("toggle-speaker-config"));
328 for (ARDOUR::DataType::iterator i = ARDOUR::DataType::begin(); i != ARDOUR::DataType::end(); ++i) {
329 _global_port_matrix[*i] = new ActionWindowProxy<GlobalPortMatrixWindow> (
330 string_compose ("GlobalPortMatrix-%1", (*i).to_string()),
331 Config->extra_xml (X_("UI")),
332 string_compose ("toggle-%1-connection-manager", (*i).to_string())
337 speaker_config_window->set (new SpeakerDialog);
339 starting.connect (sigc::mem_fun(*this, &ARDOUR_UI::startup));
340 stopping.connect (sigc::mem_fun(*this, &ARDOUR_UI::shutdown));
345 /** @return true if a session was chosen and `apply' clicked, otherwise false if `cancel' was clicked */
347 ARDOUR_UI::run_startup (bool should_be_new, string load_template)
350 _startup = new ArdourStartup ();
352 XMLNode* audio_setup = Config->extra_xml ("AudioSetup");
354 if (audio_setup && _startup->engine_control()) {
355 _startup->engine_control()->set_state (*audio_setup);
358 _startup->set_new_only (should_be_new);
359 if (!load_template.empty()) {
360 _startup->set_load_template( load_template );
362 _startup->present ();
368 switch (_startup->response()) {
377 ARDOUR_UI::create_engine ()
379 // this gets called every time by new_session()
385 loading_message (_("Starting audio engine"));
388 engine = new ARDOUR::AudioEngine (ARDOUR_COMMAND_LINE::jack_client_name, ARDOUR_COMMAND_LINE::jack_session_uuid);
395 engine->Stopped.connect (forever_connections, MISSING_INVALIDATOR, boost::bind (&ARDOUR_UI::engine_stopped, this), gui_context());
396 engine->Running.connect (forever_connections, MISSING_INVALIDATOR, boost::bind (&ARDOUR_UI::engine_running, this), gui_context());
397 engine->SampleRateChanged.connect (forever_connections, MISSING_INVALIDATOR, ui_bind (&ARDOUR_UI::update_sample_rate, this, _1), gui_context());
399 engine->Halted.connect_same_thread (forever_connections, boost::bind (&ARDOUR_UI::engine_halted, this, _1, false));
401 ARDOUR::Port::set_connecting_blocked (ARDOUR_COMMAND_LINE::no_connect_ports);
409 ARDOUR_UI::post_engine ()
411 /* Things to be done once we create the AudioEngine
414 ARDOUR::init_post_engine ();
416 ActionManager::init ();
419 if (setup_windows ()) {
420 throw failed_constructor ();
423 check_memory_locking();
425 /* this is the first point at which all the keybindings are available */
427 if (ARDOUR_COMMAND_LINE::show_key_actions) {
428 vector<string> names;
429 vector<string> paths;
430 vector<string> tooltips;
432 vector<AccelKey> bindings;
434 ActionManager::get_all_actions (names, paths, tooltips, keys, bindings);
436 vector<string>::iterator n;
437 vector<string>::iterator k;
438 for (n = names.begin(), k = keys.begin(); n != names.end(); ++n, ++k) {
439 cerr << "Action: " << (*n) << " bound to " << (*k) << endl;
445 blink_timeout_tag = -1;
447 /* this being a GUI and all, we want peakfiles */
449 AudioFileSource::set_build_peakfiles (true);
450 AudioFileSource::set_build_missing_peakfiles (true);
452 /* set default clock modes */
454 if (Profile->get_sae()) {
455 primary_clock.set_mode (AudioClock::BBT);
456 secondary_clock.set_mode (AudioClock::MinSec);
458 primary_clock.set_mode (AudioClock::Timecode);
459 secondary_clock.set_mode (AudioClock::BBT);
462 /* start the time-of-day-clock */
465 /* OS X provides a nearly-always visible wallclock, so don't be stupid */
466 update_wall_clock ();
467 Glib::signal_timeout().connect (sigc::mem_fun(*this, &ARDOUR_UI::update_wall_clock), 60000);
470 update_disk_space ();
472 update_sample_rate (engine->frame_rate());
474 Config->ParameterChanged.connect (forever_connections, MISSING_INVALIDATOR, ui_bind (&ARDOUR_UI::parameter_changed, this, _1), gui_context());
475 boost::function<void (string)> pc (boost::bind (&ARDOUR_UI::parameter_changed, this, _1));
476 Config->map_parameters (pc);
478 /* now start and maybe save state */
480 if (do_engine_start () == 0) {
481 if (_session && _session_is_new) {
482 /* we need to retain initial visual
483 settings for a new session
485 _session->save_state ("");
490 ARDOUR_UI::~ARDOUR_UI ()
495 delete add_route_dialog;
499 ARDOUR_UI::pop_back_splash ()
501 if (Splash::instance()) {
502 // Splash::instance()->pop_back();
503 Splash::instance()->hide ();
508 ARDOUR_UI::configure_timeout ()
510 if (last_configure_time == 0) {
511 /* no configure events yet */
515 /* force a gap of 0.5 seconds since the last configure event
518 if (get_microseconds() - last_configure_time < 500000) {
521 have_configure_timeout = false;
522 cerr << "config event-driven save\n";
523 save_ardour_state ();
529 ARDOUR_UI::configure_handler (GdkEventConfigure* /*conf*/)
531 if (have_configure_timeout) {
532 last_configure_time = get_microseconds();
534 Glib::signal_timeout().connect (sigc::mem_fun(*this, &ARDOUR_UI::configure_timeout), 100);
535 have_configure_timeout = true;
542 ARDOUR_UI::set_transport_controllable_state (const XMLNode& node)
544 const XMLProperty* prop;
546 if ((prop = node.property ("roll")) != 0) {
547 roll_controllable->set_id (prop->value());
549 if ((prop = node.property ("stop")) != 0) {
550 stop_controllable->set_id (prop->value());
552 if ((prop = node.property ("goto-start")) != 0) {
553 goto_start_controllable->set_id (prop->value());
555 if ((prop = node.property ("goto-end")) != 0) {
556 goto_end_controllable->set_id (prop->value());
558 if ((prop = node.property ("auto-loop")) != 0) {
559 auto_loop_controllable->set_id (prop->value());
561 if ((prop = node.property ("play-selection")) != 0) {
562 play_selection_controllable->set_id (prop->value());
564 if ((prop = node.property ("rec")) != 0) {
565 rec_controllable->set_id (prop->value());
567 if ((prop = node.property ("shuttle")) != 0) {
568 shuttle_controllable->set_id (prop->value());
573 ARDOUR_UI::get_transport_controllable_state ()
575 XMLNode* node = new XMLNode(X_("TransportControllables"));
578 roll_controllable->id().print (buf, sizeof (buf));
579 node->add_property (X_("roll"), buf);
580 stop_controllable->id().print (buf, sizeof (buf));
581 node->add_property (X_("stop"), buf);
582 goto_start_controllable->id().print (buf, sizeof (buf));
583 node->add_property (X_("goto_start"), buf);
584 goto_end_controllable->id().print (buf, sizeof (buf));
585 node->add_property (X_("goto_end"), buf);
586 auto_loop_controllable->id().print (buf, sizeof (buf));
587 node->add_property (X_("auto_loop"), buf);
588 play_selection_controllable->id().print (buf, sizeof (buf));
589 node->add_property (X_("play_selection"), buf);
590 rec_controllable->id().print (buf, sizeof (buf));
591 node->add_property (X_("rec"), buf);
592 shuttle_controllable->id().print (buf, sizeof (buf));
593 node->add_property (X_("shuttle"), buf);
600 ARDOUR_UI::autosave_session ()
602 if (g_main_depth() > 1) {
603 /* inside a recursive main loop,
604 give up because we may not be able to
610 if (!Config->get_periodic_safety_backups()) {
615 _session->maybe_write_autosave();
622 ARDOUR_UI::update_autosave ()
624 ENSURE_GUI_THREAD (*this, &ARDOUR_UI::update_autosave)
626 if (_session && _session->dirty()) {
627 if (_autosave_connection.connected()) {
628 _autosave_connection.disconnect();
631 _autosave_connection = Glib::signal_timeout().connect (sigc::mem_fun (*this, &ARDOUR_UI::autosave_session),
632 Config->get_periodic_safety_backup_interval() * 1000);
635 if (_autosave_connection.connected()) {
636 _autosave_connection.disconnect();
642 ARDOUR_UI::backend_audio_error (bool we_set_params, Gtk::Window* toplevel)
646 title = string_compose (_("%1 could not start JACK"), PROGRAM_NAME);
648 title = string_compose (_("%1 could not connect to JACK."), PROGRAM_NAME);
651 MessageDialog win (title,
657 win.set_secondary_text(_("There are several possible reasons:\n\
659 1) You requested audio parameters that are not supported..\n\
660 2) JACK is running as another user.\n\
662 Please consider the possibilities, and perhaps try different parameters."));
664 win.set_secondary_text(_("There are several possible reasons:\n\
666 1) JACK is not running.\n\
667 2) JACK is running as another user, perhaps root.\n\
668 3) There is already another client called \"ardour\".\n\
670 Please consider the possibilities, and perhaps (re)start JACK."));
674 win.set_transient_for (*toplevel);
678 win.add_button (Stock::OK, RESPONSE_CLOSE);
680 win.add_button (Stock::QUIT, RESPONSE_CLOSE);
683 win.set_default_response (RESPONSE_CLOSE);
686 win.set_position (Gtk::WIN_POS_CENTER);
689 /* we just don't care about the result, but we want to block */
695 ARDOUR_UI::startup ()
697 Application* app = Application::instance ();
699 app->ShouldQuit.connect (sigc::mem_fun (*this, &ARDOUR_UI::queue_finish));
700 app->ShouldLoad.connect (sigc::mem_fun (*this, &ARDOUR_UI::idle_load));
703 call_the_mothership (VERSIONSTRING);
708 if (get_session_parameters (true, ARDOUR_COMMAND_LINE::new_session, ARDOUR_COMMAND_LINE::load_template)) {
714 goto_editor_window ();
716 /* Add the window proxies here; their addition may cause windows to be opened, and we want them
717 to be opened on top of the editor window that goto_editor_window() just opened.
719 add_window_proxy (location_ui);
720 add_window_proxy (big_clock_window);
721 for (ARDOUR::DataType::iterator i = ARDOUR::DataType::begin(); i != ARDOUR::DataType::end(); ++i) {
722 add_window_proxy (_global_port_matrix[*i]);
725 BootMessage (string_compose (_("%1 is ready for use"), PROGRAM_NAME));
729 ARDOUR_UI::no_memory_warning ()
731 XMLNode node (X_("no-memory-warning"));
732 Config->add_instant_xml (node);
736 ARDOUR_UI::check_memory_locking ()
739 /* OS X doesn't support mlockall(2), and so testing for memory locking capability there is pointless */
743 XMLNode* memory_warning_node = Config->instant_xml (X_("no-memory-warning"));
745 if (engine->is_realtime() && memory_warning_node == 0) {
747 struct rlimit limits;
749 long pages, page_size;
751 if ((page_size = sysconf (_SC_PAGESIZE)) < 0 ||(pages = sysconf (_SC_PHYS_PAGES)) < 0) {
754 ram = (int64_t) pages * (int64_t) page_size;
757 if (getrlimit (RLIMIT_MEMLOCK, &limits)) {
761 if (limits.rlim_cur != RLIM_INFINITY) {
763 if (ram == 0 || ((double) limits.rlim_cur / ram) < 0.75) {
767 string_compose (_("WARNING: Your system has a limit for maximum amount of locked memory. "
768 "This might cause %1 to run out of memory before your system "
769 "runs out of memory. \n\n"
770 "You can view the memory limit with 'ulimit -l', "
771 "and it is normally controlled by /etc/security/limits.conf"),
772 PROGRAM_NAME).c_str());
774 VBox* vbox = msg.get_vbox();
776 CheckButton cb (_("Do not show this window again"));
778 cb.signal_toggled().connect (sigc::mem_fun (*this, &ARDOUR_UI::no_memory_warning));
780 hbox.pack_start (cb, true, false);
781 vbox->pack_start (hbox);
788 editor->ensure_float (msg);
798 ARDOUR_UI::queue_finish ()
800 Glib::signal_idle().connect (mem_fun (*this, &ARDOUR_UI::idle_finish));
804 ARDOUR_UI::idle_finish ()
807 return false; /* do not call again */
816 if (_session->transport_rolling() && (++tries < 8)) {
817 _session->request_stop (false, true);
821 if (_session->dirty()) {
822 switch (ask_about_saving_session(_("quit"))) {
827 /* use the default name */
828 if (save_state_canfail ("")) {
829 /* failed - don't quit */
830 MessageDialog msg (*editor,
832 Ardour was unable to save your session.\n\n\
833 If you still wish to quit, please use the\n\n\
834 \"Just quit\" option."));
845 second_connection.disconnect ();
846 point_one_second_connection.disconnect ();
847 point_oh_five_second_connection.disconnect ();
848 point_zero_one_second_connection.disconnect();
851 /* Save state before deleting the session, as that causes some
852 windows to be destroyed before their visible state can be
855 save_ardour_state ();
858 // _session->set_deletion_in_progress ();
859 _session->set_clean ();
860 _session->remove_pending_capture_state ();
865 ArdourDialog::close_all_dialogs ();
871 ARDOUR_UI::ask_about_saving_session (const string & what)
873 ArdourDialog window (_("Unsaved Session"));
874 Gtk::HBox dhbox; // the hbox for the image and text
875 Gtk::Label prompt_label;
876 Gtk::Image* dimage = manage (new Gtk::Image(Stock::DIALOG_WARNING, Gtk::ICON_SIZE_DIALOG));
880 msg = string_compose(_("Don't %1"), what);
881 window.add_button (msg, RESPONSE_REJECT);
882 msg = string_compose(_("Just %1"), what);
883 window.add_button (msg, RESPONSE_APPLY);
884 msg = string_compose(_("Save and %1"), what);
885 window.add_button (msg, RESPONSE_ACCEPT);
887 window.set_default_response (RESPONSE_ACCEPT);
889 Gtk::Button noquit_button (msg);
890 noquit_button.set_name ("EditorGTKButton");
895 if (_session->snap_name() == _session->name()) {
898 type = _("snapshot");
900 prompt = string_compose(_("The %1 \"%2\"\nhas not been saved.\n\nAny changes made this time\nwill be lost unless you save it.\n\nWhat do you want to do?"),
901 type, _session->snap_name());
903 prompt_label.set_text (prompt);
904 prompt_label.set_name (X_("PrompterLabel"));
905 prompt_label.set_alignment(ALIGN_LEFT, ALIGN_TOP);
907 dimage->set_alignment(ALIGN_CENTER, ALIGN_TOP);
908 dhbox.set_homogeneous (false);
909 dhbox.pack_start (*dimage, false, false, 5);
910 dhbox.pack_start (prompt_label, true, false, 5);
911 window.get_vbox()->pack_start (dhbox);
913 window.set_name (_("Prompter"));
914 window.set_position (Gtk::WIN_POS_MOUSE);
915 window.set_modal (true);
916 window.set_resizable (false);
922 window.set_keep_above (true);
925 ResponseType r = (ResponseType) window.run();
930 case RESPONSE_ACCEPT: // save and get out of here
932 case RESPONSE_APPLY: // get out of here
942 ARDOUR_UI::every_second ()
945 update_buffer_load ();
946 update_disk_space ();
951 ARDOUR_UI::every_point_one_seconds ()
953 update_speed_display ();
954 RapidScreenUpdate(); /* EMIT_SIGNAL */
959 ARDOUR_UI::every_point_zero_one_seconds ()
961 // august 2007: actual update frequency: 40Hz, not 100Hz
963 SuperRapidScreenUpdate(); /* EMIT_SIGNAL */
968 ARDOUR_UI::update_sample_rate (framecnt_t)
972 ENSURE_GUI_THREAD (*this, &ARDOUR_UI::update_sample_rate, ignored)
974 if (!engine->connected()) {
976 snprintf (buf, sizeof (buf), _("disconnected"));
980 framecnt_t rate = engine->frame_rate();
982 if (fmod (rate, 1000.0) != 0.0) {
983 snprintf (buf, sizeof (buf), _("%.1f kHz / %4.1f ms"),
984 (float) rate/1000.0f,
985 (engine->frames_per_cycle() / (float) rate) * 1000.0f);
987 snprintf (buf, sizeof (buf), _("%" PRId64 " kHz / %4.1f ms"),
989 (engine->frames_per_cycle() / (float) rate) * 1000.0f);
993 sample_rate_label.set_text (buf);
997 ARDOUR_UI::update_cpu_load ()
1000 snprintf (buf, sizeof (buf), _("DSP: %5.1f%%"), engine->get_cpu_load());
1001 cpu_load_label.set_text (buf);
1005 ARDOUR_UI::update_buffer_load ()
1011 c = _session->capture_load ();
1012 p = _session->playback_load ();
1014 snprintf (buf, sizeof (buf), _("Buffers p:%" PRIu32 "%% c:%" PRIu32 "%%"),
1015 _session->playback_load(), _session->capture_load());
1016 buffer_load_label.set_text (buf);
1018 buffer_load_label.set_text ("");
1023 ARDOUR_UI::count_recenabled_streams (Route& route)
1025 Track* track = dynamic_cast<Track*>(&route);
1026 if (track && track->record_enabled()) {
1027 rec_enabled_streams += track->n_inputs().n_total();
1032 ARDOUR_UI::update_disk_space()
1034 if (_session == 0) {
1038 framecnt_t frames = _session->available_capture_duration();
1040 framecnt_t fr = _session->frame_rate();
1042 if (frames == max_framecnt) {
1043 strcpy (buf, _("Disk: 24hrs+"));
1045 rec_enabled_streams = 0;
1046 _session->foreach_route (this, &ARDOUR_UI::count_recenabled_streams);
1048 if (rec_enabled_streams) {
1049 frames /= rec_enabled_streams;
1056 hrs = frames / (fr * 3600);
1057 frames -= hrs * fr * 3600;
1058 mins = frames / (fr * 60);
1059 frames -= mins * fr * 60;
1062 snprintf (buf, sizeof(buf), _("Disk: %02dh:%02dm:%02ds"), hrs, mins, secs);
1065 disk_space_label.set_text (buf);
1067 // An attempt to make the disk space label flash red when space has run out.
1069 if (frames < fr * 60 * 5) {
1070 /* disk_space_box.style ("disk_space_label_empty"); */
1072 /* disk_space_box.style ("disk_space_label"); */
1078 ARDOUR_UI::update_wall_clock ()
1085 tm_now = localtime (&now);
1087 sprintf (buf, "%02d:%02d", tm_now->tm_hour, tm_now->tm_min);
1088 wall_clock_label.set_text (buf);
1094 ARDOUR_UI::session_menu (GdkEventButton */*ev*/)
1096 session_popup_menu->popup (0, 0);
1101 ARDOUR_UI::redisplay_recent_sessions ()
1103 std::vector<sys::path> session_directories;
1104 RecentSessionsSorter cmp;
1106 recent_session_display.set_model (Glib::RefPtr<TreeModel>(0));
1107 recent_session_model->clear ();
1109 ARDOUR::RecentSessions rs;
1110 ARDOUR::read_recent_sessions (rs);
1113 recent_session_display.set_model (recent_session_model);
1117 // sort them alphabetically
1118 sort (rs.begin(), rs.end(), cmp);
1120 for (ARDOUR::RecentSessions::iterator i = rs.begin(); i != rs.end(); ++i) {
1121 session_directories.push_back ((*i).second);
1124 for (vector<sys::path>::const_iterator i = session_directories.begin();
1125 i != session_directories.end(); ++i)
1127 std::vector<sys::path> state_file_paths;
1129 // now get available states for this session
1131 get_state_files_in_directory (*i, state_file_paths);
1133 vector<string*>* states;
1134 vector<const gchar*> item;
1135 string fullpath = (*i).to_string();
1137 /* remove any trailing / */
1139 if (fullpath[fullpath.length()-1] == '/') {
1140 fullpath = fullpath.substr (0, fullpath.length()-1);
1143 /* check whether session still exists */
1144 if (!Glib::file_test(fullpath.c_str(), Glib::FILE_TEST_EXISTS)) {
1145 /* session doesn't exist */
1146 cerr << "skipping non-existent session " << fullpath << endl;
1150 /* now get available states for this session */
1152 if ((states = Session::possible_states (fullpath)) == 0) {
1153 /* no state file? */
1157 std::vector<string> state_file_names(get_file_names_no_extension (state_file_paths));
1159 Gtk::TreeModel::Row row = *(recent_session_model->append());
1161 row[recent_session_columns.visible_name] = Glib::path_get_basename (fullpath);
1162 row[recent_session_columns.fullpath] = fullpath;
1164 if (state_file_names.size() > 1) {
1168 for (std::vector<std::string>::iterator i2 = state_file_names.begin();
1169 i2 != state_file_names.end(); ++i2)
1172 Gtk::TreeModel::Row child_row = *(recent_session_model->append (row.children()));
1174 child_row[recent_session_columns.visible_name] = *i2;
1175 child_row[recent_session_columns.fullpath] = fullpath;
1180 recent_session_display.set_model (recent_session_model);
1184 ARDOUR_UI::build_session_selector ()
1186 session_selector_window = new ArdourDialog (_("Recent Sessions"));
1188 Gtk::ScrolledWindow *scroller = manage (new Gtk::ScrolledWindow);
1190 session_selector_window->add_button (Stock::CANCEL, RESPONSE_CANCEL);
1191 session_selector_window->add_button (Stock::OPEN, RESPONSE_ACCEPT);
1192 session_selector_window->set_default_response (RESPONSE_ACCEPT);
1193 recent_session_model = TreeStore::create (recent_session_columns);
1194 recent_session_display.set_model (recent_session_model);
1195 recent_session_display.append_column (_("Recent Sessions"), recent_session_columns.visible_name);
1196 recent_session_display.set_headers_visible (false);
1197 recent_session_display.get_selection()->set_mode (SELECTION_BROWSE);
1198 recent_session_display.signal_row_activated().connect (sigc::mem_fun (*this, &ARDOUR_UI::recent_session_row_activated));
1200 scroller->add (recent_session_display);
1201 scroller->set_policy (Gtk::POLICY_NEVER, Gtk::POLICY_AUTOMATIC);
1203 session_selector_window->set_name ("SessionSelectorWindow");
1204 session_selector_window->set_size_request (200, 400);
1205 session_selector_window->get_vbox()->pack_start (*scroller);
1207 recent_session_display.show();
1209 //session_selector_window->get_vbox()->show();
1213 ARDOUR_UI::recent_session_row_activated (const TreePath& /*path*/, TreeViewColumn* /*col*/)
1215 session_selector_window->response (RESPONSE_ACCEPT);
1219 ARDOUR_UI::open_recent_session ()
1221 bool can_return = (_session != 0);
1223 if (session_selector_window == 0) {
1224 build_session_selector ();
1227 redisplay_recent_sessions ();
1231 session_selector_window->set_position (WIN_POS_MOUSE);
1233 ResponseType r = (ResponseType) session_selector_window->run ();
1236 case RESPONSE_ACCEPT:
1240 session_selector_window->hide();
1247 if (recent_session_display.get_selection()->count_selected_rows() == 0) {
1251 session_selector_window->hide();
1253 Gtk::TreeModel::iterator i = recent_session_display.get_selection()->get_selected();
1255 if (i == recent_session_model->children().end()) {
1259 std::string path = (*i)[recent_session_columns.fullpath];
1260 std::string state = (*i)[recent_session_columns.visible_name];
1262 _session_is_new = false;
1264 if (load_session (path, state) == 0) {
1273 ARDOUR_UI::check_audioengine ()
1276 if (!engine->connected()) {
1277 MessageDialog msg (string_compose (_("%1 is not connected to JACK\n"
1278 "You cannot open or close sessions in this condition"),
1291 ARDOUR_UI::open_session ()
1293 if (!check_audioengine()) {
1298 /* popup selector window */
1300 if (open_session_selector == 0) {
1302 /* ardour sessions are folders */
1304 open_session_selector = new Gtk::FileChooserDialog (_("Open Session"), FILE_CHOOSER_ACTION_OPEN);
1305 open_session_selector->add_button (Gtk::Stock::CANCEL, Gtk::RESPONSE_CANCEL);
1306 open_session_selector->add_button (Gtk::Stock::OPEN, Gtk::RESPONSE_ACCEPT);
1307 open_session_selector->set_default_response(Gtk::RESPONSE_ACCEPT);
1309 FileFilter session_filter;
1310 session_filter.add_pattern ("*.ardour");
1311 session_filter.set_name (string_compose (_("%1 sessions"), PROGRAM_NAME));
1312 open_session_selector->add_filter (session_filter);
1313 open_session_selector->set_filter (session_filter);
1316 int response = open_session_selector->run();
1317 open_session_selector->hide ();
1320 case RESPONSE_ACCEPT:
1323 open_session_selector->hide();
1327 open_session_selector->hide();
1328 string session_path = open_session_selector->get_filename();
1332 if (session_path.length() > 0) {
1333 if (ARDOUR::find_session (session_path, path, name, isnew) == 0) {
1334 _session_is_new = isnew;
1335 load_session (path, name);
1342 ARDOUR_UI::session_add_midi_route (bool disk, RouteGroup* route_group, uint32_t how_many, string const & name_template)
1344 list<boost::shared_ptr<MidiTrack> > tracks;
1346 if (_session == 0) {
1347 warning << _("You cannot add a track without a session already loaded.") << endmsg;
1354 tracks = _session->new_midi_track (ARDOUR::Normal, route_group, how_many, name_template);
1356 if (tracks.size() != how_many) {
1357 if (how_many == 1) {
1358 error << _("could not create a new midi track") << endmsg;
1360 error << string_compose (_("could not create %1 new midi tracks"), how_many) << endmsg;
1364 if ((route = _session->new_midi_route ()) == 0) {
1365 error << _("could not create new midi bus") << endmsg;
1371 MessageDialog msg (*editor,
1372 string_compose (_("There are insufficient JACK ports available\n\
1373 to create a new track or bus.\n\
1374 You should save %1, exit and\n\
1375 restart JACK with more ports."), PROGRAM_NAME));
1382 ARDOUR_UI::session_add_audio_route (
1384 int32_t input_channels,
1385 int32_t output_channels,
1386 ARDOUR::TrackMode mode,
1387 RouteGroup* route_group,
1389 string const & name_template
1392 list<boost::shared_ptr<AudioTrack> > tracks;
1395 if (_session == 0) {
1396 warning << _("You cannot add a track or bus without a session already loaded.") << endmsg;
1402 tracks = _session->new_audio_track (input_channels, output_channels, mode, route_group, how_many, name_template);
1404 if (tracks.size() != how_many) {
1405 if (how_many == 1) {
1406 error << _("could not create a new audio track") << endmsg;
1408 error << string_compose (_("could only create %1 of %2 new audio %3"),
1409 tracks.size(), how_many, (track ? _("tracks") : _("busses"))) << endmsg;
1415 routes = _session->new_audio_route (input_channels, output_channels, route_group, how_many, name_template);
1417 if (routes.size() != how_many) {
1418 if (how_many == 1) {
1419 error << _("could not create a new audio track") << endmsg;
1421 error << string_compose (_("could not create %1 new audio tracks"), how_many) << endmsg;
1428 MessageDialog msg (*editor,
1429 string_compose (_("There are insufficient JACK ports available\n\
1430 to create a new track or bus.\n\
1431 You should save %1, exit and\n\
1432 restart JACK with more ports."), PROGRAM_NAME));
1439 ARDOUR_UI::do_transport_locate (framepos_t new_position, bool with_roll)
1441 framecnt_t _preroll = 0;
1444 // XXX CONFIG_CHANGE FIX - requires AnyTime handling
1445 // _preroll = _session->convert_to_frames_at (new_position, Config->get_preroll());
1447 if (new_position > _preroll) {
1448 new_position -= _preroll;
1453 _session->request_locate (new_position, with_roll);
1458 ARDOUR_UI::transport_goto_start ()
1461 _session->goto_start();
1463 /* force displayed area in editor to start no matter
1464 what "follow playhead" setting is.
1468 editor->center_screen (_session->current_start_frame ());
1474 ARDOUR_UI::transport_goto_zero ()
1477 _session->request_locate (0);
1479 /* force displayed area in editor to start no matter
1480 what "follow playhead" setting is.
1484 editor->reset_x_origin (0);
1490 ARDOUR_UI::transport_goto_wallclock ()
1492 if (_session && editor) {
1499 localtime_r (&now, &tmnow);
1501 frames = tmnow.tm_hour * (60 * 60 * _session->frame_rate());
1502 frames += tmnow.tm_min * (60 * _session->frame_rate());
1503 frames += tmnow.tm_sec * _session->frame_rate();
1505 _session->request_locate (frames, _session->transport_rolling ());
1507 /* force displayed area in editor to start no matter
1508 what "follow playhead" setting is.
1512 editor->center_screen (frames);
1518 ARDOUR_UI::transport_goto_end ()
1521 framepos_t const frame = _session->current_end_frame();
1522 _session->request_locate (frame);
1524 /* force displayed area in editor to start no matter
1525 what "follow playhead" setting is.
1529 editor->center_screen (frame);
1535 ARDOUR_UI::transport_stop ()
1541 if (_session->is_auditioning()) {
1542 _session->cancel_audition ();
1546 _session->request_stop (false, true);
1550 ARDOUR_UI::transport_stop_and_forget_capture ()
1553 _session->request_stop (true, true);
1558 ARDOUR_UI::remove_last_capture()
1561 editor->remove_last_capture();
1566 ARDOUR_UI::transport_record (bool roll)
1570 switch (_session->record_status()) {
1571 case Session::Disabled:
1572 if (_session->ntracks() == 0) {
1573 MessageDialog msg (*editor, _("Please create 1 or more track\nbefore trying to record.\nCheck the Session menu."));
1577 _session->maybe_enable_record ();
1582 case Session::Recording:
1584 _session->request_stop();
1586 _session->disable_record (false, true);
1590 case Session::Enabled:
1591 _session->disable_record (false, true);
1594 //cerr << "ARDOUR_UI::transport_record () called roll = " << roll << " _session->record_status() = " << _session->record_status() << endl;
1598 ARDOUR_UI::transport_roll ()
1604 if (_session->is_auditioning()) {
1609 if (_session->config.get_external_sync()) {
1610 switch (_session->config.get_sync_source()) {
1614 /* transport controlled by the master */
1620 bool rolling = _session->transport_rolling();
1622 if (_session->get_play_loop()) {
1623 /* XXX it is not possible to just leave seamless loop and keep
1624 playing at present (nov 4th 2009)
1626 if (!Config->get_seamless_loop()) {
1627 _session->request_play_loop (false, true);
1629 } else if (_session->get_play_range () && !join_play_range_button.get_active()) {
1630 /* stop playing a range if we currently are */
1631 _session->request_play_range (0, true);
1634 if (join_play_range_button.get_active()) {
1635 _session->request_play_range (&editor->get_selection().time, true);
1639 _session->request_transport_speed (1.0f);
1644 ARDOUR_UI::toggle_roll (bool with_abort, bool roll_out_of_bounded_mode)
1651 if (_session->is_auditioning()) {
1652 _session->cancel_audition ();
1657 if (_session->config.get_external_sync()) {
1658 switch (_session->config.get_sync_source()) {
1662 /* transport controlled by the master */
1668 bool rolling = _session->transport_rolling();
1669 bool affect_transport = true;
1671 if (rolling && roll_out_of_bounded_mode) {
1672 /* drop out of loop/range playback but leave transport rolling */
1673 if (_session->get_play_loop()) {
1674 if (Config->get_seamless_loop()) {
1675 /* the disk buffers contain copies of the loop - we can't
1676 just keep playing, so stop the transport. the user
1677 can restart as they wish.
1679 affect_transport = true;
1681 /* disk buffers are normal, so we can keep playing */
1682 affect_transport = false;
1684 _session->request_play_loop (false, true);
1685 } else if (_session->get_play_range ()) {
1686 affect_transport = false;
1687 _session->request_play_range (0, true);
1691 if (affect_transport) {
1693 _session->request_stop (with_abort, true);
1695 if (join_play_range_button.get_active()) {
1696 _session->request_play_range (&editor->get_selection().time, true);
1699 _session->request_transport_speed (1.0f);
1705 ARDOUR_UI::toggle_session_auto_loop ()
1711 if (_session->get_play_loop()) {
1713 if (_session->transport_rolling()) {
1715 Location * looploc = _session->locations()->auto_loop_location();
1718 _session->request_locate (looploc->start(), true);
1719 _session->request_play_loop (false);
1723 _session->request_play_loop (false);
1727 Location * looploc = _session->locations()->auto_loop_location();
1730 _session->request_play_loop (true);
1736 ARDOUR_UI::transport_play_selection ()
1742 editor->play_selection ();
1746 ARDOUR_UI::transport_rewind (int option)
1748 float current_transport_speed;
1751 current_transport_speed = _session->transport_speed();
1753 if (current_transport_speed >= 0.0f) {
1756 _session->request_transport_speed (-1.0f);
1759 _session->request_transport_speed (-4.0f);
1762 _session->request_transport_speed (-0.5f);
1767 _session->request_transport_speed (current_transport_speed * 1.5f);
1773 ARDOUR_UI::transport_forward (int option)
1775 float current_transport_speed;
1778 current_transport_speed = _session->transport_speed();
1780 if (current_transport_speed <= 0.0f) {
1783 _session->request_transport_speed (1.0f);
1786 _session->request_transport_speed (4.0f);
1789 _session->request_transport_speed (0.5f);
1794 _session->request_transport_speed (current_transport_speed * 1.5f);
1801 ARDOUR_UI::toggle_record_enable (uint32_t rid)
1803 if (_session == 0) {
1807 boost::shared_ptr<Route> r;
1809 if ((r = _session->route_by_remote_id (rid)) != 0) {
1813 if ((t = dynamic_cast<Track*>(r.get())) != 0) {
1814 t->set_record_enabled (!t->record_enabled(), this);
1817 if (_session == 0) {
1823 ARDOUR_UI::map_transport_state ()
1825 ENSURE_GUI_THREAD (*this, &ARDOUR_UI::map_transport_state)
1828 auto_loop_button.set_visual_state (0);
1829 play_selection_button.set_visual_state (0);
1830 roll_button.set_visual_state (0);
1831 stop_button.set_visual_state (1);
1835 float sp = _session->transport_speed();
1838 shuttle_fract = SHUTTLE_FRACT_SPEED1; /* speed = 1.0, believe it or not */
1839 shuttle_box.queue_draw ();
1840 } else if (sp == 0.0f) {
1842 shuttle_box.queue_draw ();
1843 update_disk_space ();
1850 if (_session->get_play_range()) {
1852 play_selection_button.set_visual_state (1);
1853 roll_button.set_visual_state (0);
1854 auto_loop_button.set_visual_state (0);
1856 } else if (_session->get_play_loop ()) {
1858 auto_loop_button.set_visual_state (1);
1859 play_selection_button.set_visual_state (0);
1860 roll_button.set_visual_state (0);
1864 roll_button.set_visual_state (1);
1865 play_selection_button.set_visual_state (0);
1866 auto_loop_button.set_visual_state (0);
1869 if (join_play_range_button.get_active()) {
1870 /* light up both roll and play-selection if they are joined */
1871 roll_button.set_visual_state (1);
1872 play_selection_button.set_visual_state (1);
1875 stop_button.set_visual_state (0);
1879 stop_button.set_visual_state (1);
1880 roll_button.set_visual_state (0);
1881 play_selection_button.set_visual_state (0);
1882 auto_loop_button.set_visual_state (0);
1887 ARDOUR_UI::engine_stopped ()
1889 ENSURE_GUI_THREAD (*this, &ARDOUR_UI::engine_stopped)
1890 ActionManager::set_sensitive (ActionManager::jack_sensitive_actions, false);
1891 ActionManager::set_sensitive (ActionManager::jack_opposite_sensitive_actions, true);
1895 ARDOUR_UI::engine_running ()
1897 ENSURE_GUI_THREAD (*this, &ARDOUR_UI::engine_running)
1898 ActionManager::set_sensitive (ActionManager::jack_sensitive_actions, true);
1899 ActionManager::set_sensitive (ActionManager::jack_opposite_sensitive_actions, false);
1901 Glib::RefPtr<Action> action;
1902 const char* action_name = 0;
1904 switch (engine->frames_per_cycle()) {
1906 action_name = X_("JACKLatency32");
1909 action_name = X_("JACKLatency64");
1912 action_name = X_("JACKLatency128");
1915 action_name = X_("JACKLatency512");
1918 action_name = X_("JACKLatency1024");
1921 action_name = X_("JACKLatency2048");
1924 action_name = X_("JACKLatency4096");
1927 action_name = X_("JACKLatency8192");
1930 /* XXX can we do anything useful ? */
1936 action = ActionManager::get_action (X_("JACK"), action_name);
1939 Glib::RefPtr<RadioAction> ract = Glib::RefPtr<RadioAction>::cast_dynamic (action);
1940 ract->set_active ();
1946 ARDOUR_UI::engine_halted (const char* reason, bool free_reason)
1948 if (!Gtkmm2ext::UI::instance()->caller_is_ui_thread()) {
1949 /* we can't rely on the original string continuing to exist when we are called
1950 again in the GUI thread, so make a copy and note that we need to
1953 char *copy = strdup (reason);
1954 Gtkmm2ext::UI::instance()->call_slot (invalidator (*this), boost::bind (&ARDOUR_UI::engine_halted, this, copy, true));
1958 ActionManager::set_sensitive (ActionManager::jack_sensitive_actions, false);
1959 ActionManager::set_sensitive (ActionManager::jack_opposite_sensitive_actions, true);
1961 update_sample_rate (0);
1965 /* if the reason is a non-empty string, it means that the backend was shutdown
1966 rather than just Ardour.
1969 if (strlen (reason)) {
1970 msgstr = string_compose (_("The audio backend (JACK) was shutdown because:\n\n%1"), reason);
1972 msgstr = string_compose (_("\
1973 JACK has either been shutdown or it\n\
1974 disconnected %1 because %1\n\
1975 was not fast enough. Try to restart\n\
1976 JACK, reconnect and save the session."), PROGRAM_NAME);
1979 MessageDialog msg (*editor, msgstr);
1984 free ((char*) reason);
1989 ARDOUR_UI::do_engine_start ()
1997 error << _("Unable to start the session running")
2007 ARDOUR_UI::setup_theme ()
2009 theme_manager->setup_theme();
2013 ARDOUR_UI::update_clocks ()
2015 if (!editor || !editor->dragging_playhead()) {
2016 Clock (_session->audible_frame(), false, editor->get_preferred_edit_position()); /* EMIT_SIGNAL */
2021 ARDOUR_UI::start_clocking ()
2023 clock_signal_connection = RapidScreenUpdate.connect (sigc::mem_fun(*this, &ARDOUR_UI::update_clocks));
2027 ARDOUR_UI::stop_clocking ()
2029 clock_signal_connection.disconnect ();
2033 ARDOUR_UI::toggle_clocking ()
2036 if (clock_button.get_active()) {
2045 ARDOUR_UI::_blink (void *arg)
2048 ((ARDOUR_UI *) arg)->blink ();
2055 Blink (blink_on = !blink_on); /* EMIT_SIGNAL */
2059 ARDOUR_UI::start_blinking ()
2061 /* Start the blink signal. Everybody with a blinking widget
2062 uses Blink to drive the widget's state.
2065 if (blink_timeout_tag < 0) {
2067 blink_timeout_tag = g_timeout_add (240, _blink, this);
2072 ARDOUR_UI::stop_blinking ()
2074 if (blink_timeout_tag >= 0) {
2075 g_source_remove (blink_timeout_tag);
2076 blink_timeout_tag = -1;
2081 /** Ask the user for the name of a new shapshot and then take it.
2085 ARDOUR_UI::snapshot_session (bool switch_to_it)
2087 ArdourPrompter prompter (true);
2090 prompter.set_name ("Prompter");
2091 prompter.add_button (Gtk::Stock::SAVE, Gtk::RESPONSE_ACCEPT);
2092 prompter.set_title (_("Take Snapshot"));
2093 prompter.set_title (_("Take Snapshot"));
2094 prompter.set_prompt (_("Name of new snapshot"));
2096 if (!switch_to_it) {
2099 struct tm local_time;
2102 localtime_r (&n, &local_time);
2103 strftime (timebuf, sizeof(timebuf), "%FT%T", &local_time);
2104 prompter.set_initial_text (timebuf);
2108 switch (prompter.run()) {
2109 case RESPONSE_ACCEPT:
2111 prompter.get_result (snapname);
2113 bool do_save = (snapname.length() != 0);
2116 if (snapname.find ('/') != string::npos) {
2117 MessageDialog msg (_("To ensure compatibility with various systems\n"
2118 "snapshot names may not contain a '/' character"));
2122 if (snapname.find ('\\') != string::npos) {
2123 MessageDialog msg (_("To ensure compatibility with various systems\n"
2124 "snapshot names may not contain a '\\' character"));
2130 vector<sys::path> p;
2131 get_state_files_in_directory (_session->session_directory().root_path(), p);
2132 vector<string> n = get_file_names_no_extension (p);
2133 if (find (n.begin(), n.end(), snapname) != n.end()) {
2135 ArdourDialog confirm (_("Confirm Snapshot Overwrite"), true);
2136 Label m (_("A snapshot already exists with that name. Do you want to overwrite it?"));
2137 confirm.get_vbox()->pack_start (m, true, true);
2138 confirm.add_button (Gtk::Stock::CANCEL, Gtk::RESPONSE_CANCEL);
2139 confirm.add_button (_("Overwrite"), Gtk::RESPONSE_ACCEPT);
2140 confirm.show_all ();
2141 switch (confirm.run()) {
2142 case RESPONSE_CANCEL:
2148 save_state (snapname, switch_to_it);
2159 ARDOUR_UI::save_state (const string & name, bool switch_to_it)
2161 XMLNode* node = new XMLNode (X_("UI"));
2163 for (list<WindowProxyBase*>::iterator i = _window_proxies.begin(); i != _window_proxies.end(); ++i) {
2164 if (!(*i)->rc_configured()) {
2165 node->add_child_nocopy (*((*i)->get_state ()));
2169 _session->add_extra_xml (*node);
2171 save_state_canfail (name, switch_to_it);
2175 ARDOUR_UI::save_state_canfail (string name, bool switch_to_it)
2180 if (name.length() == 0) {
2181 name = _session->snap_name();
2184 if ((ret = _session->save_state (name, false, switch_to_it)) != 0) {
2188 cerr << "SS canfail\n";
2189 save_ardour_state (); /* XXX cannot fail? yeah, right ... */
2194 ARDOUR_UI::primary_clock_value_changed ()
2197 _session->request_locate (primary_clock.current_time ());
2202 ARDOUR_UI::big_clock_value_changed ()
2205 _session->request_locate (big_clock.current_time ());
2210 ARDOUR_UI::secondary_clock_value_changed ()
2213 _session->request_locate (secondary_clock.current_time ());
2218 ARDOUR_UI::transport_rec_enable_blink (bool onoff)
2220 if (_session == 0) {
2224 if (_session->step_editing()) {
2228 Session::RecordState const r = _session->record_status ();
2229 bool const h = _session->have_rec_enabled_track ();
2231 if (r == Session::Enabled || (r == Session::Recording && !h)) {
2233 rec_button.set_visual_state (2);
2235 rec_button.set_visual_state (0);
2237 } else if (r == Session::Recording && h) {
2238 rec_button.set_visual_state (1);
2240 rec_button.set_visual_state (0);
2245 ARDOUR_UI::save_template ()
2247 ArdourPrompter prompter (true);
2250 if (!check_audioengine()) {
2254 prompter.set_name (X_("Prompter"));
2255 prompter.set_title (_("Save Template"));
2256 prompter.set_prompt (_("Name for template:"));
2257 prompter.set_initial_text(_session->name() + _("-template"));
2258 prompter.add_button (Gtk::Stock::SAVE, Gtk::RESPONSE_ACCEPT);
2260 switch (prompter.run()) {
2261 case RESPONSE_ACCEPT:
2262 prompter.get_result (name);
2264 if (name.length()) {
2265 _session->save_template (name);
2275 ARDOUR_UI::edit_metadata ()
2277 SessionMetadataEditor dialog;
2278 dialog.set_session (_session);
2279 editor->ensure_float (dialog);
2284 ARDOUR_UI::import_metadata ()
2286 SessionMetadataImporter dialog;
2287 dialog.set_session (_session);
2288 editor->ensure_float (dialog);
2293 ARDOUR_UI::fontconfig_dialog ()
2296 /* X11 users will always have fontconfig info around, but new GTK-OSX users
2297 may not and it can take a while to build it. Warn them.
2300 std::string fontconfig = Glib::build_filename (Glib::get_home_dir(), ".fontconfig");
2302 if (!Glib::file_test (fontconfig, Glib::FILE_TEST_EXISTS|Glib::FILE_TEST_IS_DIR)) {
2303 MessageDialog msg (*_startup,
2304 string_compose (_("Welcome to %1.\n\n"
2305 "The program will take a bit longer to start up\n"
2306 "while the system fonts are checked.\n\n"
2307 "This will only be done once, and you will\n"
2308 "not see this message again\n"), PROGRAM_NAME),
2321 ARDOUR_UI::parse_cmdline_path (const std::string& cmdline_path, std::string& session_name, std::string& session_path, bool& existing_session)
2323 existing_session = false;
2325 if (Glib::file_test (cmdline_path, Glib::FILE_TEST_IS_DIR)) {
2326 session_path = cmdline_path;
2327 existing_session = true;
2328 } else if (Glib::file_test (cmdline_path, Glib::FILE_TEST_IS_REGULAR)) {
2329 session_path = Glib::path_get_dirname (string (cmdline_path));
2330 existing_session = true;
2332 /* it doesn't exist, assume the best */
2333 session_path = Glib::path_get_dirname (string (cmdline_path));
2336 session_name = basename_nosuffix (string (cmdline_path));
2340 ARDOUR_UI::load_cmdline_session (const std::string& session_name, const std::string& session_path, bool& existing_session)
2342 /* when this is called, the backend audio system must be running */
2344 /* the main idea here is to deal with the fact that a cmdline argument for the session
2345 can be interpreted in different ways - it could be a directory or a file, and before
2346 we load, we need to know both the session directory and the snapshot (statefile) within it
2347 that we are supposed to use.
2350 if (session_name.length() == 0 || session_path.length() == 0) {
2354 if (Glib::file_test (session_path, Glib::FILE_TEST_IS_DIR)) {
2356 std::string predicted_session_file;
2358 predicted_session_file = session_path;
2359 predicted_session_file += '/';
2360 predicted_session_file += session_name;
2361 predicted_session_file += ARDOUR::statefile_suffix;
2363 if (Glib::file_test (predicted_session_file, Glib::FILE_TEST_EXISTS)) {
2364 existing_session = true;
2367 } else if (Glib::file_test (session_path, Glib::FILE_TEST_EXISTS)) {
2369 if (session_path.find (ARDOUR::statefile_suffix) == session_path.length() - 7) {
2370 /* existing .ardour file */
2371 existing_session = true;
2375 existing_session = false;
2378 /* lets just try to load it */
2380 if (create_engine ()) {
2381 backend_audio_error (false, _startup);
2385 return load_session (session_path, session_name);
2389 ARDOUR_UI::ask_about_loading_existing_session (const std::string& session_path)
2391 std::string str = string_compose (_("This session\n%1\nalready exists. Do you want to open it?"), session_path);
2393 MessageDialog msg (str,
2395 Gtk::MESSAGE_WARNING,
2396 Gtk::BUTTONS_YES_NO,
2400 msg.set_name (X_("OpenExistingDialog"));
2401 msg.set_title (_("Open Existing Session"));
2402 msg.set_wmclass (X_("existing_session"), PROGRAM_NAME);
2403 msg.set_position (Gtk::WIN_POS_MOUSE);
2406 switch (msg.run()) {
2415 ARDOUR_UI::build_session_from_nsd (const std::string& session_path, const std::string& session_name)
2417 BusProfile bus_profile;
2419 if (Profile->get_sae()) {
2421 bus_profile.master_out_channels = 2;
2422 bus_profile.input_ac = AutoConnectPhysical;
2423 bus_profile.output_ac = AutoConnectMaster;
2424 bus_profile.requested_physical_in = 0; // use all available
2425 bus_profile.requested_physical_out = 0; // use all available
2429 /* get settings from advanced section of NSD */
2431 if (_startup->create_master_bus()) {
2432 bus_profile.master_out_channels = (uint32_t) _startup->master_channel_count();
2434 bus_profile.master_out_channels = 0;
2437 if (_startup->connect_inputs()) {
2438 bus_profile.input_ac = AutoConnectPhysical;
2440 bus_profile.input_ac = AutoConnectOption (0);
2443 /// @todo some minor tweaks.
2445 bus_profile.output_ac = AutoConnectOption (0);
2447 if (_startup->connect_outputs ()) {
2448 if (_startup->connect_outs_to_master()) {
2449 bus_profile.output_ac = AutoConnectMaster;
2450 } else if (_startup->connect_outs_to_physical()) {
2451 bus_profile.output_ac = AutoConnectPhysical;
2455 bus_profile.requested_physical_in = (uint32_t) _startup->input_limit_count();
2456 bus_profile.requested_physical_out = (uint32_t) _startup->output_limit_count();
2459 if (build_session (session_path, session_name, bus_profile)) {
2467 ARDOUR_UI::idle_load (const std::string& path)
2470 if (Glib::file_test (path, Glib::FILE_TEST_IS_DIR)) {
2471 /* /path/to/foo => /path/to/foo, foo */
2472 load_session (path, basename_nosuffix (path));
2474 /* /path/to/foo/foo.ardour => /path/to/foo, foo */
2475 load_session (Glib::path_get_dirname (path), basename_nosuffix (path));
2479 ARDOUR_COMMAND_LINE::session_name = path;
2482 * new_session_dialog doens't exist in A3
2483 * Try to remove all references to it to
2484 * see if it will compile. NOTE: this will
2485 * likely cause a runtime issue is my somewhat
2489 //if (new_session_dialog) {
2492 /* make it break out of Dialog::run() and
2496 //new_session_dialog->response (1);
2502 ARDOUR_UI::end_loading_messages ()
2508 ARDOUR_UI::loading_message (const std::string& /*msg*/)
2511 // splash->message (msg);
2515 /** @param quit_on_cancel true if exit() should be called if the user clicks `cancel' in the new session dialog */
2517 ARDOUR_UI::get_session_parameters (bool quit_on_cancel, bool should_be_new, string load_template)
2519 string session_name;
2520 string session_path;
2521 string template_name;
2523 bool likely_new = false;
2525 if (! load_template.empty()) {
2526 should_be_new = true;
2527 template_name = load_template;
2532 if (!should_be_new && !ARDOUR_COMMAND_LINE::session_name.empty()) {
2534 /* if they named a specific statefile, use it, otherwise they are
2535 just giving a session folder, and we want to use it as is
2536 to find the session.
2539 if (ARDOUR_COMMAND_LINE::session_name.find (statefile_suffix) != string::npos) {
2540 session_path = Glib::path_get_dirname (ARDOUR_COMMAND_LINE::session_name);
2542 session_path = ARDOUR_COMMAND_LINE::session_name;
2545 session_name = Glib::path_get_basename (ARDOUR_COMMAND_LINE::session_name);
2549 bool const apply = run_startup (should_be_new, load_template);
2552 if (quit_on_cancel) {
2559 /* if we run the startup dialog again, offer more than just "new session" */
2561 should_be_new = false;
2563 session_name = _startup->session_name (likely_new);
2565 /* this shouldn't happen, but we catch it just in case it does */
2567 if (session_name.empty()) {
2571 if (_startup->use_session_template()) {
2572 template_name = _startup->session_template_name();
2573 _session_is_new = true;
2576 if (session_name[0] == G_DIR_SEPARATOR ||
2577 (session_name.length() > 2 && session_name[0] == '.' && session_name[1] == G_DIR_SEPARATOR) ||
2578 (session_name.length() > 3 && session_name[0] == '.' && session_name[1] == '.' && session_name[2] == G_DIR_SEPARATOR)) {
2580 /* absolute path or cwd-relative path specified for session name: infer session folder
2581 from what was given.
2584 session_path = Glib::path_get_dirname (session_name);
2585 session_name = Glib::path_get_basename (session_name);
2589 session_path = _startup->session_folder();
2591 if (session_name.find ('/') != string::npos) {
2592 MessageDialog msg (*_startup, _("To ensure compatibility with various systems\n"
2593 "session names may not contain a '/' character"));
2595 ARDOUR_COMMAND_LINE::session_name = ""; // cancel that
2599 if (session_name.find ('\\') != string::npos) {
2600 MessageDialog msg (*_startup, _("To ensure compatibility with various systems\n"
2601 "session names may not contain a '\\' character"));
2603 ARDOUR_COMMAND_LINE::session_name = ""; // cancel that
2609 if (create_engine ()) {
2613 if (Glib::file_test (session_path, Glib::FileTest (G_FILE_TEST_EXISTS | G_FILE_TEST_IS_DIR))) {
2617 std::string existing = Glib::build_filename (session_path, session_name);
2619 if (!ask_about_loading_existing_session (existing)) {
2620 ARDOUR_COMMAND_LINE::session_name = ""; // cancel that
2625 _session_is_new = false;
2630 MessageDialog msg (string_compose (_("There is no existing session at \"%1\""), session_path));
2632 ARDOUR_COMMAND_LINE::session_name = ""; // cancel that
2636 if (session_name.find ('/') != std::string::npos) {
2637 MessageDialog msg (*_startup, _("To ensure compatibility with various systems\n"
2638 "session names may not contain a '/' character"));
2640 ARDOUR_COMMAND_LINE::session_name = ""; // cancel that
2644 if (session_name.find ('\\') != std::string::npos) {
2645 MessageDialog msg (*_startup, _("To ensure compatibility with various systems\n"
2646 "session names may not contain a '\\' character"));
2648 ARDOUR_COMMAND_LINE::session_name = ""; // cancel that
2652 _session_is_new = true;
2655 if (likely_new && template_name.empty()) {
2657 ret = build_session_from_nsd (session_path, session_name);
2661 ret = load_session (session_path, session_name, template_name);
2664 /* not connected to the AudioEngine, so quit to avoid an infinite loop */
2668 if (!ARDOUR_COMMAND_LINE::immediate_save.empty()) {
2669 _session->save_state (ARDOUR_COMMAND_LINE::immediate_save, false);
2679 ARDOUR_UI::close_session()
2681 if (!check_audioengine()) {
2685 if (unload_session (true)) {
2689 ARDOUR_COMMAND_LINE::session_name = "";
2691 if (get_session_parameters (true, false)) {
2695 goto_editor_window ();
2698 /** @return -2 if the load failed because we are not connected to the AudioEngine */
2700 ARDOUR_UI::load_session (const std::string& path, const std::string& snap_name, std::string mix_template)
2702 Session *new_session;
2706 session_loaded = false;
2708 if (!check_audioengine()) {
2712 unload_status = unload_session ();
2714 if (unload_status < 0) {
2716 } else if (unload_status > 0) {
2721 loading_message (string_compose (_("Please wait while %1 loads your session"), PROGRAM_NAME));
2724 new_session = new Session (*engine, path, snap_name, 0, mix_template);
2727 /* this one is special */
2729 catch (AudioEngine::PortRegistrationFailure& err) {
2731 MessageDialog msg (err.what(),
2734 Gtk::BUTTONS_CLOSE);
2736 msg.set_title (_("Port Registration Error"));
2737 msg.set_secondary_text (_("Click the Close button to try again."));
2738 msg.set_position (Gtk::WIN_POS_CENTER);
2742 int response = msg.run ();
2747 case RESPONSE_CANCEL:
2757 MessageDialog msg (string_compose(_("Session \"%1 (snapshot %2)\" did not load successfully"),path, snap_name),
2762 msg.set_title (_("Loading Error"));
2763 msg.set_secondary_text (_("Click the Refresh button to try again."));
2764 msg.add_button (Stock::REFRESH, 1);
2765 msg.set_position (Gtk::WIN_POS_CENTER);
2769 int response = msg.run ();
2784 list<string> const u = new_session->unknown_processors ();
2786 MissingPluginDialog d (_session, u);
2791 /* Now the session been created, add the transport controls */
2792 new_session->add_controllable(roll_controllable);
2793 new_session->add_controllable(stop_controllable);
2794 new_session->add_controllable(goto_start_controllable);
2795 new_session->add_controllable(goto_end_controllable);
2796 new_session->add_controllable(auto_loop_controllable);
2797 new_session->add_controllable(play_selection_controllable);
2798 new_session->add_controllable(rec_controllable);
2800 set_session (new_session);
2802 session_loaded = true;
2804 goto_editor_window ();
2807 _session->set_clean ();
2818 ARDOUR_UI::build_session (const std::string& path, const std::string& snap_name, BusProfile& bus_profile)
2820 Session *new_session;
2823 if (!check_audioengine()) {
2827 session_loaded = false;
2829 x = unload_session ();
2837 _session_is_new = true;
2840 new_session = new Session (*engine, path, snap_name, &bus_profile);
2845 MessageDialog msg (string_compose(_("Could not create session in \"%1\""), path));
2851 /* Give the new session the default GUI state, if such things exist */
2854 n = Config->instant_xml (X_("Editor"));
2856 new_session->add_instant_xml (*n, false);
2858 n = Config->instant_xml (X_("Mixer"));
2860 new_session->add_instant_xml (*n, false);
2863 /* Put the playhead at 0 and scroll fully left */
2864 n = new_session->instant_xml (X_("Editor"));
2866 n->add_property (X_("playhead"), X_("0"));
2867 n->add_property (X_("left-frame"), X_("0"));
2870 set_session (new_session);
2872 session_loaded = true;
2874 new_session->save_state(new_session->name());
2880 ARDOUR_UI::launch_chat ()
2883 open_uri("http://webchat.freenode.net/?channels=ardour-osx");
2885 open_uri("http://webchat.freenode.net/?channels=ardour");
2890 ARDOUR_UI::show_about ()
2894 about->signal_response().connect(sigc::mem_fun (*this, &ARDOUR_UI::about_signal_response) );
2897 about->set_transient_for(*editor);
2902 ARDOUR_UI::launch_manual ()
2904 PBD::open_uri("http://ardour.org/flossmanual");
2908 ARDOUR_UI::launch_reference ()
2910 PBD::open_uri("http://ardour.org/refmanual");
2914 ARDOUR_UI::hide_about ()
2917 about->get_window()->set_cursor ();
2923 ARDOUR_UI::about_signal_response (int /*response*/)
2929 ARDOUR_UI::show_splash ()
2933 splash = new Splash;
2941 splash->queue_draw ();
2942 splash->get_window()->process_updates (true);
2947 ARDOUR_UI::hide_splash ()
2955 ARDOUR_UI::display_cleanup_results (ARDOUR::CleanupReport& rep, const gchar* list_title,
2956 const string& plural_msg, const string& singular_msg)
2960 removed = rep.paths.size();
2963 MessageDialog msgd (*editor,
2964 _("No files were ready for cleanup"),
2967 (Gtk::ButtonsType)(Gtk::BUTTONS_OK) );
2968 msgd.set_secondary_text (_("If this seems suprising, \n\
2969 check for any existing snapshots.\n\
2970 These may still include regions that\n\
2971 require some unused files to continue to exist."));
2977 ArdourDialog results (_("Clean-up"), true, false);
2979 struct CleanupResultsModelColumns : public Gtk::TreeModel::ColumnRecord {
2980 CleanupResultsModelColumns() {
2984 Gtk::TreeModelColumn<std::string> visible_name;
2985 Gtk::TreeModelColumn<std::string> fullpath;
2989 CleanupResultsModelColumns results_columns;
2990 Glib::RefPtr<Gtk::ListStore> results_model;
2991 Gtk::TreeView results_display;
2993 results_model = ListStore::create (results_columns);
2994 results_display.set_model (results_model);
2995 results_display.append_column (list_title, results_columns.visible_name);
2997 results_display.set_name ("CleanupResultsList");
2998 results_display.set_headers_visible (true);
2999 results_display.set_headers_clickable (false);
3000 results_display.set_reorderable (false);
3002 Gtk::ScrolledWindow list_scroller;
3005 Gtk::HBox dhbox; // the hbox for the image and text
3006 Gtk::HBox ddhbox; // the hbox we eventually pack into the dialog's vbox
3007 Gtk::Image* dimage = manage (new Gtk::Image(Stock::DIALOG_INFO, Gtk::ICON_SIZE_DIALOG));
3009 dimage->set_alignment(ALIGN_LEFT, ALIGN_TOP);
3011 const string dead_directory = _session->session_directory().dead_path().to_string();
3014 %1 - number of files removed
3015 %2 - location of "dead"
3016 %3 - size of files affected
3017 %4 - prefix for "bytes" to produce sensible results (e.g. mega, kilo, giga)
3020 const char* bprefix;
3021 double space_adjusted = 0;
3023 if (rep.space < 1000) {
3025 space_adjusted = rep.space;
3026 } else if (rep.space < 1000000) {
3027 bprefix = X_("kilo");
3028 space_adjusted = truncf((float)rep.space / 1000.0);
3029 } else if (rep.space < 1000000 * 1000) {
3030 bprefix = X_("mega");
3031 space_adjusted = truncf((float)rep.space / (1000.0 * 1000.0));
3033 bprefix = X_("giga");
3034 space_adjusted = truncf((float)rep.space / (1000.0 * 1000 * 1000.0));
3038 txt.set_text (string_compose (plural_msg, removed, dead_directory, space_adjusted, bprefix));
3040 txt.set_text (string_compose (singular_msg, removed, dead_directory, space_adjusted, bprefix));
3043 dhbox.pack_start (*dimage, true, false, 5);
3044 dhbox.pack_start (txt, true, false, 5);
3046 for (vector<string>::iterator i = rep.paths.begin(); i != rep.paths.end(); ++i) {
3047 TreeModel::Row row = *(results_model->append());
3048 row[results_columns.visible_name] = *i;
3049 row[results_columns.fullpath] = *i;
3052 list_scroller.add (results_display);
3053 list_scroller.set_size_request (-1, 150);
3054 list_scroller.set_policy (Gtk::POLICY_NEVER, Gtk::POLICY_AUTOMATIC);
3056 dvbox.pack_start (dhbox, true, false, 5);
3057 dvbox.pack_start (list_scroller, true, false, 5);
3058 ddhbox.pack_start (dvbox, true, false, 5);
3060 results.get_vbox()->pack_start (ddhbox, true, false, 5);
3061 results.add_button (Stock::CLOSE, RESPONSE_CLOSE);
3062 results.set_default_response (RESPONSE_CLOSE);
3063 results.set_position (Gtk::WIN_POS_MOUSE);
3065 results_display.show();
3066 list_scroller.show();
3073 //results.get_vbox()->show();
3074 results.set_resizable (false);
3081 ARDOUR_UI::cleanup ()
3083 if (_session == 0) {
3084 /* shouldn't happen: menu item is insensitive */
3089 MessageDialog checker (_("Are you sure you want to cleanup?"),
3091 Gtk::MESSAGE_QUESTION,
3092 (Gtk::ButtonsType)(Gtk::BUTTONS_NONE));
3094 checker.set_secondary_text(_("Cleanup is a destructive operation.\n\
3095 ALL undo/redo information will be lost if you cleanup.\n\
3096 Cleanup will move all unused files to a \"dead\" location."));
3098 checker.add_button (Stock::CANCEL, RESPONSE_CANCEL);
3099 checker.add_button (_("Clean Up"), RESPONSE_ACCEPT);
3100 checker.set_default_response (RESPONSE_CANCEL);
3102 checker.set_name (_("CleanupDialog"));
3103 checker.set_wmclass (X_("ardour_cleanup"), PROGRAM_NAME);
3104 checker.set_position (Gtk::WIN_POS_MOUSE);
3106 switch (checker.run()) {
3107 case RESPONSE_ACCEPT:
3113 ARDOUR::CleanupReport rep;
3115 editor->prepare_for_cleanup ();
3117 /* do not allow flush until a session is reloaded */
3119 Glib::RefPtr<Action> act = ActionManager::get_action (X_("Main"), X_("FlushWastebasket"));
3121 act->set_sensitive (false);
3124 if (_session->cleanup_sources (rep)) {
3125 editor->finish_cleanup ();
3129 editor->finish_cleanup ();
3132 display_cleanup_results (rep,
3135 The following %1 files were not in use and \n\
3136 have been moved to:\n\n\
3138 After a restart of Ardour,\n\n\
3139 Session -> Cleanup -> Flush Wastebasket\n\n\
3140 will release an additional\n\
3141 %3 %4bytes of disk space.\n"),
3143 The following file was not in use and \n \
3144 has been moved to:\n \
3146 After a restart of Ardour,\n\n\
3147 Session -> Cleanup -> Flush Wastebasket\n\n\
3148 will release an additional\n\
3149 %3 %4bytes of disk space.\n"
3155 ARDOUR_UI::flush_trash ()
3157 if (_session == 0) {
3158 /* shouldn't happen: menu item is insensitive */
3162 ARDOUR::CleanupReport rep;
3164 if (_session->cleanup_trash_sources (rep)) {
3168 display_cleanup_results (rep,
3170 _("The following %1 files were deleted from\n\
3172 releasing %3 %4bytes of disk space"),
3173 _("The following file was deleted from\n\
3175 releasing %3 %4bytes of disk space"));
3179 ARDOUR_UI::add_route (Gtk::Window* float_window)
3187 if (add_route_dialog == 0) {
3188 add_route_dialog = new AddRouteDialog (_session);
3190 add_route_dialog->set_transient_for (*float_window);
3194 if (add_route_dialog->is_visible()) {
3195 /* we're already doing this */
3199 ResponseType r = (ResponseType) add_route_dialog->run ();
3201 add_route_dialog->hide();
3204 case RESPONSE_ACCEPT:
3211 if ((count = add_route_dialog->count()) <= 0) {
3215 string template_path = add_route_dialog->track_template();
3217 if (!template_path.empty()) {
3218 _session->new_route_from_template (count, template_path);
3222 uint32_t input_chan = add_route_dialog->channels ();
3223 uint32_t output_chan;
3224 string name_template = add_route_dialog->name_template ();
3225 bool track = add_route_dialog->track ();
3226 RouteGroup* route_group = add_route_dialog->route_group ();
3228 AutoConnectOption oac = Config->get_output_auto_connect();
3230 if (oac & AutoConnectMaster) {
3231 output_chan = (_session->master_out() ? _session->master_out()->n_inputs().n_audio() : input_chan);
3233 output_chan = input_chan;
3236 /* XXX do something with name template */
3238 if (add_route_dialog->type() == ARDOUR::DataType::MIDI) {
3240 session_add_midi_track (route_group, count, name_template);
3242 MessageDialog msg (*editor,
3243 _("Sorry, MIDI Busses are not supported at this time."));
3245 //session_add_midi_bus();
3249 session_add_audio_track (input_chan, output_chan, add_route_dialog->mode(), route_group, count, name_template);
3251 session_add_audio_bus (input_chan, output_chan, route_group, count, name_template);
3257 ARDOUR_UI::mixer_settings () const
3262 node = _session->instant_xml(X_("Mixer"));
3264 node = Config->instant_xml(X_("Mixer"));
3268 node = new XMLNode (X_("Mixer"));
3275 ARDOUR_UI::editor_settings () const
3280 node = _session->instant_xml(X_("Editor"));
3282 node = Config->instant_xml(X_("Editor"));
3286 if (getenv("ARDOUR_INSTANT_XML_PATH")) {
3287 node = Config->instant_xml(getenv("ARDOUR_INSTANT_XML_PATH"));
3292 node = new XMLNode (X_("Editor"));
3299 ARDOUR_UI::keyboard_settings () const
3303 node = Config->extra_xml(X_("Keyboard"));
3306 node = new XMLNode (X_("Keyboard"));
3313 ARDOUR_UI::create_xrun_marker (framepos_t where)
3315 editor->mouse_add_new_marker (where, false, true);
3319 ARDOUR_UI::halt_on_xrun_message ()
3321 MessageDialog msg (*editor,
3322 _("Recording was stopped because your system could not keep up."));
3327 ARDOUR_UI::xrun_handler (framepos_t where)
3333 ENSURE_GUI_THREAD (*this, &ARDOUR_UI::xrun_handler, where)
3335 if (_session && Config->get_create_xrun_marker() && _session->actively_recording()) {
3336 create_xrun_marker(where);
3339 if (_session && Config->get_stop_recording_on_xrun() && _session->actively_recording()) {
3340 halt_on_xrun_message ();
3345 ARDOUR_UI::disk_overrun_handler ()
3347 ENSURE_GUI_THREAD (*this, &ARDOUR_UI::disk_overrun_handler)
3349 if (!have_disk_speed_dialog_displayed) {
3350 have_disk_speed_dialog_displayed = true;
3351 MessageDialog* msg = new MessageDialog (*editor, string_compose (_("\
3352 The disk system on your computer\n\
3353 was not able to keep up with %1.\n\
3355 Specifically, it failed to write data to disk\n\
3356 quickly enough to keep up with recording.\n"), PROGRAM_NAME));
3357 msg->signal_response().connect (sigc::bind (sigc::mem_fun (*this, &ARDOUR_UI::disk_speed_dialog_gone), msg));
3363 ARDOUR_UI::disk_underrun_handler ()
3365 ENSURE_GUI_THREAD (*this, &ARDOUR_UI::disk_underrun_handler)
3367 if (!have_disk_speed_dialog_displayed) {
3368 have_disk_speed_dialog_displayed = true;
3369 MessageDialog* msg = new MessageDialog (*editor,
3370 string_compose (_("The disk system on your computer\n\
3371 was not able to keep up with %1.\n\
3373 Specifically, it failed to read data from disk\n\
3374 quickly enough to keep up with playback.\n"), PROGRAM_NAME));
3375 msg->signal_response().connect (sigc::bind (sigc::mem_fun (*this, &ARDOUR_UI::disk_speed_dialog_gone), msg));
3381 ARDOUR_UI::disk_speed_dialog_gone (int /*ignored_response*/, MessageDialog* msg)
3383 have_disk_speed_dialog_displayed = false;
3388 ARDOUR_UI::session_dialog (std::string msg)
3390 ENSURE_GUI_THREAD (*this, &ARDOUR_UI::session_dialog, msg)
3395 d = new MessageDialog (*editor, msg, false, MESSAGE_INFO, BUTTONS_OK, true);
3397 d = new MessageDialog (msg, false, MESSAGE_INFO, BUTTONS_OK, true);
3406 ARDOUR_UI::pending_state_dialog ()
3408 HBox* hbox = new HBox();
3409 Image* image = new Image (Stock::DIALOG_QUESTION, ICON_SIZE_DIALOG);
3410 ArdourDialog dialog (_("Crash Recovery"), true);
3412 This session appears to have been in\n\
3413 middle of recording when ardour or\n\
3414 the computer was shutdown.\n\
3416 Ardour can recover any captured audio for\n\
3417 you, or it can ignore it. Please decide\n\
3418 what you would like to do.\n"));
3419 image->set_alignment(ALIGN_CENTER, ALIGN_TOP);
3420 hbox->pack_start (*image, PACK_EXPAND_WIDGET, 12);
3421 hbox->pack_end (message, PACK_EXPAND_PADDING, 12);
3422 dialog.get_vbox()->pack_start(*hbox, PACK_EXPAND_PADDING, 6);
3423 dialog.add_button (_("Ignore crash data"), RESPONSE_REJECT);
3424 dialog.add_button (_("Recover from crash"), RESPONSE_ACCEPT);
3425 dialog.set_default_response (RESPONSE_ACCEPT);
3426 dialog.set_position (WIN_POS_CENTER);
3431 switch (dialog.run ()) {
3432 case RESPONSE_ACCEPT:
3440 ARDOUR_UI::sr_mismatch_dialog (framecnt_t desired, framecnt_t actual)
3442 HBox* hbox = new HBox();
3443 Image* image = new Image (Stock::DIALOG_QUESTION, ICON_SIZE_DIALOG);
3444 ArdourDialog dialog (_("Sample Rate Mismatch"), true);
3445 Label message (string_compose (_("\
3446 This session was created with a sample rate of %1 Hz\n\
3448 The audioengine is currently running at %2 Hz\n"), desired, actual));
3450 image->set_alignment(ALIGN_CENTER, ALIGN_TOP);
3451 hbox->pack_start (*image, PACK_EXPAND_WIDGET, 12);
3452 hbox->pack_end (message, PACK_EXPAND_PADDING, 12);
3453 dialog.get_vbox()->pack_start(*hbox, PACK_EXPAND_PADDING, 6);
3454 dialog.add_button (_("Do not load session"), RESPONSE_REJECT);
3455 dialog.add_button (_("Load session anyway"), RESPONSE_ACCEPT);
3456 dialog.set_default_response (RESPONSE_ACCEPT);
3457 dialog.set_position (WIN_POS_CENTER);
3462 switch (dialog.run ()) {
3463 case RESPONSE_ACCEPT:
3472 ARDOUR_UI::disconnect_from_jack ()
3475 if( engine->disconnect_from_jack ()) {
3476 MessageDialog msg (*editor, _("Could not disconnect from JACK"));
3480 update_sample_rate (0);
3485 ARDOUR_UI::reconnect_to_jack ()
3488 if (engine->reconnect_to_jack ()) {
3489 MessageDialog msg (*editor, _("Could not reconnect to JACK"));
3493 update_sample_rate (0);
3498 ARDOUR_UI::use_config ()
3500 XMLNode* node = Config->extra_xml (X_("TransportControllables"));
3502 set_transport_controllable_state (*node);
3507 ARDOUR_UI::update_transport_clocks (framepos_t pos)
3509 if (Config->get_primary_clock_delta_edit_cursor()) {
3510 primary_clock.set (pos, false, editor->get_preferred_edit_position(), 1);
3512 primary_clock.set (pos, 0, true);
3515 if (Config->get_secondary_clock_delta_edit_cursor()) {
3516 secondary_clock.set (pos, false, editor->get_preferred_edit_position(), 2);
3518 secondary_clock.set (pos);
3521 if (big_clock_window->get()) {
3522 big_clock.set (pos);
3528 ARDOUR_UI::step_edit_status_change (bool yn)
3530 // XXX should really store pre-step edit status of things
3531 // we make insensitive
3534 rec_button.set_visual_state (3);
3535 rec_button.set_sensitive (false);
3537 rec_button.set_visual_state (0);
3538 rec_button.set_sensitive (true);
3543 ARDOUR_UI::record_state_changed ()
3545 ENSURE_GUI_THREAD (*this, &ARDOUR_UI::record_state_changed);
3547 if (!_session || !big_clock_window->get()) {
3548 /* why bother - the clock isn't visible */
3552 Session::RecordState const r = _session->record_status ();
3553 bool const h = _session->have_rec_enabled_track ();
3555 if (r == Session::Recording && h) {
3556 big_clock.set_widget_name ("BigClockRecording");
3558 big_clock.set_widget_name ("BigClockNonRecording");
3563 ARDOUR_UI::first_idle ()
3566 _session->allow_auto_play (true);
3570 editor->first_idle();
3573 Keyboard::set_can_save_keybindings (true);
3578 ARDOUR_UI::store_clock_modes ()
3580 XMLNode* node = new XMLNode(X_("ClockModes"));
3582 for (vector<AudioClock*>::iterator x = AudioClock::clocks.begin(); x != AudioClock::clocks.end(); ++x) {
3583 node->add_property ((*x)->name().c_str(), enum_2_string ((*x)->mode()));
3586 _session->add_extra_xml (*node);
3587 _session->set_dirty ();
3592 ARDOUR_UI::TransportControllable::TransportControllable (std::string name, ARDOUR_UI& u, ToggleType tp)
3593 : Controllable (name), ui (u), type(tp)
3599 ARDOUR_UI::TransportControllable::set_value (double val)
3601 if (type == ShuttleControl) {
3608 fract = -((0.5 - val)/0.5);
3610 fract = ((val - 0.5)/0.5);
3614 ui.set_shuttle_fract (fract);
3619 /* do nothing: these are radio-style actions */
3623 const char *action = 0;
3627 action = X_("Roll");
3630 action = X_("Stop");
3633 action = X_("Goto Start");
3636 action = X_("Goto End");
3639 action = X_("Loop");
3642 action = X_("Play Selection");
3645 action = X_("Record");
3655 Glib::RefPtr<Action> act = ActionManager::get_action ("Transport", action);
3663 ARDOUR_UI::TransportControllable::get_value (void) const
3682 case ShuttleControl:
3692 ARDOUR_UI::TransportControllable::set_id (const string& str)
3698 ARDOUR_UI::setup_profile ()
3700 if (gdk_screen_width() < 1200) {
3701 Profile->set_small_screen ();
3705 if (getenv ("ARDOUR_SAE")) {
3706 Profile->set_sae ();
3707 Profile->set_single_package ();
3712 ARDOUR_UI::toggle_translations ()
3714 using namespace Glib;
3716 RefPtr<Action> act = ActionManager::get_action (X_("Main"), X_("EnableTranslation"));
3718 RefPtr<ToggleAction> ract = RefPtr<ToggleAction>::cast_dynamic (act);
3721 string i18n_killer = ARDOUR::translation_kill_path();
3723 bool already_enabled = !ARDOUR::translations_are_disabled ();
3725 if (ract->get_active ()) {
3726 /* we don't care about errors */
3727 int fd = ::open (i18n_killer.c_str(), O_RDONLY|O_CREAT, 0644);
3730 /* we don't care about errors */
3731 unlink (i18n_killer.c_str());
3734 if (already_enabled != ract->get_active()) {
3735 MessageDialog win (already_enabled ? _("Translations disabled") : _("Translations enabled"),
3737 Gtk::MESSAGE_WARNING,
3739 win.set_secondary_text (string_compose (_("You must restart %1 for this to take effect."), PROGRAM_NAME));
3740 win.set_position (Gtk::WIN_POS_CENTER);
3748 /** Add a window proxy to our list, so that its state will be saved.
3749 * This call also causes the window to be created and opened if its
3750 * state was saved as `visible'.
3753 ARDOUR_UI::add_window_proxy (WindowProxyBase* p)
3755 _window_proxies.push_back (p);
3759 /** Remove a window proxy from our list. Must be called if a WindowProxy
3760 * is deleted, to prevent hanging pointers.
3763 ARDOUR_UI::remove_window_proxy (WindowProxyBase* p)
3765 _window_proxies.remove (p);
3769 ARDOUR_UI::missing_file (Session*s, std::string str, DataType type)
3771 MissingFileDialog dialog (s, str, type);
3776 int result = dialog.run ();
3783 return 1; // quit entire session load
3786 result = dialog.get_action ();
3792 ARDOUR_UI::ambiguous_file (std::string file, std::string path, std::vector<std::string> hits)
3794 AmbiguousFileDialog dialog (file, hits);
3800 return dialog.get_which ();