root/gui/main.c @ 97f4150664c833e181971788d8f58962f4b5e66d

Revision 97f4150664c833e181971788d8f58962f4b5e66d, 36.5 KB (checked in by Nedko Arnaudov <nedko@…>, 7 months ago)

gui: move rest the status widgets to statusbar

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