/* -*- Mode: C ; c-basic-offset: 2 -*- */ /* * LADI Session Handler (ladish) * * Copyright (C) 2008, 2009 Nedko Arnaudov * Copyright (C) 2007 Dave Robillard * ************************************************************************** * This file contains the code that implements main() and other top-level functionality ************************************************************************** * * LADI Session Handler is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * LADI Session Handler is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with LADI Session Handler. If not, see * or write to the Free Software Foundation, Inc., * 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. */ #include "common.h" #include #include "glade.h" #include "canvas.h" #include "graph_canvas.h" #include "../jack_proxy.h" #include "dbus_helpers.h" #include "control_proxy.h" #include "../dbus_constants.h" #include "world_tree.h" #include "graph_view.h" #include "../catdup.h" #include "../studio_proxy.h" #include "ask_dialog.h" GtkWidget * g_main_win; GtkWidget * g_clear_load_button; GtkWidget * g_xrun_progress_bar; GtkWidget * g_buffer_size_combo; GtkWidget * g_menu_item_new_studio; GtkWidget * g_menu_item_start_studio; GtkWidget * g_menu_item_stop_studio; GtkWidget * g_menu_item_save_studio; GtkWidget * g_menu_item_unload_studio; GtkWidget * g_menu_item_rename_studio; GtkWidget * g_menu_item_create_room; GtkWidget * g_menu_item_destroy_room; GtkWidget * g_menu_item_load_project; GtkWidget * g_menu_item_start_app; GtkWidget * g_menu_item_daemon_exit; GtkWidget * g_menu_item_jack_configure; GtkWidget * g_studio_status_label; GtkWidget * g_menu_item_view_toolbar; GtkWidget * g_toolbar; GtkWidget * g_name_dialog; graph_view_handle g_jack_view = NULL; graph_view_handle g_studio_view = NULL; static guint g_jack_poll_source_tag; static double g_jack_max_dsp_load = 0.0; struct studio_list { int count; GtkWidget * menu_item; GtkWidget * menu; void (* item_activate_callback)(GtkWidget * item); bool add_sensitive; }; static struct studio_list g_load_studio_list; static struct studio_list g_delete_studio_list; #if 0 static void gtkmm_get_ink_pixel_size (Glib::RefPtr layout, int& width, int& height) { Pango::Rectangle ink_rect = layout->get_ink_extents (); width = (ink_rect.get_width() + PANGO_SCALE / 2) / PANGO_SCALE; height = (ink_rect.get_height() + PANGO_SCALE / 2) / PANGO_SCALE; } static void gtkmm_set_width_for_given_text (Gtk::Widget &w, const gchar *text, gint hpadding/*, gint vpadding*/) { int old_width, old_height; w.get_size_request(old_width, old_height); int width, height; w.ensure_style (); gtkmm_get_ink_pixel_size (w.create_pango_layout (text), width, height); w.set_size_request(width + hpadding, old_height);//height + vpadding); } #endif static void set_buffer_size_combo_width(void) { //gtkmm_set_width_for_given_text(*_buffer_size_combo, "4096 frames", 40); } static void buffer_size_clear() { gtk_entry_set_text(GTK_ENTRY(get_glade_widget("comboboxentry")), ""); } static void buffer_size_set(uint32_t size) { gtk_combo_box_set_active(GTK_COMBO_BOX(g_buffer_size_combo), (int)log2f(size) - 5); } static void buffer_size_change_request(void) { const int selected = gtk_combo_box_get_active(GTK_COMBO_BOX(g_buffer_size_combo)); if (selected < 0 || !jack_proxy_set_buffer_size(1 << (selected + 5))) { lash_error("cannot set JACK buffer size"); buffer_size_clear(); } } static void update_buffer_size(void) { uint32_t size; if (jack_proxy_get_buffer_size(&size)) { buffer_size_set(size); } else { buffer_size_clear(); } } static void update_load(void) { uint32_t xruns; double load; char tmp_buf[100]; if (!jack_proxy_get_xruns(&xruns) || !jack_proxy_get_dsp_load(&load)) { gtk_progress_bar_set_text(GTK_PROGRESS_BAR(g_xrun_progress_bar), "error"); gtk_progress_bar_set_fraction(GTK_PROGRESS_BAR(g_xrun_progress_bar), 0.0); } snprintf(tmp_buf, sizeof(tmp_buf), "%" PRIu32 " Dropouts", xruns); gtk_progress_bar_set_text(GTK_PROGRESS_BAR(g_xrun_progress_bar), tmp_buf); load /= 100.0; // dbus returns it in percents, we use 0..1 if (load > g_jack_max_dsp_load) { g_jack_max_dsp_load = load; gtk_progress_bar_set_fraction(GTK_PROGRESS_BAR(g_xrun_progress_bar), load); } } static void clear_load(void) { jack_proxy_reset_xruns(); g_jack_max_dsp_load = 0.0; gtk_progress_bar_set_fraction(GTK_PROGRESS_BAR(g_xrun_progress_bar), 0.0); } bool name_dialog(const char * title, const char * object, const char * old_name, char ** new_name) { guint result; bool ok; GtkEntry * entry = GTK_ENTRY(get_glade_widget("name_entry")); gtk_widget_show(g_name_dialog); gtk_label_set_text(GTK_LABEL(get_glade_widget("name_label")), object); gtk_entry_set_text(entry, old_name); gtk_editable_select_region(GTK_EDITABLE(entry), 0, -1); result = gtk_dialog_run(GTK_DIALOG(g_name_dialog)); ok = result == 2; if (ok) { *new_name = strdup(gtk_entry_get_text(entry)); if (*new_name == NULL) { lash_error("strdup failed for new name (name_dialog)"); ok = false; } } gtk_widget_hide(g_name_dialog); return ok; } void error_message_box(const char * failed_operation) { GtkWidget * dialog; dialog = get_glade_widget("error_dialog"); gtk_message_dialog_set_markup(GTK_MESSAGE_DIALOG(dialog), "Error"); gtk_message_dialog_format_secondary_markup(GTK_MESSAGE_DIALOG(dialog), "%s", failed_operation); gtk_widget_show(dialog); gtk_dialog_run(GTK_DIALOG(dialog)); gtk_widget_hide(dialog); } static void arrange(void) { canvas_handle canvas; lash_info("arrange request"); canvas = get_current_canvas(); if (canvas != NULL) { canvas_arrange(canvas); } } static void daemon_exit(GtkWidget * item) { lash_info("Daemon exit request"); if (!control_proxy_exit()) { error_message_box("Daemon exit command failed, please inspect logs."); } } static void jack_configure(GtkWidget * item) { GError * error_ptr; gchar * argv[] = {"ladiconf", NULL}; GtkWidget * dialog; lash_info("JACK configure request"); error_ptr = NULL; if (!g_spawn_async( NULL, /* working directory */ argv, NULL, /* envp */ G_SPAWN_SEARCH_PATH, /* flags */ NULL, /* child_setup callback */ NULL, /* user_data */ NULL, &error_ptr)) { dialog = get_glade_widget("error_dialog"); gtk_message_dialog_set_markup(GTK_MESSAGE_DIALOG(dialog), "Error executing ladiconf.\nAre LADI Tools installed?"); gtk_message_dialog_format_secondary_markup(GTK_MESSAGE_DIALOG(dialog), "%s", error_ptr->message); gtk_widget_show(dialog); gtk_dialog_run(GTK_DIALOG(dialog)); gtk_widget_hide(dialog); g_error_free(error_ptr); } } static void on_load_studio(GtkWidget * item) { const char * studio_name; studio_name = gtk_label_get_text(GTK_LABEL(gtk_bin_get_child(GTK_BIN(item)))); lash_info("Load studio \"%s\"", studio_name); if (!control_proxy_load_studio(studio_name)) { error_message_box("Studio load failed, please inspect logs."); } } static void on_delete_studio(GtkWidget * item) { const char * studio_name; bool result; studio_name = gtk_label_get_text(GTK_LABEL(gtk_bin_get_child(GTK_BIN(item)))); if (!ask_dialog(&result, "Confirm studio delete", "Studio \"%s\" will be deleted. Are you sure?", studio_name) || !result) { return; } lash_info("Delete studio \"%s\"", studio_name); if (!control_proxy_delete_studio(studio_name)) { error_message_box("Studio delete failed, please inspect logs."); } } #define studio_list_ptr ((struct studio_list *)context) static void remove_studio_list_menu_entry(GtkWidget * item, gpointer context) { GtkWidget * label; label = gtk_bin_get_child(GTK_BIN(item)); //lash_debug("removing studio menu item \"%s\"", gtk_menu_item_get_label(GTK_MENU_ITEM(item)); // gtk_menu_item_get_label() requries gtk 2.16 lash_debug("removing studio menu item \"%s\"", gtk_label_get_text(GTK_LABEL(label))); gtk_container_remove(GTK_CONTAINER(item), label); /* destroy the label and drop the item refcount by one */ //lash_info("refcount == %d", (unsigned int)G_OBJECT(item)->ref_count); gtk_container_remove(GTK_CONTAINER(studio_list_ptr->menu), item); /* drop the refcount of item by one and thus destroy it */ studio_list_ptr->count--; } static void add_studio_list_menu_entry(void * context, const char * studio_name) { GtkWidget * item; item = gtk_menu_item_new_with_label(studio_name); //lash_info("refcount == %d", (unsigned int)G_OBJECT(item)->ref_count); // refcount == 2 because of the label gtk_widget_set_sensitive(item, studio_list_ptr->add_sensitive); gtk_widget_show(item); gtk_menu_shell_append(GTK_MENU_SHELL(studio_list_ptr->menu), item); g_signal_connect(G_OBJECT(item), "activate", G_CALLBACK(studio_list_ptr->item_activate_callback), item); studio_list_ptr->count++; } #undef studio_list_ptr static void menu_studio_list_clear(struct studio_list * studio_list_ptr) { gtk_container_foreach(GTK_CONTAINER(studio_list_ptr->menu), remove_studio_list_menu_entry, studio_list_ptr); assert(studio_list_ptr->count == 0); studio_list_ptr->count = 0; } static void populate_studio_list_menu(GtkMenuItem * menu_item, struct studio_list * studio_list_ptr) { menu_studio_list_clear(studio_list_ptr); studio_list_ptr->add_sensitive = true; if (!control_proxy_get_studio_list(add_studio_list_menu_entry, studio_list_ptr)) { menu_studio_list_clear(studio_list_ptr); studio_list_ptr->add_sensitive = false; add_studio_list_menu_entry(studio_list_ptr, "Error obtaining studio list"); } else if (studio_list_ptr->count == 0) { studio_list_ptr->add_sensitive = false; add_studio_list_menu_entry(studio_list_ptr, "Empty studio list"); } } static void save_studio(void) { lash_info("save studio request"); if (!studio_proxy_save()) { error_message_box("Studio save failed, please inspect logs."); } } static void new_studio(void) { char * new_name; lash_info("new studio request"); if (name_dialog("New studio", "Studio name", "", &new_name)) { if (!control_proxy_new_studio(new_name)) { error_message_box("Creation of new studio failed, please inspect logs."); } free(new_name); } } static void start_studio(void) { lash_info("start studio request"); if (!studio_proxy_start()) { error_message_box("Studio start failed, please inspect logs."); } } static void stop_studio(void) { lash_info("stop studio request"); if (!studio_proxy_stop()) { error_message_box("Studio stop failed, please inspect logs."); } } static void unload_studio(void) { lash_info("unload studio request"); if (!studio_proxy_unload()) { error_message_box("Studio unload failed, please inspect logs."); } } static void rename_studio(void) { char * new_name; if (name_dialog("Rename studio", "Studio name", get_view_name(g_studio_view), &new_name)) { if (!studio_proxy_rename(new_name)) { error_message_box("Studio rename failed, please inspect logs."); } free(new_name); } } static gboolean poll_jack(gpointer data) { update_load(); update_buffer_size(); return TRUE; } void control_proxy_on_studio_appeared(void) { char * name; if (!studio_proxy_get_name(&name)) { lash_error("failed to get studio name"); goto exit; } if (g_studio_view != NULL) { lash_error("studio appear signal received but studio already exists"); goto free_name; } if (!create_view(name, SERVICE_NAME, STUDIO_OBJECT_PATH, true, false, &g_studio_view)) { lash_error("create_view() failed for studio"); goto free_name; } gtk_widget_set_sensitive(g_menu_item_start_studio, true); gtk_widget_set_sensitive(g_menu_item_stop_studio, true); gtk_widget_set_sensitive(g_menu_item_save_studio, true); gtk_widget_set_sensitive(g_menu_item_unload_studio, true); gtk_widget_set_sensitive(g_menu_item_rename_studio, true); //gtk_widget_set_sensitive(g_menu_item_create_room, true); //gtk_widget_set_sensitive(g_menu_item_destroy_room, true); //gtk_widget_set_sensitive(g_menu_item_load_project, true); //gtk_widget_set_sensitive(g_menu_item_start_app, true); gtk_label_set_text(GTK_LABEL(g_studio_status_label), name); free_name: free(name); exit: return; } void control_proxy_on_studio_disappeared(void) { if (g_studio_view == NULL) { lash_error("studio disappear signal received but studio does not exists"); return; } gtk_widget_set_sensitive(g_menu_item_start_studio, false); gtk_widget_set_sensitive(g_menu_item_stop_studio, false); gtk_widget_set_sensitive(g_menu_item_save_studio, false); gtk_widget_set_sensitive(g_menu_item_unload_studio, false); gtk_widget_set_sensitive(g_menu_item_rename_studio, false); //gtk_widget_set_sensitive(g_menu_item_create_room, false); //gtk_widget_set_sensitive(g_menu_item_destroy_room, false); //gtk_widget_set_sensitive(g_menu_item_load_project, false); //gtk_widget_set_sensitive(g_menu_item_start_app, false); gtk_label_set_text(GTK_LABEL(g_studio_status_label), "No studio loaded"); if (g_studio_view != NULL) { destroy_view(g_studio_view); g_studio_view = NULL; } } static void on_studio_renamed(const char * new_studio_name) { if (g_studio_view != NULL) { set_view_name(g_studio_view, new_studio_name); } } void jack_started(void) { lash_info("JACK started"); gtk_widget_set_sensitive(g_buffer_size_combo, true); gtk_widget_set_sensitive(g_clear_load_button, true); g_jack_poll_source_tag = g_timeout_add(100, poll_jack, NULL); } void jack_stopped(void) { lash_info("JACK stopped"); g_source_remove(g_jack_poll_source_tag); gtk_widget_set_sensitive(g_buffer_size_combo, false); buffer_size_clear(); gtk_widget_set_sensitive(g_clear_load_button, false); gtk_progress_bar_set_fraction(GTK_PROGRESS_BAR(g_xrun_progress_bar), 0.0); gtk_progress_bar_set_text(GTK_PROGRESS_BAR(g_xrun_progress_bar), "Stopped"); } void jack_appeared(void) { lash_info("JACK appeared"); if (!create_view("Raw JACK", JACKDBUS_SERVICE_NAME, JACKDBUS_OBJECT_PATH, false, true, &g_jack_view)) { lash_error("create_view() failed for jack"); return; } } void jack_disappeared(void) { lash_info("JACK disappeared"); if (g_jack_view != NULL) { destroy_view(g_jack_view); g_jack_view = NULL; } } void set_main_window_title( graph_view_handle view) { char * title = catdup(get_view_name(view), " - LADI Session Handler"); gtk_window_set_title(GTK_WINDOW(g_main_win), title); free(title); } static void init_studio_list( struct studio_list * studio_list_ptr, const char * menu_item, const char * menu, void (* item_activate_callback)(GtkWidget * item)) { studio_list_ptr->count = 0; studio_list_ptr->menu_item = get_glade_widget(menu_item); studio_list_ptr->menu = get_glade_widget(menu); studio_list_ptr->item_activate_callback = item_activate_callback; gtk_menu_item_set_submenu(GTK_MENU_ITEM(studio_list_ptr->menu_item), studio_list_ptr->menu); g_signal_connect(G_OBJECT(studio_list_ptr->menu_item), "activate", G_CALLBACK(populate_studio_list_menu), studio_list_ptr); } static void toggle_toolbar(void) { if (gtk_check_menu_item_get_active(GTK_CHECK_MENU_ITEM(g_menu_item_view_toolbar))) { gtk_widget_show(g_toolbar); } else { gtk_widget_hide(g_toolbar); } } static void show_about(void) { GtkWidget * dialog; dialog = get_glade_widget("about_win"); gtk_about_dialog_set_version(GTK_ABOUT_DIALOG(dialog), PACKAGE_VERSION); gtk_widget_show(dialog); gtk_dialog_run(GTK_DIALOG(dialog)); gtk_widget_hide(dialog); } int main(int argc, char** argv) { gtk_init(&argc, &argv); if (!canvas_init()) { lash_error("Canvas initialization failed."); return 1; } if (!init_glade()) { return 1; } g_main_win = get_glade_widget("main_win"); g_clear_load_button = get_glade_widget("clear_load_button"); g_xrun_progress_bar = get_glade_widget("xrun_progress_bar"); g_buffer_size_combo = get_glade_widget("buffer_size_combo"); g_menu_item_new_studio = get_glade_widget("menu_item_new_studio"); g_menu_item_start_studio = get_glade_widget("menu_item_start_studio"); g_menu_item_stop_studio = get_glade_widget("menu_item_stop_studio"); g_menu_item_save_studio = get_glade_widget("menu_item_save_studio"); g_menu_item_unload_studio = get_glade_widget("menu_item_unload_studio"); g_menu_item_rename_studio = get_glade_widget("menu_item_rename_studio"); g_menu_item_create_room = get_glade_widget("menu_item_create_room"); g_menu_item_destroy_room = get_glade_widget("menu_item_destroy_room"); g_menu_item_load_project = get_glade_widget("menu_item_load_project"); g_menu_item_start_app = get_glade_widget("menu_item_start_app"); g_menu_item_daemon_exit = get_glade_widget("menu_item_daemon_exit"); g_menu_item_jack_configure = get_glade_widget("menu_item_jack_configure"); g_studio_status_label = get_glade_widget("studio_status_label"); g_menu_item_view_toolbar = get_glade_widget("menu_item_view_toolbar"); g_toolbar = get_glade_widget("toolbar"); g_name_dialog = get_glade_widget("name_dialog"); init_studio_list(&g_load_studio_list, "menu_item_load_studio", "load_studio_menu", on_load_studio); init_studio_list(&g_delete_studio_list, "menu_item_delete_studio", "delete_studio_menu", on_delete_studio); world_tree_init(); view_init(); patchage_dbus_init(); if (!jack_proxy_init(jack_started, jack_stopped, jack_appeared, jack_disappeared)) { return 1; } if (!control_proxy_init()) { return 1; } if (!studio_proxy_init()) { return 1; } studio_proxy_set_renamed_callback(on_studio_renamed); set_buffer_size_combo_width(); g_signal_connect(G_OBJECT(g_main_win), "destroy", G_CALLBACK(gtk_main_quit), NULL); g_signal_connect(G_OBJECT(get_glade_widget("menu_item_quit")), "activate", G_CALLBACK(gtk_main_quit), NULL); g_signal_connect(G_OBJECT(g_buffer_size_combo), "changed", G_CALLBACK(buffer_size_change_request), NULL); g_signal_connect(G_OBJECT(g_clear_load_button), "clicked", G_CALLBACK(clear_load), NULL); g_signal_connect(G_OBJECT(get_glade_widget("menu_item_view_arrange")), "activate", G_CALLBACK(arrange), NULL); g_signal_connect(G_OBJECT(g_menu_item_view_toolbar), "activate", G_CALLBACK(toggle_toolbar), NULL); g_signal_connect(G_OBJECT(g_menu_item_new_studio), "activate", G_CALLBACK(new_studio), NULL); g_signal_connect(G_OBJECT(g_menu_item_start_studio), "activate", G_CALLBACK(start_studio), NULL); g_signal_connect(G_OBJECT(g_menu_item_stop_studio), "activate", G_CALLBACK(stop_studio), NULL); g_signal_connect(G_OBJECT(g_menu_item_unload_studio), "activate", G_CALLBACK(unload_studio), NULL); g_signal_connect(G_OBJECT(g_menu_item_save_studio), "activate", G_CALLBACK(save_studio), NULL); g_signal_connect(G_OBJECT(g_menu_item_rename_studio), "activate", G_CALLBACK(rename_studio), NULL); g_signal_connect(G_OBJECT(g_menu_item_daemon_exit), "activate", G_CALLBACK(daemon_exit), NULL); g_signal_connect(G_OBJECT(g_menu_item_jack_configure), "activate", G_CALLBACK(jack_configure), NULL); g_signal_connect(G_OBJECT(get_glade_widget("menu_item_help_about")), "activate", G_CALLBACK(show_about), NULL); gtk_widget_show(g_main_win); //_about_win->set_transient_for(*_main_win); gtk_main(); studio_proxy_uninit(); control_proxy_uninit(); uninit_glade(); return 0; }