root/gui/main.c @ 42429315c5e0fc08b65a7fd8c942ec869de639ae

Revision 42429315c5e0fc08b65a7fd8c942ec869de639ae, 34.2 KB (checked in by Nedko Arnaudov <nedko@…>, 3 years ago)

Move jack latency selection to menu

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