root/gui/main.c @ 3abed82be77c9f7efdc2dc5a20ca1e2d5672e3ff

Revision 3abed82be77c9f7efdc2dc5a20ca1e2d5672e3ff, 39.9 KB (checked in by Nedko Arnaudov <nedko@…>, 6 months ago)

Add callbacks for room studio signals to studio proxy

  • Property mode set to 100644
Line 
1/* -*- Mode: C ; c-basic-offset: 2 -*- */
2/*
3 * LADI Session Handler (ladish)
4 *
5 * Copyright (C) 2008, 2009, 2010 Nedko Arnaudov <nedko@arnaudov.name>
6 * Copyright (C) 2007 Dave Robillard <http://drobilla.net>
7 *
8 **************************************************************************
9 * This file contains the code that implements main() and other top-level functionality
10 **************************************************************************
11 *
12 * LADI Session Handler is free software; you can redistribute it and/or modify
13 * it under the terms of the GNU General Public License as published by
14 * the Free Software Foundation; either version 2 of the License, or
15 * (at your option) any later version.
16 *
17 * LADI Session Handler is distributed in the hope that it will be useful,
18 * but WITHOUT ANY WARRANTY; without even the implied warranty of
19 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
20 * GNU General Public License for more details.
21 *
22 * You should have received a copy of the GNU General Public License
23 * along with LADI Session Handler. If not, see <http://www.gnu.org/licenses/>
24 * or write to the Free Software Foundation, Inc.,
25 * 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
26 */
27
28#include "common.h"
29
30#include <dbus/dbus-glib.h>
31#include <dbus/dbus-glib-lowlevel.h>
32
33#include <math.h>
34
35#include <sys/types.h>
36#include <sys/stat.h>
37#include <fcntl.h>
38#include <unistd.h>
39
40#include "gtk_builder.h"
41#include "canvas.h"
42#include "graph_canvas.h"
43#include "../proxies/jack_proxy.h"
44#include "../proxies/control_proxy.h"
45#include "../dbus_constants.h"
46#include "world_tree.h"
47#include "graph_view.h"
48#include "../catdup.h"
49#include "../proxies/studio_proxy.h"
50#include "ask_dialog.h"
51#include "../proxies/app_supervisor_proxy.h"
52
53GtkWidget * g_main_win;
54
55GtkAction * g_clear_xruns_and_max_dsp_action;
56GtkStatusbar * g_statusbar;
57
58GtkWidget * g_menu_item_new_studio;
59GtkWidget * g_menu_item_start_studio;
60GtkWidget * g_menu_item_stop_studio;
61GtkWidget * g_menu_item_save_studio;
62GtkWidget * g_menu_item_save_as_studio;
63GtkWidget * g_menu_item_unload_studio;
64GtkWidget * g_menu_item_rename_studio;
65GtkWidget * g_menu_item_create_room;
66GtkWidget * g_menu_item_destroy_room;
67GtkWidget * g_menu_item_load_project;
68GtkWidget * g_menu_item_daemon_exit;
69GtkWidget * g_menu_item_jack_configure;
70GtkCheckMenuItem * g_menu_item_jack_latency_32;
71GtkCheckMenuItem * g_menu_item_jack_latency_64;
72GtkCheckMenuItem * g_menu_item_jack_latency_128;
73GtkCheckMenuItem * g_menu_item_jack_latency_256;
74GtkCheckMenuItem * g_menu_item_jack_latency_512;
75GtkCheckMenuItem * g_menu_item_jack_latency_1024;
76GtkCheckMenuItem * g_menu_item_jack_latency_2048;
77GtkCheckMenuItem * g_menu_item_jack_latency_4096;
78GtkCheckMenuItem * g_menu_item_jack_latency_8192;
79GtkWidget * g_studio_status_label;
80GtkWidget * g_menu_item_view_toolbar;
81GtkWidget * g_toolbar;
82GtkWidget * g_menu_item_start_app;
83GtkWidget * g_status_image;
84
85GtkWidget * g_name_dialog;
86GtkWidget * g_app_dialog;
87
88GtkWidget * g_sample_rate_label;
89uint32_t g_sample_rate;
90
91GtkWidget * g_latency_label;
92GtkWidget * g_dsp_load_label;
93GtkWidget * g_xruns_label;
94
95GtkWidget * g_xrun_progress_bar;
96
97graph_view_handle g_jack_view = NULL;
98graph_view_handle g_studio_view = NULL;
99
100static guint g_jack_poll_source_tag;
101static guint g_ladishd_poll_source_tag;
102static double g_jack_max_dsp_load = 0.0;
103
104#define STUDIO_STATE_UNKNOWN    0
105#define STUDIO_STATE_UNLOADED   1
106#define STUDIO_STATE_STOPPED    2
107#define STUDIO_STATE_STARTED    3
108#define STUDIO_STATE_CRASHED    4
109#define STUDIO_STATE_NA         5
110#define STUDIO_STATE_SICK       6
111
112static unsigned int g_studio_state = STUDIO_STATE_UNKNOWN;
113
114#define JACK_STATE_NA         0
115#define JACK_STATE_STOPPED    1
116#define JACK_STATE_STARTED    2
117
118static unsigned int g_jack_state = JACK_STATE_NA;
119
120#define ABOUT_DIALOG_LOGO     "ladish-logo-128x128.png"
121#define STATUS_ICON_DOWN      "status_down.png"         /* temporary down during service restart */
122#define STATUS_ICON_UNLOADED  "status_unloaded.png"
123#define STATUS_ICON_STARTED   "status_started.png"
124#define STATUS_ICON_STOPPED   "status_stopped.png"
125#define STATUS_ICON_WARNING   "status_warning.png"      /* xruns */
126#define STATUS_ICON_ERROR     "status_error.png"        /* bad error */
127
128struct studio_list
129{
130  int count;
131  GtkWidget * menu_item;
132  GtkWidget * menu;
133  void (* item_activate_callback)(GtkWidget * item);
134  bool add_sensitive;
135};
136
137static struct studio_list g_load_studio_list;
138static struct studio_list g_delete_studio_list;
139
140#if 0
141static void
142gtkmm_get_ink_pixel_size (Glib::RefPtr<Pango::Layout> layout,
143             int& width,
144             int& height)
145{
146  Pango::Rectangle ink_rect = layout->get_ink_extents ();
147 
148  width = (ink_rect.get_width() + PANGO_SCALE / 2) / PANGO_SCALE;
149  height = (ink_rect.get_height() + PANGO_SCALE / 2) / PANGO_SCALE;
150}
151
152static void
153gtkmm_set_width_for_given_text (Gtk::Widget &w, const gchar *text,
154               gint hpadding/*, gint vpadding*/)
155 
156{
157  int old_width, old_height;
158  w.get_size_request(old_width, old_height);
159
160  int width, height;
161  w.ensure_style ();
162 
163  gtkmm_get_ink_pixel_size (w.create_pango_layout (text), width, height);
164  w.set_size_request(width + hpadding, old_height);//height + vpadding);
165}
166
167#endif
168
169static GdkPixbuf * load_pixbuf_internal(const char * directory, const char * filename)
170{
171  char * fullpath;
172  GdkPixbuf * pixbuf;
173
174  fullpath = catdup(directory, filename);
175  if (fullpath == NULL)
176  {
177    return NULL;
178  }
179
180  pixbuf = gdk_pixbuf_new_from_file(fullpath, NULL);
181
182  free(fullpath);
183
184  return pixbuf;
185}
186
187static GdkPixbuf * load_pixbuf(const char * filename)
188{
189  GdkPixbuf * pixbuf;
190  static const char * pixbuf_dirs[] = {"./art/", DATA_DIR "/", NULL};
191  const char ** dir;
192
193  for (dir = pixbuf_dirs; *dir != NULL; dir++)
194  {
195    pixbuf = load_pixbuf_internal(*dir, filename);
196    if (pixbuf != NULL)
197    {
198      return pixbuf;
199    }
200  }
201
202  return NULL;
203}
204
205void set_latency_items_sensivity(bool sensitive)
206{
207  gtk_widget_set_sensitive(GTK_WIDGET(g_menu_item_jack_latency_32), sensitive);
208  gtk_widget_set_sensitive(GTK_WIDGET(g_menu_item_jack_latency_64), sensitive);
209  gtk_widget_set_sensitive(GTK_WIDGET(g_menu_item_jack_latency_128), sensitive);
210  gtk_widget_set_sensitive(GTK_WIDGET(g_menu_item_jack_latency_256), sensitive);
211  gtk_widget_set_sensitive(GTK_WIDGET(g_menu_item_jack_latency_512), sensitive);
212  gtk_widget_set_sensitive(GTK_WIDGET(g_menu_item_jack_latency_1024), sensitive);
213  gtk_widget_set_sensitive(GTK_WIDGET(g_menu_item_jack_latency_2048), sensitive);
214  gtk_widget_set_sensitive(GTK_WIDGET(g_menu_item_jack_latency_4096), sensitive);
215  gtk_widget_set_sensitive(GTK_WIDGET(g_menu_item_jack_latency_8192), sensitive);
216}
217
218static void buffer_size_clear(void)
219{
220  set_latency_items_sensivity(false);
221  gtk_label_set_text(GTK_LABEL(g_latency_label), "");
222}
223
224static bool latency_changing = false;
225
226static void buffer_size_set(uint32_t size, bool force)
227{
228  GtkCheckMenuItem * item_ptr;
229
230  switch (size)
231  {
232  case 32:
233    item_ptr = g_menu_item_jack_latency_32;
234    break;
235  case 64:
236    item_ptr = g_menu_item_jack_latency_64;
237    break;
238  case 128:
239    item_ptr = g_menu_item_jack_latency_128;
240    break;
241  case 256:
242    item_ptr = g_menu_item_jack_latency_256;
243    break;
244  case 512:
245    item_ptr = g_menu_item_jack_latency_512;
246    break;
247  case 1024:
248    item_ptr = g_menu_item_jack_latency_1024;
249    break;
250  case 2048:
251    item_ptr = g_menu_item_jack_latency_2048;
252    break;
253  case 4096:
254    item_ptr = g_menu_item_jack_latency_4096;
255    break;
256  case 8192:
257    item_ptr = g_menu_item_jack_latency_8192;
258    break;
259  default:
260    //log_error("unknown jack buffer size %"PRIu32, size);
261    buffer_size_clear();
262    return;
263  }
264
265  if (force || !item_ptr->active)
266  {
267    log_info("JACK latency changed: %"PRIu32" samples", size);
268    latency_changing = true;
269    gtk_check_menu_item_set_active(item_ptr, TRUE);
270    latency_changing = false;
271
272    {
273      char buf[100];
274      snprintf(buf, sizeof(buf), "%4.1f ms (%"PRIu32")", (float)size / (float)g_sample_rate * 1000.0f, size);
275      gtk_label_set_text(GTK_LABEL(g_latency_label), buf);
276    }
277  }
278}
279
280static void buffer_size_change_request(GtkCheckMenuItem * item_ptr, gpointer user_data)
281{
282  uint32_t size;
283
284  if (latency_changing)
285  { /* skip activations because of gtk_check_menu_item_set_active() called from buffer_size_set() */
286    return;
287  }
288
289  if (!item_ptr->active)
290  { /* skip radio button deactivations, we are interested only in activations */
291    return;
292  }
293
294  size = (uint32_t)(guintptr)user_data;
295
296  log_info("JACK latency change request: %"PRIu32" samples", size);
297
298  if (!jack_proxy_set_buffer_size(size))
299  {
300    log_error("cannot set JACK buffer size");
301  }
302}
303
304static void update_buffer_size(bool force)
305{
306  uint32_t size;
307
308  if (jack_proxy_get_buffer_size(&size))
309  {
310    buffer_size_set(size, force);
311  }
312  else
313  {
314    buffer_size_clear();
315  }
316}
317
318static void update_load(void)
319{
320  uint32_t xruns;
321  double load;
322  char tmp_buf[100];
323
324  if (jack_proxy_get_xruns(&xruns))
325  {
326    snprintf(tmp_buf, sizeof(tmp_buf), "%" PRIu32 " dropouts", xruns);
327    gtk_label_set_text(GTK_LABEL(g_xruns_label), tmp_buf);
328    gtk_progress_bar_set_text(GTK_PROGRESS_BAR(g_xrun_progress_bar), tmp_buf);
329  }
330  else
331  {
332    gtk_progress_bar_set_text(GTK_PROGRESS_BAR(g_xrun_progress_bar), "error");
333    gtk_progress_bar_set_fraction(GTK_PROGRESS_BAR(g_xrun_progress_bar), 0.0);
334    gtk_label_set_text(GTK_LABEL(g_xruns_label), "?");
335  }
336
337  if (jack_proxy_get_dsp_load(&load))
338  {
339    if (load > g_jack_max_dsp_load)
340    {
341      g_jack_max_dsp_load = load;
342      gtk_progress_bar_set_fraction(GTK_PROGRESS_BAR(g_xrun_progress_bar), load / 100.0);
343    }
344
345    snprintf(tmp_buf, sizeof(tmp_buf), "DSP: %5.1f%% (%5.1f%%)", (float)load, (float)g_jack_max_dsp_load);
346    gtk_label_set_text(GTK_LABEL(g_dsp_load_label), tmp_buf);
347  }
348  else
349  {
350    gtk_label_set_text(GTK_LABEL(g_xruns_label), "?");
351  }
352}
353
354static void clear_xruns_and_max_dsp(void)
355{
356  log_info("clearing xruns and max dsp load");
357  jack_proxy_reset_xruns();
358  g_jack_max_dsp_load = 0.0;
359}
360
361bool name_dialog(const char * title, const char * object, const char * old_name, char ** new_name)
362{
363  guint result;
364  bool ok;
365  GtkEntry * entry = GTK_ENTRY(get_gtk_builder_widget("name_entry"));
366
367  gtk_window_set_title(GTK_WINDOW(g_app_dialog), title);
368
369  gtk_widget_show(g_name_dialog);
370
371  gtk_label_set_text(GTK_LABEL(get_gtk_builder_widget("name_label")), object);
372  gtk_entry_set_text(entry, old_name);
373  gtk_editable_select_region(GTK_EDITABLE(entry), 0, -1);
374
375  result = gtk_dialog_run(GTK_DIALOG(g_name_dialog));
376  ok = result == 2;
377  if (ok)
378  {
379    *new_name = strdup(gtk_entry_get_text(entry));
380    if (*new_name == NULL)
381    {
382      log_error("strdup failed for new name (name_dialog)");
383      ok = false;
384    }
385  }
386
387  gtk_widget_hide(g_name_dialog);
388
389  return ok;
390}
391
392void error_message_box(const char * failed_operation)
393{
394  GtkWidget * dialog;
395  dialog = get_gtk_builder_widget("error_dialog");
396  gtk_message_dialog_set_markup(GTK_MESSAGE_DIALOG(dialog), "<b><big>Error</big></b>");
397  gtk_message_dialog_format_secondary_markup(GTK_MESSAGE_DIALOG(dialog), "%s", failed_operation);
398  gtk_widget_show(dialog);
399  gtk_dialog_run(GTK_DIALOG(dialog));
400  gtk_widget_hide(dialog);
401}
402
403void run_custom_command_dialog(void)
404{
405  guint result;
406  GtkEntry * command_entry = GTK_ENTRY(get_gtk_builder_widget("app_command_entry"));
407  GtkEntry * name_entry = GTK_ENTRY(get_gtk_builder_widget("app_name_entry"));
408  GtkToggleButton * terminal_button = GTK_TOGGLE_BUTTON(get_gtk_builder_widget("app_terminal_check_button"));
409  GtkToggleButton * level0_button = GTK_TOGGLE_BUTTON(get_gtk_builder_widget("app_level0"));
410  GtkToggleButton * level1_button = GTK_TOGGLE_BUTTON(get_gtk_builder_widget("app_level1"));
411  GtkToggleButton * level2_button = GTK_TOGGLE_BUTTON(get_gtk_builder_widget("app_level2"));
412  GtkToggleButton * level3_button = GTK_TOGGLE_BUTTON(get_gtk_builder_widget("app_level3"));
413  uint8_t level;
414
415  gtk_entry_set_text(name_entry, "");
416  gtk_entry_set_text(command_entry, "");
417  gtk_toggle_button_set_active(terminal_button, FALSE);
418
419  gtk_widget_set_sensitive(GTK_WIDGET(command_entry), TRUE);
420  gtk_widget_set_sensitive(GTK_WIDGET(terminal_button), TRUE);
421  gtk_widget_set_sensitive(GTK_WIDGET(level0_button), TRUE);
422  gtk_widget_set_sensitive(GTK_WIDGET(level1_button), TRUE);
423
424  gtk_window_set_focus(GTK_WINDOW(g_app_dialog), GTK_WIDGET(command_entry));
425  gtk_window_set_title(GTK_WINDOW(g_app_dialog), "New application");
426
427  gtk_widget_show(g_app_dialog);
428
429  result = gtk_dialog_run(GTK_DIALOG(g_app_dialog));
430  if (result == 2)
431  {
432    if (gtk_toggle_button_get_active(level0_button))
433    {
434      level = 0;
435    }
436    else if (gtk_toggle_button_get_active(level1_button))
437    {
438      level = 1;
439    }
440    else if (gtk_toggle_button_get_active(level2_button))
441    {
442      level = 2;
443    }
444    else if (gtk_toggle_button_get_active(level3_button))
445    {
446      level = 3;
447    }
448    else
449    {
450      log_error("unknown level");
451      ASSERT_NO_PASS;
452      level = 0;
453    }
454
455    log_info("'%s':'%s' %s level %"PRIu8, gtk_entry_get_text(name_entry), gtk_entry_get_text(command_entry), gtk_toggle_button_get_active(terminal_button) ? "terminal" : "shell", level);
456    if (!app_run_custom(
457          g_studio_view,
458          gtk_entry_get_text(command_entry),
459          gtk_entry_get_text(name_entry),
460          gtk_toggle_button_get_active(terminal_button),
461          level))
462    {
463      error_message_box("Execution failed. I know you want to know more for the reson but currently you can only check the log file.");
464    }
465  }
466
467  gtk_widget_hide(g_app_dialog);
468}
469
470static void arrange(void)
471{
472  canvas_handle canvas;
473
474  log_info("arrange request");
475
476  canvas = get_current_canvas();
477  if (canvas != NULL)
478  {
479    canvas_arrange(canvas);
480  }
481}
482
483static void daemon_exit(GtkWidget * item)
484{
485  log_info("Daemon exit request");
486
487  if (!control_proxy_exit())
488  {
489    error_message_box("Daemon exit command failed, please inspect logs.");
490  }
491}
492
493static void jack_configure(GtkWidget * item)
494{
495  GError * error_ptr;
496  gchar * argv[] = {"ladiconf", NULL};
497  GtkWidget * dialog;
498
499  log_info("JACK configure request");
500
501  error_ptr = NULL;
502  if (!g_spawn_async(
503        NULL,                   /* working directory */
504        argv,
505        NULL,                   /* envp */
506        G_SPAWN_SEARCH_PATH,    /* flags */
507        NULL,                   /* child_setup callback */
508        NULL,                   /* user_data */
509        NULL,
510        &error_ptr))
511  {
512    dialog = get_gtk_builder_widget("error_dialog");
513    gtk_message_dialog_set_markup(GTK_MESSAGE_DIALOG(dialog), "<b><big>Error executing ladiconf.\nAre LADI Tools installed?</big></b>");
514    gtk_message_dialog_format_secondary_markup(GTK_MESSAGE_DIALOG(dialog), "%s", error_ptr->message);
515    gtk_widget_show(dialog);
516    gtk_dialog_run(GTK_DIALOG(dialog));
517    gtk_widget_hide(dialog);
518    g_error_free(error_ptr);
519  }
520}
521
522static void on_load_studio(GtkWidget * item)
523{
524  const char * studio_name;
525
526  studio_name = gtk_label_get_text(GTK_LABEL(gtk_bin_get_child(GTK_BIN(item))));
527  log_info("Load studio \"%s\"", studio_name);
528
529  if (!control_proxy_load_studio(studio_name))
530  {
531    error_message_box("Studio load failed, please inspect logs.");
532  }
533}
534
535static void on_delete_studio(GtkWidget * item)
536{
537  const char * studio_name;
538  bool result;
539
540  studio_name = gtk_label_get_text(GTK_LABEL(gtk_bin_get_child(GTK_BIN(item))));
541
542  if (!ask_dialog(&result, "<b><big>Confirm studio delete</big></b>", "Studio \"%s\" will be deleted. Are you sure?", studio_name) || !result)
543  {
544    return;
545  }
546
547  log_info("Delete studio \"%s\"", studio_name);
548
549  if (!control_proxy_delete_studio(studio_name))
550  {
551    error_message_box("Studio delete failed, please inspect logs.");
552  }
553}
554
555#define studio_list_ptr ((struct studio_list *)context)
556
557static void remove_studio_list_menu_entry(GtkWidget * item, gpointer context)
558{
559  GtkWidget * label;
560
561  label = gtk_bin_get_child(GTK_BIN(item));
562
563  //log_debug("removing studio menu item \"%s\"", gtk_menu_item_get_label(GTK_MENU_ITEM(item));
564  // gtk_menu_item_get_label() requries gtk 2.16
565  log_debug("removing studio menu item \"%s\"", gtk_label_get_text(GTK_LABEL(label)));
566
567  gtk_container_remove(GTK_CONTAINER(item), label); /* destroy the label and drop the item refcount by one */
568  //log_info("refcount == %d", (unsigned int)G_OBJECT(item)->ref_count);
569  gtk_container_remove(GTK_CONTAINER(studio_list_ptr->menu), item); /* drop the refcount of item by one and thus destroy it */
570  studio_list_ptr->count--;
571}
572
573static void add_studio_list_menu_entry(void * context, const char * studio_name)
574{
575  GtkWidget * item;
576
577  item = gtk_menu_item_new_with_label(studio_name);
578  //log_info("refcount == %d", (unsigned int)G_OBJECT(item)->ref_count); // refcount == 2 because of the label
579  gtk_widget_set_sensitive(item, studio_list_ptr->add_sensitive);
580  gtk_widget_show(item);
581  gtk_menu_shell_append(GTK_MENU_SHELL(studio_list_ptr->menu), item);
582  g_signal_connect(G_OBJECT(item), "activate", G_CALLBACK(studio_list_ptr->item_activate_callback), item);
583  studio_list_ptr->count++;
584}
585
586#undef studio_list_ptr
587
588static void menu_studio_list_clear(struct studio_list * studio_list_ptr)
589{
590  gtk_container_foreach(GTK_CONTAINER(studio_list_ptr->menu), remove_studio_list_menu_entry, studio_list_ptr);
591  ASSERT(studio_list_ptr->count == 0);
592  studio_list_ptr->count = 0;
593}
594
595static void populate_studio_list_menu(GtkMenuItem * menu_item, struct studio_list * studio_list_ptr)
596{
597  menu_studio_list_clear(studio_list_ptr);
598  studio_list_ptr->add_sensitive = true;
599  if (!control_proxy_get_studio_list(add_studio_list_menu_entry, studio_list_ptr))
600  {
601    menu_studio_list_clear(studio_list_ptr);
602    studio_list_ptr->add_sensitive = false;
603    add_studio_list_menu_entry(studio_list_ptr, "Error obtaining studio list");
604  }
605  else if (studio_list_ptr->count == 0)
606  {
607    studio_list_ptr->add_sensitive = false;
608    add_studio_list_menu_entry(studio_list_ptr, "Empty studio list");
609  }
610}
611
612static void save_studio(void)
613{
614  log_info("save studio request");
615  if (!studio_proxy_save())
616  {
617    error_message_box("Studio save failed, please inspect logs.");
618  }
619}
620
621static void save_as_studio(void)
622{
623  char * new_name;
624
625  log_info("save as studio request");
626
627  if (name_dialog("Save studio as", "Studio name", "", &new_name))
628  {
629    if (!studio_proxy_save_as(new_name))
630    {
631      error_message_box("Saving of studio failed, please inspect logs.");
632    }
633
634    free(new_name);
635  }
636}
637
638static void new_studio(void)
639{
640  char * new_name;
641
642  log_info("new studio request");
643
644  if (name_dialog("New studio", "Studio name", "", &new_name))
645  {
646    if (!control_proxy_new_studio(new_name))
647    {
648      error_message_box("Creation of new studio failed, please inspect logs.");
649    }
650
651    free(new_name);
652  }
653}
654
655static void start_app(void)
656{
657  run_custom_command_dialog();
658}
659
660static void start_studio(void)
661{
662  log_info("start studio request");
663  if (!studio_proxy_start())
664  {
665    error_message_box("Studio start failed, please inspect logs.");
666  }
667}
668
669static void stop_studio(void)
670{
671  log_info("stop studio request");
672  if (!studio_proxy_stop())
673  {
674    error_message_box("Studio stop failed, please inspect logs.");
675  }
676}
677
678static void unload_studio(void)
679{
680  log_info("unload studio request");
681  if (!studio_proxy_unload())
682  {
683    error_message_box("Studio unload failed, please inspect logs.");
684  }
685}
686
687static void rename_studio(void)
688{
689  char * new_name;
690
691  if (name_dialog("Rename studio", "Studio name", get_view_name(g_studio_view), &new_name))
692  {
693    if (!studio_proxy_rename(new_name))
694    {
695      error_message_box("Studio rename failed, please inspect logs.");
696    }
697
698    free(new_name);
699  }
700}
701
702static gboolean poll_jack(gpointer data)
703{
704  update_load();
705  update_buffer_size(false);
706
707  return TRUE;
708}
709
710static gboolean poll_ladishd(gpointer data)
711{
712  control_proxy_ping();
713  return TRUE;
714}
715
716bool studio_state_changed(char ** name_ptr_ptr)
717{
718  const char * status;
719  const char * name;
720  char * buffer;
721  const char * status_image_path;
722  const char * tooltip;
723  GdkPixbuf * pixbuf;
724
725  gtk_widget_set_sensitive(g_menu_item_start_studio, g_studio_state == STUDIO_STATE_STOPPED);
726  gtk_widget_set_sensitive(g_menu_item_stop_studio, g_studio_state == STUDIO_STATE_STARTED);
727  gtk_widget_set_sensitive(g_menu_item_save_studio, g_studio_state == STUDIO_STATE_STARTED);
728  gtk_widget_set_sensitive(g_menu_item_save_as_studio, g_studio_state == STUDIO_STATE_STARTED);
729  gtk_widget_set_sensitive(g_menu_item_unload_studio, g_studio_state != STUDIO_STATE_UNLOADED);
730  gtk_widget_set_sensitive(g_menu_item_rename_studio, g_studio_state == STUDIO_STATE_STOPPED || g_studio_state == STUDIO_STATE_STARTED);
731  gtk_widget_set_sensitive(g_menu_item_start_app, g_studio_state == STUDIO_STATE_STOPPED || g_studio_state == STUDIO_STATE_STARTED);
732  //gtk_widget_set_sensitive(g_menu_item_create_room, g_studio_loaded);
733  //gtk_widget_set_sensitive(g_menu_item_destroy_room, g_studio_loaded);
734  //gtk_widget_set_sensitive(g_menu_item_load_project, g_studio_loaded);
735
736  tooltip = NULL;
737  status_image_path = NULL;
738
739  switch (g_jack_state)
740  {
741  case JACK_STATE_NA:
742    tooltip = status = "JACK is sick";
743    status_image_path = STATUS_ICON_ERROR;
744    break;
745  case JACK_STATE_STOPPED:
746    status = "Stopped";
747    break;
748  case JACK_STATE_STARTED:
749    status = "xruns";
750    break;
751  default:
752    status = "???";
753    tooltip = "Internal error - unknown jack state";
754    status_image_path = STATUS_ICON_ERROR;
755  }
756
757  buffer = NULL;
758
759  switch (g_studio_state)
760  {
761  case STUDIO_STATE_NA:
762    name = "ladishd is down";
763    status_image_path = STATUS_ICON_DOWN;
764    break;
765  case STUDIO_STATE_SICK:
766  case STUDIO_STATE_UNKNOWN:
767    tooltip = name = "ladishd is sick";
768    status_image_path = STATUS_ICON_ERROR;
769    break;
770  case STUDIO_STATE_UNLOADED:
771    name = "No studio loaded";
772    status_image_path = STATUS_ICON_UNLOADED;
773    break;
774  case STUDIO_STATE_CRASHED:
775    status = "Crashed";
776    tooltip = "Crashed studio, save your work if you can and unload the studio";
777    status_image_path = STATUS_ICON_ERROR;
778    /* fall through */
779  case STUDIO_STATE_STOPPED:
780  case STUDIO_STATE_STARTED:
781    if (!studio_proxy_get_name(&buffer))
782    {
783      tooltip = "failed to get studio name";
784      log_error("%s", tooltip);
785      status_image_path = STATUS_ICON_ERROR;
786    }
787    else
788    {
789      name = buffer;
790      switch (g_studio_state)
791      {
792      case STUDIO_STATE_STARTED:
793        status_image_path = STATUS_ICON_STARTED;
794        tooltip = "Studio is started";
795        break;
796      case STUDIO_STATE_STOPPED:
797        status_image_path = STATUS_ICON_STOPPED;
798        tooltip = "Studio is stopped";
799        break;
800      }
801      break;
802    }
803  default:
804    name = "???";
805    tooltip = "Internal error - unknown studio state";
806    status_image_path = STATUS_ICON_ERROR;
807  }
808
809  gtk_progress_bar_set_text(GTK_PROGRESS_BAR(g_xrun_progress_bar), status);
810  gtk_label_set_text(GTK_LABEL(g_studio_status_label), name);
811
812  if (status_image_path == NULL || (pixbuf = load_pixbuf(status_image_path)) == NULL)
813  {
814    gtk_image_set_from_stock(GTK_IMAGE(g_status_image), GTK_STOCK_MISSING_IMAGE, GTK_ICON_SIZE_SMALL_TOOLBAR);
815  }
816  else
817  {
818    gtk_image_set_from_pixbuf(GTK_IMAGE(g_status_image), pixbuf);
819    g_object_unref(pixbuf);
820  }
821
822  //gtk_tool_item_set_tooltip_text(GTK_TOOL_ITEM(g_status_tool_item), tooltip);
823
824  if (g_jack_state == JACK_STATE_STARTED)
825  {
826    if (jack_proxy_sample_rate(&g_sample_rate))
827    {
828      char buf[100];
829
830      if (fmod(g_sample_rate, 1000.0) != 0.0)
831      {
832        snprintf(buf, sizeof(buf), "%.1f kHz", (float)g_sample_rate / 1000.0f);
833      }
834      else
835      {
836        snprintf(buf, sizeof(buf), "%u kHz", g_sample_rate / 1000);
837      }
838
839      gtk_label_set_text(GTK_LABEL(g_sample_rate_label), buf);
840    }
841    else
842    {
843      gtk_label_set_text(GTK_LABEL(g_sample_rate_label), "");
844    }
845  }
846  else
847  {
848    gtk_label_set_text(GTK_LABEL(g_sample_rate_label), "");
849    gtk_label_set_text(GTK_LABEL(g_latency_label), "");
850    gtk_label_set_text(GTK_LABEL(g_dsp_load_label), "");
851    gtk_label_set_text(GTK_LABEL(g_xruns_label), "");
852  }
853
854  if (buffer == NULL)
855  {
856    return false;
857  }
858
859  if (name_ptr_ptr != NULL)
860  {
861    *name_ptr_ptr = buffer;
862  }
863  else
864  {
865    free(buffer);
866  }
867
868  return true;
869}
870
871void control_proxy_on_daemon_appeared(void)
872{
873  if (g_studio_state == STUDIO_STATE_NA || g_studio_state == STUDIO_STATE_SICK)
874  {
875    log_info("ladishd appeared");
876    g_source_remove(g_ladishd_poll_source_tag);
877  }
878
879  g_studio_state = STUDIO_STATE_UNLOADED;
880  studio_state_changed(NULL);
881}
882
883void control_proxy_on_daemon_disappeared(bool clean_exit)
884{
885  log_info("ladishd disappeared");
886
887  if (!clean_exit)
888  {
889    error_message_box("ladish daemon crashed");
890    g_studio_state = STUDIO_STATE_SICK;
891  }
892  else
893  {
894    g_studio_state = STUDIO_STATE_NA;
895  }
896
897  studio_state_changed(NULL);
898
899  if (g_studio_view != NULL)
900  {
901    destroy_view(g_studio_view);
902    g_studio_view = NULL;
903  }
904
905  g_ladishd_poll_source_tag = g_timeout_add(500, poll_ladishd, NULL);
906}
907
908void control_proxy_on_studio_appeared(bool initial)
909{
910  char * name;
911  bool started;
912
913  g_studio_state = STUDIO_STATE_STOPPED;
914
915  if (initial)
916  {
917    if (!studio_proxy_is_started(&started))
918    {
919      log_error("intially, studio is present but is_started() check failed.");
920      return;
921    }
922
923    if (started)
924    {
925      g_studio_state = STUDIO_STATE_STARTED;
926    }
927  }
928
929  if (studio_state_changed(&name))
930  {
931    if (g_studio_view != NULL)
932    {
933      log_error("studio appear signal received but studio already exists");
934    }
935    else if (!create_view(name, SERVICE_NAME, STUDIO_OBJECT_PATH, true, true, false, &g_studio_view))
936    {
937      log_error("create_view() failed for studio");
938    }
939
940    free(name);
941  }
942}
943
944void control_proxy_on_studio_disappeared(void)
945{
946  g_studio_state = STUDIO_STATE_UNLOADED;
947  studio_state_changed(NULL);
948
949  if (g_studio_view == NULL)
950  {
951    log_error("studio disappear signal received but studio does not exists");
952    return;
953  }
954
955  if (g_studio_view != NULL)
956  {
957    destroy_view(g_studio_view);
958    g_studio_view = NULL;
959  }
960}
961
962static void on_studio_renamed(const char * new_studio_name)
963{
964  if (g_studio_view != NULL)
965  {
966    set_view_name(g_studio_view, new_studio_name);
967    gtk_label_set_text(GTK_LABEL(g_studio_status_label), new_studio_name);
968  }
969}
970
971void on_studio_started(void)
972{
973  g_studio_state = STUDIO_STATE_STARTED;
974  studio_state_changed(NULL);
975}
976
977void on_studio_stopped(void)
978{
979  g_studio_state = STUDIO_STATE_STOPPED;
980  studio_state_changed(NULL);
981}
982
983void on_studio_crashed(void)
984{
985  g_studio_state = STUDIO_STATE_CRASHED;
986  studio_state_changed(NULL);
987  error_message_box("JACK crashed or stopped unexpectedly. Save your work, then unload and reload the studio.");
988}
989
990void jack_started(void)
991{
992  log_info("JACK started");
993
994  g_jack_state = JACK_STATE_STARTED;
995  studio_state_changed(NULL);
996
997  set_latency_items_sensivity(true);
998  update_buffer_size(true);
999  gtk_action_set_sensitive(g_clear_xruns_and_max_dsp_action, true);
1000
1001  g_jack_poll_source_tag = g_timeout_add(100, poll_jack, NULL);
1002}
1003
1004void jack_stopped(void)
1005{
1006  if (g_jack_state == JACK_STATE_STARTED)
1007  {
1008    log_info("JACK stopped");
1009
1010    g_source_remove(g_jack_poll_source_tag);
1011  }
1012
1013  g_jack_state = JACK_STATE_STOPPED;
1014  studio_state_changed(NULL);
1015
1016  set_latency_items_sensivity(false);
1017  buffer_size_clear();
1018  gtk_action_set_sensitive(g_clear_xruns_and_max_dsp_action, false);
1019  gtk_progress_bar_set_fraction(GTK_PROGRESS_BAR(g_xrun_progress_bar), 0.0);
1020}
1021
1022void jack_appeared(void)
1023{
1024  log_info("JACK appeared");
1025
1026  g_jack_state = JACK_STATE_STOPPED;
1027  studio_state_changed(NULL);
1028
1029#if defined(SHOW_RAW_JACK)
1030  if (!create_view("Raw JACK", JACKDBUS_SERVICE_NAME, JACKDBUS_OBJECT_PATH, false, false, true, &g_jack_view))
1031  {
1032    log_error("create_view() failed for jack");
1033    return;
1034  }
1035#endif
1036}
1037
1038void jack_disappeared(void)
1039{
1040  log_info("JACK disappeared");
1041
1042  jack_stopped();
1043
1044  g_jack_state = JACK_STATE_NA;
1045  studio_state_changed(NULL);
1046
1047#if defined(SHOW_RAW_JACK)
1048  if (g_jack_view != NULL)
1049  {
1050    destroy_view(g_jack_view);
1051    g_jack_view = NULL;
1052  }
1053#endif
1054}
1055
1056static void room_appeared(const char * opath, const char * name, const char * template)
1057{
1058  log_info("room \"%s\" appeared (%s). template is \"%s\"", name, opath, template);
1059}
1060
1061static void room_disappeared(const char * opath, const char * name, const char * template)
1062{
1063  log_info("room \"%s\" disappeared (%s). template is \"%s\"", name, opath, template);
1064}
1065
1066static void room_changed(const char * opath, const char * name, const char * template)
1067{
1068  log_info("%s changed. name is \"%s\". template is \"%s\"", opath, name, template);
1069}
1070
1071void
1072set_main_window_title(
1073  graph_view_handle view)
1074{
1075  char * title;
1076
1077  if (view != NULL)
1078  {
1079    title = catdup(get_view_name(view), " - LADI Session Handler");
1080    gtk_window_set_title(GTK_WINDOW(g_main_win), title);
1081    free(title);
1082  }
1083  else
1084  {
1085    gtk_window_set_title(GTK_WINDOW(g_main_win), "LADI Session Handler");
1086  }
1087}
1088
1089static
1090void
1091init_studio_list(
1092  struct studio_list * studio_list_ptr,
1093  const char * menu_item,
1094  const char * menu,
1095  void (* item_activate_callback)(GtkWidget * item))
1096{
1097  studio_list_ptr->count = 0;
1098  studio_list_ptr->menu_item = get_gtk_builder_widget(menu_item);
1099  studio_list_ptr->menu = get_gtk_builder_widget(menu);
1100  studio_list_ptr->item_activate_callback = item_activate_callback;
1101  gtk_menu_item_set_submenu(GTK_MENU_ITEM(studio_list_ptr->menu_item), studio_list_ptr->menu);
1102  g_signal_connect(G_OBJECT(studio_list_ptr->menu_item), "activate", G_CALLBACK(populate_studio_list_menu), studio_list_ptr);
1103}
1104
1105static void toggle_toolbar(void)
1106{
1107        if (gtk_check_menu_item_get_active(GTK_CHECK_MENU_ITEM(g_menu_item_view_toolbar)))
1108  {
1109                gtk_widget_show(g_toolbar);
1110  }
1111        else
1112  {
1113                gtk_widget_hide(g_toolbar);
1114  }
1115}
1116
1117static char * read_file_contents(const char * filename)
1118{
1119  int fd;
1120  struct stat st;
1121  char * buffer;
1122
1123  if (stat(filename, &st) != 0)
1124  {
1125    return NULL;
1126  }
1127
1128  fd = open(filename, O_RDONLY);
1129  if (fd == -1)
1130  {
1131    return NULL;
1132  }
1133
1134  buffer = malloc(st.st_size + 1);
1135  if (buffer == NULL)
1136  {
1137    close(fd);
1138    return NULL;
1139  }
1140
1141  if (read(fd, buffer, (size_t)st.st_size) != (ssize_t)st.st_size)
1142  {
1143    free(buffer);
1144    buffer = NULL;
1145  }
1146  else
1147  {
1148    buffer[st.st_size] = 0;
1149  }
1150
1151  close(fd);
1152
1153  return buffer;
1154}
1155
1156static void show_about(void)
1157{
1158  GtkWidget * dialog;
1159  GdkPixbuf * pixbuf;
1160  const char * authors[] = {"Nedko Arnaudov", NULL};
1161  const char * artists[] = {"Lapo Calamandrei", NULL};
1162  char * license;
1163
1164  pixbuf =  load_pixbuf(ABOUT_DIALOG_LOGO);
1165  license = read_file_contents(DATA_DIR "/COPYING");
1166
1167  dialog = get_gtk_builder_widget("about_win");
1168  gtk_about_dialog_set_version(GTK_ABOUT_DIALOG(dialog), PACKAGE_VERSION);
1169
1170  if (pixbuf != NULL)
1171  {
1172    gtk_about_dialog_set_logo(GTK_ABOUT_DIALOG(dialog), pixbuf);
1173    g_object_unref(pixbuf);
1174  }
1175
1176  if (license != NULL)
1177  {
1178    gtk_about_dialog_set_license(GTK_ABOUT_DIALOG(dialog), license);
1179    free(license);
1180  }
1181
1182  gtk_about_dialog_set_authors(GTK_ABOUT_DIALOG(dialog), authors);
1183  gtk_about_dialog_set_artists(GTK_ABOUT_DIALOG(dialog), artists);
1184
1185  gtk_widget_show(dialog);
1186  gtk_dialog_run(GTK_DIALOG(dialog));
1187  gtk_widget_hide(dialog);
1188}
1189
1190static void dbus_init(void)
1191{
1192  dbus_error_init(&g_dbus_error);
1193
1194  // Connect to the bus
1195  g_dbus_connection = dbus_bus_get(DBUS_BUS_SESSION, &g_dbus_error);
1196  if (dbus_error_is_set(&g_dbus_error))
1197  {
1198    //error_msg("dbus_bus_get() failed");
1199    //error_msg(g_dbus_error.message);
1200    dbus_error_free(&g_dbus_error);
1201  }
1202
1203  dbus_connection_setup_with_g_main(g_dbus_connection, NULL);
1204}
1205
1206void dbus_uninit(void)
1207{
1208  if (g_dbus_connection)
1209  {
1210    dbus_connection_flush(g_dbus_connection);
1211  }
1212
1213  if (dbus_error_is_set(&g_dbus_error))
1214  {
1215    dbus_error_free(&g_dbus_error);
1216  }
1217}
1218
1219void setup_accelerators(void)
1220{
1221  static GtkActionGroup * action_group_ptr;
1222  static GtkAccelGroup * accel_group_ptr;
1223  struct
1224  {
1225    GtkAction * action_ptr;
1226    const char * shortcut;
1227  } * descriptor_ptr, descriptors [] =
1228      {
1229        {g_clear_xruns_and_max_dsp_action, "c"},
1230        {NULL, NULL}
1231      };
1232
1233  action_group_ptr = gtk_action_group_new("main");
1234  accel_group_ptr = gtk_accel_group_new();
1235
1236  for (descriptor_ptr = descriptors; descriptor_ptr->action_ptr != NULL; descriptor_ptr++)
1237  {
1238    //log_info("action '%s' -> shortcut \"%s\"", gtk_action_get_name(descriptor_ptr->action_ptr), descriptor_ptr->shortcut);
1239    gtk_action_group_add_action_with_accel(action_group_ptr, descriptor_ptr->action_ptr, descriptor_ptr->shortcut);
1240    gtk_action_set_accel_group(descriptor_ptr->action_ptr, accel_group_ptr);
1241    gtk_action_connect_accelerator(descriptor_ptr->action_ptr);
1242  }
1243
1244  gtk_window_add_accel_group(GTK_WINDOW(g_main_win), accel_group_ptr);
1245}
1246
1247int main(int argc, char** argv)
1248{
1249  gtk_init(&argc, &argv);
1250
1251  if (!canvas_init())
1252  {
1253    log_error("Canvas initialization failed.");
1254    return 1;
1255  }
1256
1257  if (!init_gtk_builder())
1258  {
1259    return 1;
1260  }
1261
1262  g_main_win = get_gtk_builder_widget("main_win");
1263  g_clear_xruns_and_max_dsp_action = GTK_ACTION(get_gtk_builder_object("clear_xruns_and_max_dsp_load_action"));
1264  g_menu_item_new_studio = get_gtk_builder_widget("menu_item_new_studio");
1265  g_menu_item_start_app = get_gtk_builder_widget("menu_item_start_app");
1266  g_menu_item_start_studio = get_gtk_builder_widget("menu_item_start_studio");
1267  g_menu_item_stop_studio = get_gtk_builder_widget("menu_item_stop_studio");
1268  g_menu_item_save_studio = get_gtk_builder_widget("menu_item_save_studio");
1269  g_menu_item_save_as_studio = get_gtk_builder_widget("menu_item_save_as_studio");
1270  g_menu_item_unload_studio = get_gtk_builder_widget("menu_item_unload_studio");
1271  g_menu_item_rename_studio = get_gtk_builder_widget("menu_item_rename_studio");
1272  g_menu_item_create_room = get_gtk_builder_widget("menu_item_create_room");
1273  g_menu_item_destroy_room = get_gtk_builder_widget("menu_item_destroy_room");
1274  g_menu_item_load_project = get_gtk_builder_widget("menu_item_load_project");
1275  g_menu_item_daemon_exit = get_gtk_builder_widget("menu_item_daemon_exit");
1276  g_menu_item_jack_configure = get_gtk_builder_widget("menu_item_jack_configure");
1277  g_menu_item_view_toolbar = get_gtk_builder_widget("menu_item_view_toolbar");
1278  g_toolbar = get_gtk_builder_widget("toolbar");
1279  g_statusbar = GTK_STATUSBAR(get_gtk_builder_widget("statusbar"));
1280  g_xrun_progress_bar = get_gtk_builder_widget("xrun_progress_bar");
1281
1282  g_name_dialog = get_gtk_builder_widget("name_dialog");
1283  g_app_dialog = get_gtk_builder_widget("app_dialog");
1284
1285  init_studio_list(&g_load_studio_list, "menu_item_load_studio", "load_studio_menu", on_load_studio);
1286  init_studio_list(&g_delete_studio_list, "menu_item_delete_studio", "delete_studio_menu", on_delete_studio);
1287
1288  g_menu_item_jack_latency_32   = GTK_CHECK_MENU_ITEM(get_gtk_builder_widget("menu_item_jack_latency_32"));
1289  g_menu_item_jack_latency_64   = GTK_CHECK_MENU_ITEM(get_gtk_builder_widget("menu_item_jack_latency_64"));
1290  g_menu_item_jack_latency_128  = GTK_CHECK_MENU_ITEM(get_gtk_builder_widget("menu_item_jack_latency_128"));
1291  g_menu_item_jack_latency_256  = GTK_CHECK_MENU_ITEM(get_gtk_builder_widget("menu_item_jack_latency_256"));
1292  g_menu_item_jack_latency_512  = GTK_CHECK_MENU_ITEM(get_gtk_builder_widget("menu_item_jack_latency_512"));
1293  g_menu_item_jack_latency_1024 = GTK_CHECK_MENU_ITEM(get_gtk_builder_widget("menu_item_jack_latency_1024"));
1294  g_menu_item_jack_latency_2048 = GTK_CHECK_MENU_ITEM(get_gtk_builder_widget("menu_item_jack_latency_2048"));
1295  g_menu_item_jack_latency_4096 = GTK_CHECK_MENU_ITEM(get_gtk_builder_widget("menu_item_jack_latency_4096"));
1296  g_menu_item_jack_latency_8192 = GTK_CHECK_MENU_ITEM(get_gtk_builder_widget("menu_item_jack_latency_8192"));
1297
1298  g_studio_status_label = gtk_label_new("studioname");
1299  gtk_widget_show(g_studio_status_label);
1300  gtk_box_pack_start(GTK_BOX(g_statusbar), g_studio_status_label, FALSE, TRUE, 10);
1301  gtk_box_reorder_child(GTK_BOX(g_statusbar), g_studio_status_label, 0);
1302
1303  g_status_image = gtk_image_new();
1304  gtk_widget_show(g_status_image);
1305  gtk_box_pack_start(GTK_BOX(g_statusbar), g_status_image, FALSE, TRUE, 10);
1306  gtk_box_reorder_child(GTK_BOX(g_statusbar), g_status_image, 1);
1307
1308  g_sample_rate_label = gtk_label_new("srate");
1309  gtk_widget_show(g_sample_rate_label);
1310  gtk_box_pack_start(GTK_BOX(g_statusbar), g_sample_rate_label, FALSE, TRUE, 10);
1311  gtk_box_reorder_child(GTK_BOX(g_statusbar), g_sample_rate_label, 2);
1312
1313  g_latency_label = gtk_label_new("latency");
1314  gtk_widget_show(g_latency_label);
1315  gtk_box_pack_start(GTK_BOX(g_statusbar), g_latency_label, FALSE, TRUE, 10);
1316  gtk_box_reorder_child(GTK_BOX(g_statusbar), g_latency_label, 3);
1317
1318  g_xruns_label = gtk_label_new("xruns");
1319  gtk_widget_show(g_xruns_label);
1320  gtk_box_pack_start(GTK_BOX(g_statusbar), g_xruns_label, FALSE, TRUE, 10);
1321  gtk_box_reorder_child(GTK_BOX(g_statusbar), g_xruns_label, 4);
1322
1323  g_dsp_load_label = gtk_label_new("load");
1324  gtk_widget_show(g_dsp_load_label);
1325  gtk_box_pack_start(GTK_BOX(g_statusbar), g_dsp_load_label, FALSE, TRUE, 10);
1326  gtk_box_reorder_child(GTK_BOX(g_statusbar), g_dsp_load_label, 5);
1327
1328  buffer_size_clear();
1329
1330  world_tree_init();
1331  view_init();
1332
1333  setup_accelerators();
1334
1335  dbus_init();
1336
1337  if (!jack_proxy_init(jack_started, jack_stopped, jack_appeared, jack_disappeared))
1338  {
1339    return 1;
1340  }
1341
1342  if (!control_proxy_init())
1343  {
1344    return 1;
1345  }
1346
1347  if (!studio_proxy_init())
1348  {
1349    return 1;
1350  }
1351
1352  studio_proxy_set_startstop_callbacks(on_studio_started, on_studio_stopped, on_studio_crashed);
1353
1354  studio_proxy_set_renamed_callback(on_studio_renamed);
1355  studio_proxy_set_room_callbacks(room_appeared, room_disappeared, room_changed);
1356
1357  g_signal_connect(G_OBJECT(g_main_win), "destroy", G_CALLBACK(gtk_main_quit), NULL);
1358  g_signal_connect(G_OBJECT(get_gtk_builder_widget("menu_item_quit")), "activate", G_CALLBACK(gtk_main_quit), NULL);
1359  g_signal_connect(G_OBJECT(get_gtk_builder_widget("menu_item_view_arrange")), "activate", G_CALLBACK(arrange), NULL);
1360  g_signal_connect(G_OBJECT(g_menu_item_view_toolbar), "activate", G_CALLBACK(toggle_toolbar), NULL);
1361  g_signal_connect(G_OBJECT(g_menu_item_new_studio), "activate", G_CALLBACK(new_studio), NULL);
1362  g_signal_connect(G_OBJECT(g_menu_item_start_studio), "activate", G_CALLBACK(start_studio), NULL);
1363  g_signal_connect(G_OBJECT(g_menu_item_stop_studio), "activate", G_CALLBACK(stop_studio), NULL);
1364  g_signal_connect(G_OBJECT(g_menu_item_unload_studio), "activate", G_CALLBACK(unload_studio), NULL);
1365  g_signal_connect(G_OBJECT(g_menu_item_save_studio), "activate", G_CALLBACK(save_studio), NULL);
1366  g_signal_connect(G_OBJECT(g_menu_item_save_as_studio), "activate", G_CALLBACK(save_as_studio), NULL);
1367  g_signal_connect(G_OBJECT(g_menu_item_rename_studio), "activate", G_CALLBACK(rename_studio), NULL);
1368  g_signal_connect(G_OBJECT(g_menu_item_daemon_exit), "activate", G_CALLBACK(daemon_exit), NULL);
1369  g_signal_connect(G_OBJECT(g_menu_item_jack_configure), "activate", G_CALLBACK(jack_configure), NULL);
1370  g_signal_connect(G_OBJECT(get_gtk_builder_widget("menu_item_help_about")), "activate", G_CALLBACK(show_about), NULL);
1371  g_signal_connect(G_OBJECT(g_menu_item_start_app), "activate", G_CALLBACK(start_app), NULL);
1372
1373  g_signal_connect(G_OBJECT(g_menu_item_jack_latency_32), "toggled", G_CALLBACK(buffer_size_change_request), (gpointer)32);
1374  g_signal_connect(G_OBJECT(g_menu_item_jack_latency_64), "toggled", G_CALLBACK(buffer_size_change_request), (gpointer)64);
1375  g_signal_connect(G_OBJECT(g_menu_item_jack_latency_128), "toggled", G_CALLBACK(buffer_size_change_request), (gpointer)128);
1376  g_signal_connect(G_OBJECT(g_menu_item_jack_latency_256), "toggled", G_CALLBACK(buffer_size_change_request), (gpointer)256);
1377  g_signal_connect(G_OBJECT(g_menu_item_jack_latency_512), "toggled", G_CALLBACK(buffer_size_change_request), (gpointer)512);
1378  g_signal_connect(G_OBJECT(g_menu_item_jack_latency_1024), "toggled", G_CALLBACK(buffer_size_change_request), (gpointer)1024);
1379  g_signal_connect(G_OBJECT(g_menu_item_jack_latency_2048), "toggled", G_CALLBACK(buffer_size_change_request), (gpointer)2048);
1380  g_signal_connect(G_OBJECT(g_menu_item_jack_latency_4096), "toggled", G_CALLBACK(buffer_size_change_request), (gpointer)4096);
1381  g_signal_connect(G_OBJECT(g_menu_item_jack_latency_8192), "toggled", G_CALLBACK(buffer_size_change_request), (gpointer)8192);
1382
1383  g_signal_connect(G_OBJECT(g_clear_xruns_and_max_dsp_action), "activate", G_CALLBACK(clear_xruns_and_max_dsp), NULL);
1384
1385  gtk_widget_show(g_main_win);
1386
1387  //_about_win->set_transient_for(*_main_win);
1388
1389  gtk_main();
1390
1391  studio_proxy_uninit();
1392  control_proxy_uninit();
1393  dbus_uninit();
1394  uninit_gtk_builder();
1395
1396  return 0;
1397}
Note: See TracBrowser for help on using the browser.