root/gui/main.c @ 31900ca551bcb7feafcc024dba2405b559272501

Revision 31900ca551bcb7feafcc024dba2405b559272501, 34.3 KB (checked in by Nedko Arnaudov <nedko@…>, 3 years ago)

gui: workaround gtk bug that causes progress bar redraw to be skipped sometimes

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