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 _("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 (
1278 _("%1 is not connected to JACK\n"
1279 "You cannot open or close sessions in this condition"),
1292 ARDOUR_UI::open_session ()
1294 if (!check_audioengine()) {
1299 /* popup selector window */
1301 if (open_session_selector == 0) {
1303 /* ardour sessions are folders */
1305 open_session_selector = new Gtk::FileChooserDialog (_("Open Session"), FILE_CHOOSER_ACTION_OPEN);
1306 open_session_selector->add_button (Gtk::Stock::CANCEL, Gtk::RESPONSE_CANCEL);
1307 open_session_selector->add_button (Gtk::Stock::OPEN, Gtk::RESPONSE_ACCEPT);
1308 open_session_selector->set_default_response(Gtk::RESPONSE_ACCEPT);
1310 FileFilter session_filter;
1311 session_filter.add_pattern ("*.ardour");
1312 session_filter.set_name (string_compose (_("%1 sessions"), PROGRAM_NAME));
1313 open_session_selector->add_filter (session_filter);
1314 open_session_selector->set_filter (session_filter);
1317 int response = open_session_selector->run();
1318 open_session_selector->hide ();
1321 case RESPONSE_ACCEPT:
1324 open_session_selector->hide();
1328 open_session_selector->hide();
1329 string session_path = open_session_selector->get_filename();
1333 if (session_path.length() > 0) {
1334 if (ARDOUR::find_session (session_path, path, name, isnew) == 0) {
1335 _session_is_new = isnew;
1336 load_session (path, name);
1343 ARDOUR_UI::session_add_midi_route (bool disk, RouteGroup* route_group, uint32_t how_many, string const & name_template)
1345 list<boost::shared_ptr<MidiTrack> > tracks;
1347 if (_session == 0) {
1348 warning << _("You cannot add a track without a session already loaded.") << endmsg;
1355 tracks = _session->new_midi_track (ARDOUR::Normal, route_group, how_many, name_template);
1357 if (tracks.size() != how_many) {
1358 if (how_many == 1) {
1359 error << _("could not create a new midi track") << endmsg;
1361 error << string_compose (_("could not create %1 new midi tracks"), how_many) << endmsg;
1365 if ((route = _session->new_midi_route ()) == 0) {
1366 error << _("could not create new midi bus") << endmsg;
1372 MessageDialog msg (*editor,
1373 string_compose (_("There are insufficient JACK ports available\n\
1374 to create a new track or bus.\n\
1375 You should save %1, exit and\n\
1376 restart JACK with more ports."), PROGRAM_NAME));
1383 ARDOUR_UI::session_add_audio_route (
1385 int32_t input_channels,
1386 int32_t output_channels,
1387 ARDOUR::TrackMode mode,
1388 RouteGroup* route_group,
1390 string const & name_template
1393 list<boost::shared_ptr<AudioTrack> > tracks;
1396 if (_session == 0) {
1397 warning << _("You cannot add a track or bus without a session already loaded.") << endmsg;
1403 tracks = _session->new_audio_track (input_channels, output_channels, mode, route_group, how_many, name_template);
1405 if (tracks.size() != how_many) {
1406 if (how_many == 1) {
1407 error << _("could not create a new audio track") << endmsg;
1409 error << string_compose (_("could only create %1 of %2 new audio %3"),
1410 tracks.size(), how_many, (track ? _("tracks") : _("busses"))) << endmsg;
1416 routes = _session->new_audio_route (input_channels, output_channels, route_group, how_many, name_template);
1418 if (routes.size() != how_many) {
1419 if (how_many == 1) {
1420 error << _("could not create a new audio track") << endmsg;
1422 error << string_compose (_("could not create %1 new audio tracks"), how_many) << endmsg;
1429 MessageDialog msg (*editor,
1430 string_compose (_("There are insufficient JACK ports available\n\
1431 to create a new track or bus.\n\
1432 You should save %1, exit and\n\
1433 restart JACK with more ports."), PROGRAM_NAME));
1440 ARDOUR_UI::do_transport_locate (framepos_t new_position, bool with_roll)
1442 framecnt_t _preroll = 0;
1445 // XXX CONFIG_CHANGE FIX - requires AnyTime handling
1446 // _preroll = _session->convert_to_frames_at (new_position, Config->get_preroll());
1448 if (new_position > _preroll) {
1449 new_position -= _preroll;
1454 _session->request_locate (new_position, with_roll);
1459 ARDOUR_UI::transport_goto_start ()
1462 _session->goto_start();
1464 /* force displayed area in editor to start no matter
1465 what "follow playhead" setting is.
1469 editor->center_screen (_session->current_start_frame ());
1475 ARDOUR_UI::transport_goto_zero ()
1478 _session->request_locate (0);
1480 /* force displayed area in editor to start no matter
1481 what "follow playhead" setting is.
1485 editor->reset_x_origin (0);
1491 ARDOUR_UI::transport_goto_wallclock ()
1493 if (_session && editor) {
1500 localtime_r (&now, &tmnow);
1502 frames = tmnow.tm_hour * (60 * 60 * _session->frame_rate());
1503 frames += tmnow.tm_min * (60 * _session->frame_rate());
1504 frames += tmnow.tm_sec * _session->frame_rate();
1506 _session->request_locate (frames, _session->transport_rolling ());
1508 /* force displayed area in editor to start no matter
1509 what "follow playhead" setting is.
1513 editor->center_screen (frames);
1519 ARDOUR_UI::transport_goto_end ()
1522 framepos_t const frame = _session->current_end_frame();
1523 _session->request_locate (frame);
1525 /* force displayed area in editor to start no matter
1526 what "follow playhead" setting is.
1530 editor->center_screen (frame);
1536 ARDOUR_UI::transport_stop ()
1542 if (_session->is_auditioning()) {
1543 _session->cancel_audition ();
1547 _session->request_stop (false, true);
1551 ARDOUR_UI::transport_stop_and_forget_capture ()
1554 _session->request_stop (true, true);
1559 ARDOUR_UI::remove_last_capture()
1562 editor->remove_last_capture();
1567 ARDOUR_UI::transport_record (bool roll)
1571 switch (_session->record_status()) {
1572 case Session::Disabled:
1573 if (_session->ntracks() == 0) {
1574 MessageDialog msg (*editor, _("Please create 1 or more track\nbefore trying to record.\nCheck the Session menu."));
1578 _session->maybe_enable_record ();
1583 case Session::Recording:
1585 _session->request_stop();
1587 _session->disable_record (false, true);
1591 case Session::Enabled:
1592 _session->disable_record (false, true);
1595 //cerr << "ARDOUR_UI::transport_record () called roll = " << roll << " _session->record_status() = " << _session->record_status() << endl;
1599 ARDOUR_UI::transport_roll ()
1605 if (_session->is_auditioning()) {
1610 if (_session->config.get_external_sync()) {
1611 switch (_session->config.get_sync_source()) {
1615 /* transport controlled by the master */
1621 bool rolling = _session->transport_rolling();
1623 if (_session->get_play_loop()) {
1624 /* XXX it is not possible to just leave seamless loop and keep
1625 playing at present (nov 4th 2009)
1627 if (!Config->get_seamless_loop()) {
1628 _session->request_play_loop (false, true);
1630 } else if (_session->get_play_range () && !join_play_range_button.get_active()) {
1631 /* stop playing a range if we currently are */
1632 _session->request_play_range (0, true);
1635 if (join_play_range_button.get_active()) {
1636 _session->request_play_range (&editor->get_selection().time, true);
1640 _session->request_transport_speed (1.0f);
1645 ARDOUR_UI::toggle_roll (bool with_abort, bool roll_out_of_bounded_mode)
1652 if (_session->is_auditioning()) {
1653 _session->cancel_audition ();
1657 if (_session->config.get_external_sync()) {
1658 switch (_session->config.get_sync_source()) {
1662 /* transport controlled by the master */
1667 bool rolling = _session->transport_rolling();
1668 bool affect_transport = true;
1670 if (rolling && roll_out_of_bounded_mode) {
1671 /* drop out of loop/range playback but leave transport rolling */
1672 if (_session->get_play_loop()) {
1673 if (Config->get_seamless_loop()) {
1674 /* the disk buffers contain copies of the loop - we can't
1675 just keep playing, so stop the transport. the user
1676 can restart as they wish.
1678 affect_transport = true;
1680 /* disk buffers are normal, so we can keep playing */
1681 affect_transport = false;
1683 _session->request_play_loop (false, true);
1684 } else if (_session->get_play_range ()) {
1685 affect_transport = false;
1686 _session->request_play_range (0, true);
1690 if (affect_transport) {
1692 _session->request_stop (with_abort, true);
1694 if (join_play_range_button.get_active()) {
1695 _session->request_play_range (&editor->get_selection().time, true);
1698 _session->request_transport_speed (1.0f);
1704 ARDOUR_UI::toggle_session_auto_loop ()
1710 if (_session->get_play_loop()) {
1712 if (_session->transport_rolling()) {
1714 Location * looploc = _session->locations()->auto_loop_location();
1717 _session->request_locate (looploc->start(), true);
1718 _session->request_play_loop (false);
1722 _session->request_play_loop (false);
1726 Location * looploc = _session->locations()->auto_loop_location();
1729 _session->request_play_loop (true);
1735 ARDOUR_UI::transport_play_selection ()
1741 editor->play_selection ();
1745 ARDOUR_UI::transport_rewind (int option)
1747 float current_transport_speed;
1750 current_transport_speed = _session->transport_speed();
1752 if (current_transport_speed >= 0.0f) {
1755 _session->request_transport_speed (-1.0f);
1758 _session->request_transport_speed (-4.0f);
1761 _session->request_transport_speed (-0.5f);
1766 _session->request_transport_speed (current_transport_speed * 1.5f);
1772 ARDOUR_UI::transport_forward (int option)
1774 float current_transport_speed;
1777 current_transport_speed = _session->transport_speed();
1779 if (current_transport_speed <= 0.0f) {
1782 _session->request_transport_speed (1.0f);
1785 _session->request_transport_speed (4.0f);
1788 _session->request_transport_speed (0.5f);
1793 _session->request_transport_speed (current_transport_speed * 1.5f);
1800 ARDOUR_UI::toggle_record_enable (uint32_t rid)
1802 if (_session == 0) {
1806 boost::shared_ptr<Route> r;
1808 if ((r = _session->route_by_remote_id (rid)) != 0) {
1812 if ((t = dynamic_cast<Track*>(r.get())) != 0) {
1813 t->set_record_enabled (!t->record_enabled(), this);
1816 if (_session == 0) {
1822 ARDOUR_UI::map_transport_state ()
1824 ENSURE_GUI_THREAD (*this, &ARDOUR_UI::map_transport_state)
1827 auto_loop_button.set_visual_state (0);
1828 play_selection_button.set_visual_state (0);
1829 roll_button.set_visual_state (0);
1830 stop_button.set_visual_state (1);
1834 float sp = _session->transport_speed();
1837 shuttle_fract = SHUTTLE_FRACT_SPEED1; /* speed = 1.0, believe it or not */
1838 shuttle_box.queue_draw ();
1839 } else if (sp == 0.0f) {
1841 shuttle_box.queue_draw ();
1842 update_disk_space ();
1849 if (_session->get_play_range()) {
1851 play_selection_button.set_visual_state (1);
1852 roll_button.set_visual_state (0);
1853 auto_loop_button.set_visual_state (0);
1855 } else if (_session->get_play_loop ()) {
1857 auto_loop_button.set_visual_state (1);
1858 play_selection_button.set_visual_state (0);
1859 roll_button.set_visual_state (0);
1863 roll_button.set_visual_state (1);
1864 play_selection_button.set_visual_state (0);
1865 auto_loop_button.set_visual_state (0);
1868 if (join_play_range_button.get_active()) {
1869 /* light up both roll and play-selection if they are joined */
1870 roll_button.set_visual_state (1);
1871 play_selection_button.set_visual_state (1);
1874 stop_button.set_visual_state (0);
1878 stop_button.set_visual_state (1);
1879 roll_button.set_visual_state (0);
1880 play_selection_button.set_visual_state (0);
1881 auto_loop_button.set_visual_state (0);
1886 ARDOUR_UI::engine_stopped ()
1888 ENSURE_GUI_THREAD (*this, &ARDOUR_UI::engine_stopped)
1889 ActionManager::set_sensitive (ActionManager::jack_sensitive_actions, false);
1890 ActionManager::set_sensitive (ActionManager::jack_opposite_sensitive_actions, true);
1894 ARDOUR_UI::engine_running ()
1896 ENSURE_GUI_THREAD (*this, &ARDOUR_UI::engine_running)
1897 ActionManager::set_sensitive (ActionManager::jack_sensitive_actions, true);
1898 ActionManager::set_sensitive (ActionManager::jack_opposite_sensitive_actions, false);
1900 Glib::RefPtr<Action> action;
1901 const char* action_name = 0;
1903 switch (engine->frames_per_cycle()) {
1905 action_name = X_("JACKLatency32");
1908 action_name = X_("JACKLatency64");
1911 action_name = X_("JACKLatency128");
1914 action_name = X_("JACKLatency512");
1917 action_name = X_("JACKLatency1024");
1920 action_name = X_("JACKLatency2048");
1923 action_name = X_("JACKLatency4096");
1926 action_name = X_("JACKLatency8192");
1929 /* XXX can we do anything useful ? */
1935 action = ActionManager::get_action (X_("JACK"), action_name);
1938 Glib::RefPtr<RadioAction> ract = Glib::RefPtr<RadioAction>::cast_dynamic (action);
1939 ract->set_active ();
1945 ARDOUR_UI::engine_halted (const char* reason, bool free_reason)
1947 if (!Gtkmm2ext::UI::instance()->caller_is_ui_thread()) {
1948 /* we can't rely on the original string continuing to exist when we are called
1949 again in the GUI thread, so make a copy and note that we need to
1952 char *copy = strdup (reason);
1953 Gtkmm2ext::UI::instance()->call_slot (invalidator (*this), boost::bind (&ARDOUR_UI::engine_halted, this, copy, true));
1957 ActionManager::set_sensitive (ActionManager::jack_sensitive_actions, false);
1958 ActionManager::set_sensitive (ActionManager::jack_opposite_sensitive_actions, true);
1960 update_sample_rate (0);
1964 /* if the reason is a non-empty string, it means that the backend was shutdown
1965 rather than just Ardour.
1968 if (strlen (reason)) {
1969 msgstr = string_compose (_("The audio backend (JACK) was shutdown because:\n\n%1"), reason);
1971 msgstr = string_compose (_("\
1972 JACK has either been shutdown or it\n\
1973 disconnected %1 because %1\n\
1974 was not fast enough. Try to restart\n\
1975 JACK, reconnect and save the session."), PROGRAM_NAME);
1978 MessageDialog msg (*editor, msgstr);
1983 free ((char*) reason);
1988 ARDOUR_UI::do_engine_start ()
1996 error << _("Unable to start the session running")
2006 ARDOUR_UI::setup_theme ()
2008 theme_manager->setup_theme();
2012 ARDOUR_UI::update_clocks ()
2014 if (!editor || !editor->dragging_playhead()) {
2015 Clock (_session->audible_frame(), false, editor->get_preferred_edit_position()); /* EMIT_SIGNAL */
2020 ARDOUR_UI::start_clocking ()
2022 clock_signal_connection = RapidScreenUpdate.connect (sigc::mem_fun(*this, &ARDOUR_UI::update_clocks));
2026 ARDOUR_UI::stop_clocking ()
2028 clock_signal_connection.disconnect ();
2032 ARDOUR_UI::toggle_clocking ()
2035 if (clock_button.get_active()) {
2044 ARDOUR_UI::_blink (void *arg)
2047 ((ARDOUR_UI *) arg)->blink ();
2054 Blink (blink_on = !blink_on); /* EMIT_SIGNAL */
2058 ARDOUR_UI::start_blinking ()
2060 /* Start the blink signal. Everybody with a blinking widget
2061 uses Blink to drive the widget's state.
2064 if (blink_timeout_tag < 0) {
2066 blink_timeout_tag = g_timeout_add (240, _blink, this);
2071 ARDOUR_UI::stop_blinking ()
2073 if (blink_timeout_tag >= 0) {
2074 g_source_remove (blink_timeout_tag);
2075 blink_timeout_tag = -1;
2080 /** Ask the user for the name of a new shapshot and then take it.
2084 ARDOUR_UI::snapshot_session (bool switch_to_it)
2086 ArdourPrompter prompter (true);
2089 prompter.set_name ("Prompter");
2090 prompter.add_button (Gtk::Stock::SAVE, Gtk::RESPONSE_ACCEPT);
2091 prompter.set_title (_("Take Snapshot"));
2092 prompter.set_title (_("Take Snapshot"));
2093 prompter.set_prompt (_("Name of new snapshot"));
2095 if (!switch_to_it) {
2098 struct tm local_time;
2101 localtime_r (&n, &local_time);
2102 strftime (timebuf, sizeof(timebuf), "%FT%T", &local_time);
2103 prompter.set_initial_text (timebuf);
2107 switch (prompter.run()) {
2108 case RESPONSE_ACCEPT:
2110 prompter.get_result (snapname);
2112 bool do_save = (snapname.length() != 0);
2115 if (snapname.find ('/') != string::npos) {
2116 MessageDialog msg (_("To ensure compatibility with various systems\n"
2117 "snapshot names may not contain a '/' character"));
2121 if (snapname.find ('\\') != string::npos) {
2122 MessageDialog msg (_("To ensure compatibility with various systems\n"
2123 "snapshot names may not contain a '\\' character"));
2129 vector<sys::path> p;
2130 get_state_files_in_directory (_session->session_directory().root_path(), p);
2131 vector<string> n = get_file_names_no_extension (p);
2132 if (find (n.begin(), n.end(), snapname) != n.end()) {
2134 ArdourDialog confirm (_("Confirm Snapshot Overwrite"), true);
2135 Label m (_("A snapshot already exists with that name. Do you want to overwrite it?"));
2136 confirm.get_vbox()->pack_start (m, true, true);
2137 confirm.add_button (Gtk::Stock::CANCEL, Gtk::RESPONSE_CANCEL);
2138 confirm.add_button (_("Overwrite"), Gtk::RESPONSE_ACCEPT);
2139 confirm.show_all ();
2140 switch (confirm.run()) {
2141 case RESPONSE_CANCEL:
2147 save_state (snapname, switch_to_it);
2158 ARDOUR_UI::save_state (const string & name, bool switch_to_it)
2160 XMLNode* node = new XMLNode (X_("UI"));
2162 for (list<WindowProxyBase*>::iterator i = _window_proxies.begin(); i != _window_proxies.end(); ++i) {
2163 if (!(*i)->rc_configured()) {
2164 node->add_child_nocopy (*((*i)->get_state ()));
2168 _session->add_extra_xml (*node);
2170 save_state_canfail (name, switch_to_it);
2174 ARDOUR_UI::save_state_canfail (string name, bool switch_to_it)
2179 if (name.length() == 0) {
2180 name = _session->snap_name();
2183 if ((ret = _session->save_state (name, false, switch_to_it)) != 0) {
2187 cerr << "SS canfail\n";
2188 save_ardour_state (); /* XXX cannot fail? yeah, right ... */
2193 ARDOUR_UI::primary_clock_value_changed ()
2196 _session->request_locate (primary_clock.current_time ());
2201 ARDOUR_UI::big_clock_value_changed ()
2204 _session->request_locate (big_clock.current_time ());
2209 ARDOUR_UI::secondary_clock_value_changed ()
2212 _session->request_locate (secondary_clock.current_time ());
2217 ARDOUR_UI::transport_rec_enable_blink (bool onoff)
2219 if (_session == 0) {
2223 if (_session->step_editing()) {
2227 Session::RecordState const r = _session->record_status ();
2228 bool const h = _session->have_rec_enabled_track ();
2230 if (r == Session::Enabled || (r == Session::Recording && !h)) {
2232 rec_button.set_visual_state (2);
2234 rec_button.set_visual_state (0);
2236 } else if (r == Session::Recording && h) {
2237 rec_button.set_visual_state (1);
2239 rec_button.set_visual_state (0);
2244 ARDOUR_UI::save_template ()
2246 ArdourPrompter prompter (true);
2249 if (!check_audioengine()) {
2253 prompter.set_name (X_("Prompter"));
2254 prompter.set_title (_("Save Template"));
2255 prompter.set_prompt (_("Name for template:"));
2256 prompter.set_initial_text(_session->name() + _("-template"));
2257 prompter.add_button (Gtk::Stock::SAVE, Gtk::RESPONSE_ACCEPT);
2259 switch (prompter.run()) {
2260 case RESPONSE_ACCEPT:
2261 prompter.get_result (name);
2263 if (name.length()) {
2264 _session->save_template (name);
2274 ARDOUR_UI::edit_metadata ()
2276 SessionMetadataEditor dialog;
2277 dialog.set_session (_session);
2278 editor->ensure_float (dialog);
2283 ARDOUR_UI::import_metadata ()
2285 SessionMetadataImporter dialog;
2286 dialog.set_session (_session);
2287 editor->ensure_float (dialog);
2292 ARDOUR_UI::fontconfig_dialog ()
2295 /* X11 users will always have fontconfig info around, but new GTK-OSX users
2296 may not and it can take a while to build it. Warn them.
2299 std::string fontconfig = Glib::build_filename (Glib::get_home_dir(), ".fontconfig");
2301 if (!Glib::file_test (fontconfig, Glib::FILE_TEST_EXISTS|Glib::FILE_TEST_IS_DIR)) {
2302 MessageDialog msg (*_startup,
2303 string_compose (_("Welcome to %1.\n\n"
2304 "The program will take a bit longer to start up\n"
2305 "while the system fonts are checked.\n\n"
2306 "This will only be done once, and you will\n"
2307 "not see this message again\n"), PROGRAM_NAME),
2320 ARDOUR_UI::parse_cmdline_path (const std::string& cmdline_path, std::string& session_name, std::string& session_path, bool& existing_session)
2322 existing_session = false;
2324 if (Glib::file_test (cmdline_path, Glib::FILE_TEST_IS_DIR)) {
2325 session_path = cmdline_path;
2326 existing_session = true;
2327 } else if (Glib::file_test (cmdline_path, Glib::FILE_TEST_IS_REGULAR)) {
2328 session_path = Glib::path_get_dirname (string (cmdline_path));
2329 existing_session = true;
2331 /* it doesn't exist, assume the best */
2332 session_path = Glib::path_get_dirname (string (cmdline_path));
2335 session_name = basename_nosuffix (string (cmdline_path));
2339 ARDOUR_UI::load_cmdline_session (const std::string& session_name, const std::string& session_path, bool& existing_session)
2341 /* when this is called, the backend audio system must be running */
2343 /* the main idea here is to deal with the fact that a cmdline argument for the session
2344 can be interpreted in different ways - it could be a directory or a file, and before
2345 we load, we need to know both the session directory and the snapshot (statefile) within it
2346 that we are supposed to use.
2349 if (session_name.length() == 0 || session_path.length() == 0) {
2353 if (Glib::file_test (session_path, Glib::FILE_TEST_IS_DIR)) {
2355 std::string predicted_session_file;
2357 predicted_session_file = session_path;
2358 predicted_session_file += '/';
2359 predicted_session_file += session_name;
2360 predicted_session_file += ARDOUR::statefile_suffix;
2362 if (Glib::file_test (predicted_session_file, Glib::FILE_TEST_EXISTS)) {
2363 existing_session = true;
2366 } else if (Glib::file_test (session_path, Glib::FILE_TEST_EXISTS)) {
2368 if (session_path.find (ARDOUR::statefile_suffix) == session_path.length() - 7) {
2369 /* existing .ardour file */
2370 existing_session = true;
2374 existing_session = false;
2377 /* lets just try to load it */
2379 if (create_engine ()) {
2380 backend_audio_error (false, _startup);
2384 return load_session (session_path, session_name);
2388 ARDOUR_UI::ask_about_loading_existing_session (const std::string& session_path)
2390 std::string str = string_compose (_("This session\n%1\nalready exists. Do you want to open it?"), session_path);
2392 MessageDialog msg (str,
2394 Gtk::MESSAGE_WARNING,
2395 Gtk::BUTTONS_YES_NO,
2399 msg.set_name (X_("OpenExistingDialog"));
2400 msg.set_title (_("Open Existing Session"));
2401 msg.set_wmclass (X_("existing_session"), PROGRAM_NAME);
2402 msg.set_position (Gtk::WIN_POS_MOUSE);
2405 switch (msg.run()) {
2414 ARDOUR_UI::build_session_from_nsd (const std::string& session_path, const std::string& session_name)
2416 BusProfile bus_profile;
2418 if (Profile->get_sae()) {
2420 bus_profile.master_out_channels = 2;
2421 bus_profile.input_ac = AutoConnectPhysical;
2422 bus_profile.output_ac = AutoConnectMaster;
2423 bus_profile.requested_physical_in = 0; // use all available
2424 bus_profile.requested_physical_out = 0; // use all available
2428 /* get settings from advanced section of NSD */
2430 if (_startup->create_master_bus()) {
2431 bus_profile.master_out_channels = (uint32_t) _startup->master_channel_count();
2433 bus_profile.master_out_channels = 0;
2436 if (_startup->connect_inputs()) {
2437 bus_profile.input_ac = AutoConnectPhysical;
2439 bus_profile.input_ac = AutoConnectOption (0);
2442 /// @todo some minor tweaks.
2444 bus_profile.output_ac = AutoConnectOption (0);
2446 if (_startup->connect_outputs ()) {
2447 if (_startup->connect_outs_to_master()) {
2448 bus_profile.output_ac = AutoConnectMaster;
2449 } else if (_startup->connect_outs_to_physical()) {
2450 bus_profile.output_ac = AutoConnectPhysical;
2454 bus_profile.requested_physical_in = (uint32_t) _startup->input_limit_count();
2455 bus_profile.requested_physical_out = (uint32_t) _startup->output_limit_count();
2458 if (build_session (session_path, session_name, bus_profile)) {
2466 ARDOUR_UI::idle_load (const std::string& path)
2469 if (Glib::file_test (path, Glib::FILE_TEST_IS_DIR)) {
2470 /* /path/to/foo => /path/to/foo, foo */
2471 load_session (path, basename_nosuffix (path));
2473 /* /path/to/foo/foo.ardour => /path/to/foo, foo */
2474 load_session (Glib::path_get_dirname (path), basename_nosuffix (path));
2478 ARDOUR_COMMAND_LINE::session_name = path;
2481 * new_session_dialog doens't exist in A3
2482 * Try to remove all references to it to
2483 * see if it will compile. NOTE: this will
2484 * likely cause a runtime issue is my somewhat
2488 //if (new_session_dialog) {
2491 /* make it break out of Dialog::run() and
2495 //new_session_dialog->response (1);
2501 ARDOUR_UI::end_loading_messages ()
2507 ARDOUR_UI::loading_message (const std::string& /*msg*/)
2510 // splash->message (msg);
2514 /** @param quit_on_cancel true if exit() should be called if the user clicks `cancel' in the new session dialog */
2516 ARDOUR_UI::get_session_parameters (bool quit_on_cancel, bool should_be_new, string load_template)
2518 string session_name;
2519 string session_path;
2520 string template_name;
2522 bool likely_new = false;
2524 if (! load_template.empty()) {
2525 should_be_new = true;
2526 template_name = load_template;
2531 if (!should_be_new && !ARDOUR_COMMAND_LINE::session_name.empty()) {
2533 /* if they named a specific statefile, use it, otherwise they are
2534 just giving a session folder, and we want to use it as is
2535 to find the session.
2538 if (ARDOUR_COMMAND_LINE::session_name.find (statefile_suffix) != string::npos) {
2539 session_path = Glib::path_get_dirname (ARDOUR_COMMAND_LINE::session_name);
2541 session_path = ARDOUR_COMMAND_LINE::session_name;
2544 session_name = Glib::path_get_basename (ARDOUR_COMMAND_LINE::session_name);
2548 bool const apply = run_startup (should_be_new, load_template);
2551 if (quit_on_cancel) {
2558 /* if we run the startup dialog again, offer more than just "new session" */
2560 should_be_new = false;
2562 session_name = _startup->session_name (likely_new);
2564 /* this shouldn't happen, but we catch it just in case it does */
2566 if (session_name.empty()) {
2570 if (_startup->use_session_template()) {
2571 template_name = _startup->session_template_name();
2572 _session_is_new = true;
2575 if (session_name[0] == G_DIR_SEPARATOR ||
2576 (session_name.length() > 2 && session_name[0] == '.' && session_name[1] == G_DIR_SEPARATOR) ||
2577 (session_name.length() > 3 && session_name[0] == '.' && session_name[1] == '.' && session_name[2] == G_DIR_SEPARATOR)) {
2579 /* absolute path or cwd-relative path specified for session name: infer session folder
2580 from what was given.
2583 session_path = Glib::path_get_dirname (session_name);
2584 session_name = Glib::path_get_basename (session_name);
2588 session_path = _startup->session_folder();
2590 if (session_name.find ('/') != string::npos) {
2591 MessageDialog msg (*_startup,
2592 _("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,
2601 _("To ensure compatibility with various systems\n"
2602 "session names may not contain a '\\' character"));
2604 ARDOUR_COMMAND_LINE::session_name = ""; // cancel that
2610 if (create_engine ()) {
2614 if (Glib::file_test (session_path, Glib::FileTest (G_FILE_TEST_EXISTS | G_FILE_TEST_IS_DIR))) {
2618 std::string existing = Glib::build_filename (session_path, session_name);
2620 if (!ask_about_loading_existing_session (existing)) {
2621 ARDOUR_COMMAND_LINE::session_name = ""; // cancel that
2626 _session_is_new = false;
2631 MessageDialog msg (string_compose (_("There is no existing session at \"%1\""), session_path));
2633 ARDOUR_COMMAND_LINE::session_name = ""; // cancel that
2637 if (session_name.find ('/') != std::string::npos) {
2638 MessageDialog msg (*_startup,
2639 _("To ensure compatibility with various systems\n"
2640 "session names may not contain a '/' character"));
2642 ARDOUR_COMMAND_LINE::session_name = ""; // cancel that
2646 if (session_name.find ('\\') != std::string::npos) {
2647 MessageDialog msg (*_startup,
2648 _("To ensure compatibility with various systems\n"
2649 "session names may not contain a '\\' character"));
2651 ARDOUR_COMMAND_LINE::session_name = ""; // cancel that
2655 _session_is_new = true;
2658 if (likely_new && template_name.empty()) {
2660 ret = build_session_from_nsd (session_path, session_name);
2664 ret = load_session (session_path, session_name, template_name);
2667 /* not connected to the AudioEngine, so quit to avoid an infinite loop */
2671 if (!ARDOUR_COMMAND_LINE::immediate_save.empty()) {
2672 _session->save_state (ARDOUR_COMMAND_LINE::immediate_save, false);
2682 ARDOUR_UI::close_session()
2684 if (!check_audioengine()) {
2688 if (unload_session (true)) {
2692 ARDOUR_COMMAND_LINE::session_name = "";
2694 if (get_session_parameters (true, false)) {
2698 goto_editor_window ();
2701 /** @return -2 if the load failed because we are not connected to the AudioEngine */
2703 ARDOUR_UI::load_session (const std::string& path, const std::string& snap_name, std::string mix_template)
2705 Session *new_session;
2709 session_loaded = false;
2711 if (!check_audioengine()) {
2715 unload_status = unload_session ();
2717 if (unload_status < 0) {
2719 } else if (unload_status > 0) {
2724 loading_message (string_compose (_("Please wait while %1 loads your session"), PROGRAM_NAME));
2727 new_session = new Session (*engine, path, snap_name, 0, mix_template);
2730 /* this one is special */
2732 catch (AudioEngine::PortRegistrationFailure& err) {
2734 MessageDialog msg (err.what(),
2737 Gtk::BUTTONS_CLOSE);
2739 msg.set_title (_("Port Registration Error"));
2740 msg.set_secondary_text (_("Click the Close button to try again."));
2741 msg.set_position (Gtk::WIN_POS_CENTER);
2745 int response = msg.run ();
2750 case RESPONSE_CANCEL:
2760 MessageDialog msg (string_compose(
2761 _("Session \"%1 (snapshot %2)\" did not load successfully"),
2767 msg.set_title (_("Loading Error"));
2768 msg.set_secondary_text (_("Click the Refresh button to try again."));
2769 msg.add_button (Stock::REFRESH, 1);
2770 msg.set_position (Gtk::WIN_POS_CENTER);
2774 int response = msg.run ();
2789 list<string> const u = new_session->unknown_processors ();
2791 MissingPluginDialog d (_session, u);
2796 /* Now the session been created, add the transport controls */
2797 new_session->add_controllable(roll_controllable);
2798 new_session->add_controllable(stop_controllable);
2799 new_session->add_controllable(goto_start_controllable);
2800 new_session->add_controllable(goto_end_controllable);
2801 new_session->add_controllable(auto_loop_controllable);
2802 new_session->add_controllable(play_selection_controllable);
2803 new_session->add_controllable(rec_controllable);
2805 set_session (new_session);
2807 session_loaded = true;
2809 goto_editor_window ();
2812 _session->set_clean ();
2823 ARDOUR_UI::build_session (const std::string& path, const std::string& snap_name, BusProfile& bus_profile)
2825 Session *new_session;
2828 if (!check_audioengine()) {
2832 session_loaded = false;
2834 x = unload_session ();
2842 _session_is_new = true;
2845 new_session = new Session (*engine, path, snap_name, &bus_profile);
2850 MessageDialog msg (string_compose(_("Could not create session in \"%1\""), path));
2856 /* Give the new session the default GUI state, if such things exist */
2859 n = Config->instant_xml (X_("Editor"));
2861 new_session->add_instant_xml (*n, false);
2863 n = Config->instant_xml (X_("Mixer"));
2865 new_session->add_instant_xml (*n, false);
2868 /* Put the playhead at 0 and scroll fully left */
2869 n = new_session->instant_xml (X_("Editor"));
2871 n->add_property (X_("playhead"), X_("0"));
2872 n->add_property (X_("left-frame"), X_("0"));
2875 set_session (new_session);
2877 session_loaded = true;
2879 new_session->save_state(new_session->name());
2885 ARDOUR_UI::launch_chat ()
2888 open_uri("http://webchat.freenode.net/?channels=ardour-osx");
2890 open_uri("http://webchat.freenode.net/?channels=ardour");
2895 ARDOUR_UI::show_about ()
2899 about->signal_response().connect(sigc::mem_fun (*this, &ARDOUR_UI::about_signal_response) );
2902 about->set_transient_for(*editor);
2907 ARDOUR_UI::launch_manual ()
2909 PBD::open_uri("http://ardour.org/flossmanual");
2913 ARDOUR_UI::launch_reference ()
2915 PBD::open_uri("http://ardour.org/refmanual");
2919 ARDOUR_UI::hide_about ()
2922 about->get_window()->set_cursor ();
2928 ARDOUR_UI::about_signal_response (int /*response*/)
2934 ARDOUR_UI::show_splash ()
2938 splash = new Splash;
2946 splash->queue_draw ();
2947 splash->get_window()->process_updates (true);
2952 ARDOUR_UI::hide_splash ()
2960 ARDOUR_UI::display_cleanup_results (ARDOUR::CleanupReport& rep, const gchar* list_title,
2961 const string& plural_msg, const string& singular_msg)
2965 removed = rep.paths.size();
2968 MessageDialog msgd (*editor,
2969 _("No files were ready for cleanup"),
2972 (Gtk::ButtonsType)(Gtk::BUTTONS_OK) );
2973 msgd.set_secondary_text (_("If this seems suprising, \n\
2974 check for any existing snapshots.\n\
2975 These may still include regions that\n\
2976 require some unused files to continue to exist."));
2982 ArdourDialog results (_("Clean-up"), true, false);
2984 struct CleanupResultsModelColumns : public Gtk::TreeModel::ColumnRecord {
2985 CleanupResultsModelColumns() {
2989 Gtk::TreeModelColumn<std::string> visible_name;
2990 Gtk::TreeModelColumn<std::string> fullpath;
2994 CleanupResultsModelColumns results_columns;
2995 Glib::RefPtr<Gtk::ListStore> results_model;
2996 Gtk::TreeView results_display;
2998 results_model = ListStore::create (results_columns);
2999 results_display.set_model (results_model);
3000 results_display.append_column (list_title, results_columns.visible_name);
3002 results_display.set_name ("CleanupResultsList");
3003 results_display.set_headers_visible (true);
3004 results_display.set_headers_clickable (false);
3005 results_display.set_reorderable (false);
3007 Gtk::ScrolledWindow list_scroller;
3010 Gtk::HBox dhbox; // the hbox for the image and text
3011 Gtk::HBox ddhbox; // the hbox we eventually pack into the dialog's vbox
3012 Gtk::Image* dimage = manage (new Gtk::Image(Stock::DIALOG_INFO, Gtk::ICON_SIZE_DIALOG));
3014 dimage->set_alignment(ALIGN_LEFT, ALIGN_TOP);
3016 const string dead_directory = _session->session_directory().dead_path().to_string();
3019 %1 - number of files removed
3020 %2 - location of "dead"
3021 %3 - size of files affected
3022 %4 - prefix for "bytes" to produce sensible results (e.g. mega, kilo, giga)
3025 const char* bprefix;
3026 double space_adjusted = 0;
3028 if (rep.space < 1000) {
3030 space_adjusted = rep.space;
3031 } else if (rep.space < 1000000) {
3032 bprefix = X_("kilo");
3033 space_adjusted = truncf((float)rep.space / 1000.0);
3034 } else if (rep.space < 1000000 * 1000) {
3035 bprefix = X_("mega");
3036 space_adjusted = truncf((float)rep.space / (1000.0 * 1000.0));
3038 bprefix = X_("giga");
3039 space_adjusted = truncf((float)rep.space / (1000.0 * 1000 * 1000.0));
3043 txt.set_text (string_compose (plural_msg, removed, dead_directory, space_adjusted, bprefix));
3045 txt.set_text (string_compose (singular_msg, removed, dead_directory, space_adjusted, bprefix));
3048 dhbox.pack_start (*dimage, true, false, 5);
3049 dhbox.pack_start (txt, true, false, 5);
3051 for (vector<string>::iterator i = rep.paths.begin(); i != rep.paths.end(); ++i) {
3052 TreeModel::Row row = *(results_model->append());
3053 row[results_columns.visible_name] = *i;
3054 row[results_columns.fullpath] = *i;
3057 list_scroller.add (results_display);
3058 list_scroller.set_size_request (-1, 150);
3059 list_scroller.set_policy (Gtk::POLICY_NEVER, Gtk::POLICY_AUTOMATIC);
3061 dvbox.pack_start (dhbox, true, false, 5);
3062 dvbox.pack_start (list_scroller, true, false, 5);
3063 ddhbox.pack_start (dvbox, true, false, 5);
3065 results.get_vbox()->pack_start (ddhbox, true, false, 5);
3066 results.add_button (Stock::CLOSE, RESPONSE_CLOSE);
3067 results.set_default_response (RESPONSE_CLOSE);
3068 results.set_position (Gtk::WIN_POS_MOUSE);
3070 results_display.show();
3071 list_scroller.show();
3078 //results.get_vbox()->show();
3079 results.set_resizable (false);
3086 ARDOUR_UI::cleanup ()
3088 if (_session == 0) {
3089 /* shouldn't happen: menu item is insensitive */
3094 MessageDialog checker (_("Are you sure you want to cleanup?"),
3096 Gtk::MESSAGE_QUESTION,
3097 (Gtk::ButtonsType)(Gtk::BUTTONS_NONE));
3099 checker.set_secondary_text(_("Cleanup is a destructive operation.\n\
3100 ALL undo/redo information will be lost if you cleanup.\n\
3101 Cleanup will move all unused files to a \"dead\" location."));
3103 checker.add_button (Stock::CANCEL, RESPONSE_CANCEL);
3104 checker.add_button (_("Clean Up"), RESPONSE_ACCEPT);
3105 checker.set_default_response (RESPONSE_CANCEL);
3107 checker.set_name (_("CleanupDialog"));
3108 checker.set_wmclass (X_("ardour_cleanup"), PROGRAM_NAME);
3109 checker.set_position (Gtk::WIN_POS_MOUSE);
3111 switch (checker.run()) {
3112 case RESPONSE_ACCEPT:
3118 ARDOUR::CleanupReport rep;
3120 editor->prepare_for_cleanup ();
3122 /* do not allow flush until a session is reloaded */
3124 Glib::RefPtr<Action> act = ActionManager::get_action (X_("Main"), X_("FlushWastebasket"));
3126 act->set_sensitive (false);
3129 if (_session->cleanup_sources (rep)) {
3130 editor->finish_cleanup ();
3134 editor->finish_cleanup ();
3137 display_cleanup_results (rep,
3140 The following %1 files were not in use and \n\
3141 have been moved to:\n\n\
3143 After a restart of Ardour,\n\n\
3144 Session -> Cleanup -> Flush Wastebasket\n\n\
3145 will release an additional\n\
3146 %3 %4bytes of disk space.\n"),
3148 The following file was not in use and \n \
3149 has been moved to:\n \
3151 After a restart of Ardour,\n\n\
3152 Session -> Cleanup -> Flush Wastebasket\n\n\
3153 will release an additional\n\
3154 %3 %4bytes of disk space.\n"
3160 ARDOUR_UI::flush_trash ()
3162 if (_session == 0) {
3163 /* shouldn't happen: menu item is insensitive */
3167 ARDOUR::CleanupReport rep;
3169 if (_session->cleanup_trash_sources (rep)) {
3173 display_cleanup_results (rep,
3175 _("The following %1 files were deleted from\n\
3177 releasing %3 %4bytes of disk space"),
3178 _("The following file was deleted from\n\
3180 releasing %3 %4bytes of disk space"));
3184 ARDOUR_UI::add_route (Gtk::Window* float_window)
3192 if (add_route_dialog == 0) {
3193 add_route_dialog = new AddRouteDialog (_session);
3195 add_route_dialog->set_transient_for (*float_window);
3199 if (add_route_dialog->is_visible()) {
3200 /* we're already doing this */
3204 ResponseType r = (ResponseType) add_route_dialog->run ();
3206 add_route_dialog->hide();
3209 case RESPONSE_ACCEPT:
3216 if ((count = add_route_dialog->count()) <= 0) {
3220 string template_path = add_route_dialog->track_template();
3222 if (!template_path.empty()) {
3223 _session->new_route_from_template (count, template_path);
3227 uint32_t input_chan = add_route_dialog->channels ();
3228 uint32_t output_chan;
3229 string name_template = add_route_dialog->name_template ();
3230 bool track = add_route_dialog->track ();
3231 RouteGroup* route_group = add_route_dialog->route_group ();
3233 AutoConnectOption oac = Config->get_output_auto_connect();
3235 if (oac & AutoConnectMaster) {
3236 output_chan = (_session->master_out() ? _session->master_out()->n_inputs().n_audio() : input_chan);
3238 output_chan = input_chan;
3241 /* XXX do something with name template */
3243 if (add_route_dialog->type() == ARDOUR::DataType::MIDI) {
3245 session_add_midi_track (route_group, count, name_template);
3247 MessageDialog msg (*editor,
3248 _("Sorry, MIDI Busses are not supported at this time."));
3250 //session_add_midi_bus();
3254 session_add_audio_track (input_chan, output_chan, add_route_dialog->mode(), route_group, count, name_template);
3256 session_add_audio_bus (input_chan, output_chan, route_group, count, name_template);
3262 ARDOUR_UI::mixer_settings () const
3267 node = _session->instant_xml(X_("Mixer"));
3269 node = Config->instant_xml(X_("Mixer"));
3273 node = new XMLNode (X_("Mixer"));
3280 ARDOUR_UI::editor_settings () const
3285 node = _session->instant_xml(X_("Editor"));
3287 node = Config->instant_xml(X_("Editor"));
3291 if (getenv("ARDOUR_INSTANT_XML_PATH")) {
3292 node = Config->instant_xml(getenv("ARDOUR_INSTANT_XML_PATH"));
3297 node = new XMLNode (X_("Editor"));
3304 ARDOUR_UI::keyboard_settings () const
3308 node = Config->extra_xml(X_("Keyboard"));
3311 node = new XMLNode (X_("Keyboard"));
3318 ARDOUR_UI::create_xrun_marker (framepos_t where)
3320 editor->mouse_add_new_marker (where, false, true);
3324 ARDOUR_UI::halt_on_xrun_message ()
3326 MessageDialog msg (*editor,
3327 _("Recording was stopped because your system could not keep up."));
3332 ARDOUR_UI::xrun_handler (framepos_t where)
3338 ENSURE_GUI_THREAD (*this, &ARDOUR_UI::xrun_handler, where)
3340 if (_session && Config->get_create_xrun_marker() && _session->actively_recording()) {
3341 create_xrun_marker(where);
3344 if (_session && Config->get_stop_recording_on_xrun() && _session->actively_recording()) {
3345 halt_on_xrun_message ();
3350 ARDOUR_UI::disk_overrun_handler ()
3352 ENSURE_GUI_THREAD (*this, &ARDOUR_UI::disk_overrun_handler)
3354 if (!have_disk_speed_dialog_displayed) {
3355 have_disk_speed_dialog_displayed = true;
3356 MessageDialog* msg = new MessageDialog (*editor, string_compose (_("\
3357 The disk system on your computer\n\
3358 was not able to keep up with %1.\n\
3360 Specifically, it failed to write data to disk\n\
3361 quickly enough to keep up with recording.\n"), PROGRAM_NAME));
3362 msg->signal_response().connect (sigc::bind (sigc::mem_fun (*this, &ARDOUR_UI::disk_speed_dialog_gone), msg));
3368 ARDOUR_UI::disk_underrun_handler ()
3370 ENSURE_GUI_THREAD (*this, &ARDOUR_UI::disk_underrun_handler)
3372 if (!have_disk_speed_dialog_displayed) {
3373 have_disk_speed_dialog_displayed = true;
3374 MessageDialog* msg = new MessageDialog (
3375 *editor, string_compose (_("The disk system on your computer\n\
3376 was not able to keep up with %1.\n\
3378 Specifically, it failed to read data from disk\n\
3379 quickly enough to keep up with playback.\n"), PROGRAM_NAME));
3380 msg->signal_response().connect (sigc::bind (sigc::mem_fun (*this, &ARDOUR_UI::disk_speed_dialog_gone), msg));
3386 ARDOUR_UI::disk_speed_dialog_gone (int /*ignored_response*/, MessageDialog* msg)
3388 have_disk_speed_dialog_displayed = false;
3393 ARDOUR_UI::session_dialog (std::string msg)
3395 ENSURE_GUI_THREAD (*this, &ARDOUR_UI::session_dialog, msg)
3400 d = new MessageDialog (*editor, msg, false, MESSAGE_INFO, BUTTONS_OK, true);
3402 d = new MessageDialog (msg, false, MESSAGE_INFO, BUTTONS_OK, true);
3411 ARDOUR_UI::pending_state_dialog ()
3413 HBox* hbox = new HBox();
3414 Image* image = new Image (Stock::DIALOG_QUESTION, ICON_SIZE_DIALOG);
3415 ArdourDialog dialog (_("Crash Recovery"), true);
3417 This session appears to have been in\n\
3418 middle of recording when ardour or\n\
3419 the computer was shutdown.\n\
3421 Ardour can recover any captured audio for\n\
3422 you, or it can ignore it. Please decide\n\
3423 what you would like to do.\n"));
3424 image->set_alignment(ALIGN_CENTER, ALIGN_TOP);
3425 hbox->pack_start (*image, PACK_EXPAND_WIDGET, 12);
3426 hbox->pack_end (message, PACK_EXPAND_PADDING, 12);
3427 dialog.get_vbox()->pack_start(*hbox, PACK_EXPAND_PADDING, 6);
3428 dialog.add_button (_("Ignore crash data"), RESPONSE_REJECT);
3429 dialog.add_button (_("Recover from crash"), RESPONSE_ACCEPT);
3430 dialog.set_default_response (RESPONSE_ACCEPT);
3431 dialog.set_position (WIN_POS_CENTER);
3436 switch (dialog.run ()) {
3437 case RESPONSE_ACCEPT:
3445 ARDOUR_UI::sr_mismatch_dialog (framecnt_t desired, framecnt_t actual)
3447 HBox* hbox = new HBox();
3448 Image* image = new Image (Stock::DIALOG_QUESTION, ICON_SIZE_DIALOG);
3449 ArdourDialog dialog (_("Sample Rate Mismatch"), true);
3450 Label message (string_compose (_("\
3451 This session was created with a sample rate of %1 Hz\n\
3453 The audioengine is currently running at %2 Hz\n"), desired, actual));
3455 image->set_alignment(ALIGN_CENTER, ALIGN_TOP);
3456 hbox->pack_start (*image, PACK_EXPAND_WIDGET, 12);
3457 hbox->pack_end (message, PACK_EXPAND_PADDING, 12);
3458 dialog.get_vbox()->pack_start(*hbox, PACK_EXPAND_PADDING, 6);
3459 dialog.add_button (_("Do not load session"), RESPONSE_REJECT);
3460 dialog.add_button (_("Load session anyway"), RESPONSE_ACCEPT);
3461 dialog.set_default_response (RESPONSE_ACCEPT);
3462 dialog.set_position (WIN_POS_CENTER);
3467 switch (dialog.run ()) {
3468 case RESPONSE_ACCEPT:
3477 ARDOUR_UI::disconnect_from_jack ()
3480 if( engine->disconnect_from_jack ()) {
3481 MessageDialog msg (*editor, _("Could not disconnect from JACK"));
3485 update_sample_rate (0);
3490 ARDOUR_UI::reconnect_to_jack ()
3493 if (engine->reconnect_to_jack ()) {
3494 MessageDialog msg (*editor, _("Could not reconnect to JACK"));
3498 update_sample_rate (0);
3503 ARDOUR_UI::use_config ()
3505 XMLNode* node = Config->extra_xml (X_("TransportControllables"));
3507 set_transport_controllable_state (*node);
3512 ARDOUR_UI::update_transport_clocks (framepos_t pos)
3514 if (Config->get_primary_clock_delta_edit_cursor()) {
3515 primary_clock.set (pos, false, editor->get_preferred_edit_position(), 1);
3517 primary_clock.set (pos, 0, true);
3520 if (Config->get_secondary_clock_delta_edit_cursor()) {
3521 secondary_clock.set (pos, false, editor->get_preferred_edit_position(), 2);
3523 secondary_clock.set (pos);
3526 if (big_clock_window->get()) {
3527 big_clock.set (pos);
3533 ARDOUR_UI::step_edit_status_change (bool yn)
3535 // XXX should really store pre-step edit status of things
3536 // we make insensitive
3539 rec_button.set_visual_state (3);
3540 rec_button.set_sensitive (false);
3542 rec_button.set_visual_state (0);
3543 rec_button.set_sensitive (true);
3548 ARDOUR_UI::record_state_changed ()
3550 ENSURE_GUI_THREAD (*this, &ARDOUR_UI::record_state_changed);
3552 if (!_session || !big_clock_window->get()) {
3553 /* why bother - the clock isn't visible */
3557 Session::RecordState const r = _session->record_status ();
3558 bool const h = _session->have_rec_enabled_track ();
3560 if (r == Session::Recording && h) {
3561 big_clock.set_widget_name ("BigClockRecording");
3563 big_clock.set_widget_name ("BigClockNonRecording");
3568 ARDOUR_UI::first_idle ()
3571 _session->allow_auto_play (true);
3575 editor->first_idle();
3578 Keyboard::set_can_save_keybindings (true);
3583 ARDOUR_UI::store_clock_modes ()
3585 XMLNode* node = new XMLNode(X_("ClockModes"));
3587 for (vector<AudioClock*>::iterator x = AudioClock::clocks.begin(); x != AudioClock::clocks.end(); ++x) {
3588 node->add_property ((*x)->name().c_str(), enum_2_string ((*x)->mode()));
3591 _session->add_extra_xml (*node);
3592 _session->set_dirty ();
3597 ARDOUR_UI::TransportControllable::TransportControllable (std::string name, ARDOUR_UI& u, ToggleType tp)
3598 : Controllable (name), ui (u), type(tp)
3604 ARDOUR_UI::TransportControllable::set_value (double val)
3606 if (type == ShuttleControl) {
3613 fract = -((0.5 - val)/0.5);
3615 fract = ((val - 0.5)/0.5);
3619 ui.set_shuttle_fract (fract);
3624 /* do nothing: these are radio-style actions */
3628 const char *action = 0;
3632 action = X_("Roll");
3635 action = X_("Stop");
3638 action = X_("Goto Start");
3641 action = X_("Goto End");
3644 action = X_("Loop");
3647 action = X_("Play Selection");
3650 action = X_("Record");
3660 Glib::RefPtr<Action> act = ActionManager::get_action ("Transport", action);
3668 ARDOUR_UI::TransportControllable::get_value (void) const
3687 case ShuttleControl:
3697 ARDOUR_UI::TransportControllable::set_id (const string& str)
3703 ARDOUR_UI::setup_profile ()
3705 if (gdk_screen_width() < 1200) {
3706 Profile->set_small_screen ();
3710 if (getenv ("ARDOUR_SAE")) {
3711 Profile->set_sae ();
3712 Profile->set_single_package ();
3717 ARDOUR_UI::toggle_translations ()
3719 using namespace Glib;
3721 RefPtr<Action> act = ActionManager::get_action (X_("Main"), X_("EnableTranslation"));
3723 RefPtr<ToggleAction> ract = RefPtr<ToggleAction>::cast_dynamic (act);
3726 string i18n_killer = ARDOUR::translation_kill_path();
3728 bool already_enabled = !ARDOUR::translations_are_disabled ();
3730 if (ract->get_active ()) {
3731 /* we don't care about errors */
3732 int fd = ::open (i18n_killer.c_str(), O_RDONLY|O_CREAT, 0644);
3735 /* we don't care about errors */
3736 unlink (i18n_killer.c_str());
3739 if (already_enabled != ract->get_active()) {
3740 MessageDialog win (already_enabled ? _("Translations disabled") : _("Translations enabled"),
3742 Gtk::MESSAGE_WARNING,
3744 win.set_secondary_text (string_compose (_("You must restart %1 for this to take effect."), PROGRAM_NAME));
3745 win.set_position (Gtk::WIN_POS_CENTER);
3753 /** Add a window proxy to our list, so that its state will be saved.
3754 * This call also causes the window to be created and opened if its
3755 * state was saved as `visible'.
3758 ARDOUR_UI::add_window_proxy (WindowProxyBase* p)
3760 _window_proxies.push_back (p);
3764 /** Remove a window proxy from our list. Must be called if a WindowProxy
3765 * is deleted, to prevent hanging pointers.
3768 ARDOUR_UI::remove_window_proxy (WindowProxyBase* p)
3770 _window_proxies.remove (p);
3774 ARDOUR_UI::missing_file (Session*s, std::string str, DataType type)
3776 MissingFileDialog dialog (s, str, type);
3781 int result = dialog.run ();
3788 return 1; // quit entire session load
3791 result = dialog.get_action ();
3797 ARDOUR_UI::ambiguous_file (std::string file, std::string path, std::vector<std::string> hits)
3799 AmbiguousFileDialog dialog (file, hits);
3805 return dialog.get_which ();