Resize main window early to fix load dialog position when triggered via commandline
[terminatorX.git] / src / tX_maingui.cc
1 /*
2     terminatorX - realtime audio scratching software
3     Copyright (C) 1999-2021  Alexander K├Ânig
4
5     This program is free software; you can redistribute it and/or modify
6     it under the terms of the GNU General Public License as published by
7     the Free Software Foundation; either version 2 of the License, or
8     (at your option) any later version.
9
10     This program is distributed in the hope that it will be useful,
11     but WITHOUT ANY WARRANTY; without even the implied warranty of
12     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
13     GNU General Public License for more details.
14
15     You should have received a copy of the GNU General Public License
16     along with this program.  If not, see <http://www.gnu.org/licenses/>.
17
18     File: tX_maingui.cc
19
20     Description: This implements the main gtk+ GUI of terminatorX
21                  It serves as a container for the vtt-guis.
22 */
23
24 #include "tX_maingui.h"
25 #include "tX_dialog.h"
26 #include "tX_engine.h"
27 #include "tX_flash.h"
28 #include "tX_global.h"
29 #include "tX_knobloader.h"
30 #include "tX_loaddlg.h"
31 #include "tX_midiin.h"
32 #include "tX_mouse.h"
33 #include "tX_pbutton.h"
34 #include "tX_seqpar.h"
35 #include "tX_sequencer.h"
36 #include "tX_ui_interface.h"
37 #include "tX_ui_support.h"
38 #include "tX_vtt.h"
39 #include "tX_vttgui.h"
40 #include "version.h"
41 #include <gdk/gdkkeysyms.h>
42 #include <gdk/gdkx.h>
43 #include <gtk/gtk.h>
44 #include <math.h>
45 #include <pango/pango.h>
46 #include <string.h>
47 #include <sys/time.h>
48 #include <sys/wait.h>
49 #include <unistd.h>
50
51 #ifdef USE_SCHEDULER
52 #include <sys/resource.h>
53 #endif
54
55 #ifdef USE_X11
56 #include <X11/Xlib.h>
57 #endif
58
59 #define TX_SET_ID_10 "terminatorX turntable set file - version 1.0 - data:"
60 #define TX_SET_ID_11 "terminatorX turntable set file - version 1.1 - data:"
61 #define TX_SET_ID_12 "terminatorX turntable set file - version 1.2 - data:"
62 #define TX_SET_ID_13 "terminatorX turntable set file - version 1.3 - data:"
63 #define TX_SET_ID_14 "terminatorX turntable set file - version 1.4 - data:"
64
65 #define BROWSER1 "xdg-open"
66 #define BROWSER2 "x-www-browser"
67 #define BROWSER3 "firefox"
68
69 int audioon = 0;
70 int sequencer_ready = 1;
71 int fontHeight = 16;
72
73 bool tX_shutdown = false;
74 tx_mouse mouse;
75
76 GtkWidget* tt_parent;
77 GtkWidget* control_parent;
78 GtkWidget* audio_parent;
79 GtkWidget* main_window;
80 GtkWidget* grab_button;
81 GtkWidget* main_flash;
82
83 GtkWidget* seq_rec_btn;
84 GtkWidget* seq_play_btn;
85 GtkWidget* seq_stop_btn;
86 GtkAdjustment* seq_adj;
87 GtkWidget* seq_slider;
88 GtkWidget* seq_entry;
89 GtkWidget* panel_bar;
90
91 int buttons_on_panel_bar = 0;
92
93 int seq_adj_care = 1;
94 int seq_stop_override = 0;
95
96 GtkAdjustment* volume_adj;
97 GtkAdjustment* pitch_adj;
98
99 /* seq-pars */
100 tX_seqpar_main_volume sp_main_volume;
101 tX_seqpar_main_pitch sp_main_pitch;
102
103 GtkWidget* engine_btn;
104
105 GtkWidget* main_menubar;
106 GtkWidget* rec_menu_item;
107
108 int rec_dont_care = 0;
109 gint update_tag;
110
111 #define connect_entry(wid, func, ptr) \
112     ;                                 \
113     g_signal_connect(G_OBJECT(wid), "activate", (GCallback)func, (void*)ptr);
114 #define connect_adj(wid, func, ptr) \
115     ;                               \
116     g_signal_connect(G_OBJECT(wid), "value_changed", (GCallback)func, (void*)ptr);
117 #define connect_button(wid, func, ptr) \
118     ;                                  \
119     g_signal_connect(G_OBJECT(wid), "clicked", (GCallback)func, (void*)ptr);
120
121 #ifdef USE_X11
122 Window x_window;
123 GtkWidget* fullscreen_item;
124 #endif
125
126 GdkWindow* top_window;
127 #define WID_DYN TRUE, TRUE, 0
128 #define WID_FIX FALSE, FALSE, 0
129 #define WID_FIX_BREATHE FALSE, FALSE, 5
130 extern void add_vtt(GtkWidget* ctrl, GtkWidget* audio, char* fn);
131 extern void destroy_gui(vtt_class* vtt);
132 extern void gui_show_focus(vtt_class* vtt, int show);
133
134 GdkWindow* rec_dialog_win = NULL;
135 GtkWidget* rec_dialog = NULL;
136
137 //GtkWidget *no_of_vtts=NULL;
138 GtkWidget* used_mem = NULL;
139
140 int stop_update = 0;
141 int update_delay;
142
143 vtt_class* old_focus = NULL;
144
145 int grab_status = 0;
146 int last_grab_status = 0;
147
148 GtkWidget* delete_all_item = NULL;
149 GtkWidget* delete_all_vtt_item = NULL;
150
151 void gui_set_tooltip(GtkWidget* wid, const char* tip) {
152     gtk_widget_set_tooltip_text(wid, tip);
153 }
154
155 void turn_audio_off(void) {
156     if (audioon) {
157         gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(engine_btn), 0);
158         while (gtk_events_pending())
159             gtk_main_iteration();
160     }
161 }
162
163 gint pos_update(gpointer data) {
164     f_prec temp, temp2;
165
166     if (stop_update) {
167         cleanup_all_vtts();
168         tX_seqpar ::update_all_graphics();
169         if (old_focus)
170             gui_show_focus(old_focus, 0);
171         old_focus = NULL;
172         gtk_tx_flash_clear(main_flash);
173         gdk_display_flush(gdk_display_get_default());
174         update_tag = 0;
175         return FALSE;
176     } else {
177         update_all_vtts();
178
179         /*main vu meter */
180         temp = vtt_class::mix_max_l;
181         vtt_class::mix_max_l = 0;
182         temp2 = vtt_class::mix_max_r;
183         vtt_class::mix_max_r = 0;
184         gtk_tx_flash_set_level(main_flash, temp / FL_SHRT_MAX, temp2 / FL_SHRT_MAX);
185
186         if (vtt_class::focused_vtt != old_focus) {
187             if (old_focus)
188                 gui_show_focus(old_focus, 0);
189             old_focus = vtt_class::focused_vtt;
190             if (old_focus)
191                 gui_show_focus(old_focus, 1);
192         }
193
194         grab_status = mouse.is_grabbed();
195
196         if (grab_status != last_grab_status) {
197             last_grab_status = grab_status;
198             if (!grab_status) {
199                 gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(grab_button), 0);
200             }
201         }
202         gdk_display_flush(gdk_display_get_default());
203         update_delay--;
204
205         if (update_delay < 0) {
206             seq_update();
207             tX_seqpar::update_all_graphics();
208             update_delay = globals.update_delay;
209         }
210
211         if (tX_engine::get_instance()->check_error()) {
212             tX_error("ouch - error while playback...");
213             gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(engine_btn), 0);
214             return FALSE;
215         }
216
217         // let the audio engine we got the chance to do something
218         tX_engine::get_instance()->reset_cycles_ctr();
219
220         return TRUE;
221     }
222 }
223
224 #define TX_MEMTAG "VmData:"
225
226 void mg_update_status() {
227     FILE* procfs;
228     pid_t mypid;
229     char filename[PATH_MAX];
230     char buffer[256];
231     int found = 0;
232     int mem;
233
234     mypid = getpid();
235     sprintf(filename, "/proc/%i/status", mypid);
236     procfs = fopen(filename, "r");
237     if (procfs) {
238         while ((!feof(procfs)) && !found) {
239             if (fgets(buffer, 256, procfs) && (strncmp(TX_MEMTAG, buffer, sizeof(TX_MEMTAG) - 1) == 0)) {
240                 found = 1;
241                 sscanf(buffer, TX_MEMTAG " %i kB", &mem);
242                 sprintf(buffer, "%.1lf M", ((double)mem) / 1024.0);
243                 gtk_label_set_text(GTK_LABEL(used_mem), buffer);
244             }
245         }
246         fclose(procfs);
247     } else {
248         gtk_label_set_text(GTK_LABEL(used_mem), "");
249     }
250
251     /*sprintf(buffer, "%i", vtt_class::vtt_amount);
252         gtk_label_set_text(GTK_LABEL(no_of_vtts), buffer);*/
253 }
254
255 GCallback new_table(GtkWidget*, char* fn) {
256     //turn_audio_off();
257
258     if (fn) {
259         ld_create_loaddlg(TX_LOADDLG_MODE_SINGLE, 1);
260         ld_set_filename(fn);
261     }
262
263     add_vtt(control_parent, audio_parent, fn);
264
265     if (fn)
266         ld_destroy();
267     mg_update_status();
268
269 #ifdef USE_ALSA_MIDI_IN
270     if (globals.auto_assign_midi)
271         tX_midiin::auto_assign_midi_mappings(NULL, NULL);
272 #endif
273
274     return NULL;
275 }
276
277 bool tx_mg_have_setname = false;
278 char tx_mg_current_setname[PATH_MAX] = "";
279
280 GCallback new_tables() {
281     GtkWidget* dialog = gtk_message_dialog_new(GTK_WINDOW(main_window),
282         GTK_DIALOG_DESTROY_WITH_PARENT, GTK_MESSAGE_QUESTION, GTK_BUTTONS_YES_NO,
283         "Are you sure you want to lose all turntables and events?");
284
285     int res = gtk_dialog_run(GTK_DIALOG(dialog));
286     gtk_widget_destroy(dialog);
287
288     if (res != GTK_RESPONSE_YES) {
289         return NULL;
290     }
291
292     vtt_class::delete_all();
293     new_table(NULL, NULL);
294
295 #ifdef USE_ALSA_MIDI_IN
296     if (globals.auto_assign_midi)
297         tX_midiin::auto_assign_midi_mappings(NULL, NULL);
298 #endif
299
300     gtk_window_set_title(GTK_WINDOW(main_window), "terminatorX");
301
302     return NULL;
303 }
304
305 /* Loading saved setups */
306 void load_tt_part(char* buffer) {
307     char wbuf[PATH_MAX];
308     xmlDocPtr doc;
309 #ifdef ENABLE_TX_LEGACY
310     FILE* in;
311 #endif
312     turn_audio_off();
313
314     strncpy(globals.tables_filename, buffer, sizeof(globals.tables_filename));
315
316     doc = xmlParseFile(buffer);
317     if (doc) {
318         vtt_class::load_all(doc, buffer);
319         xmlFreeDoc(doc);
320     }
321
322 #ifdef ENABLE_TX_LEGACY
323     else {
324         in = fopen(buffer, "r");
325
326         if (in) {
327             char idbuff[256];
328
329             fread(idbuff, strlen(TX_SET_ID_10), 1, in);
330             if (strncmp(idbuff, TX_SET_ID_10, strlen(TX_SET_ID_10)) == 0) {
331                 if (vtt_class::load_all_10(in, buffer))
332                     tx_note("Error while reading set.", true);
333             } else if (strncmp(idbuff, TX_SET_ID_11, strlen(TX_SET_ID_11)) == 0) {
334                 if (vtt_class::load_all_11(in, buffer))
335                     tx_note("Error while reading set.", true);
336             } else if (strncmp(idbuff, TX_SET_ID_12, strlen(TX_SET_ID_12)) == 0) {
337                 if (vtt_class::load_all_12(in, buffer))
338                     tx_note("Error while reading set.", true);
339             } else if (strncmp(idbuff, TX_SET_ID_13, strlen(TX_SET_ID_13)) == 0) {
340                 if (vtt_class::load_all_13(in, buffer))
341                     tx_note("Error while reading set.", true);
342             } else if (strncmp(idbuff, TX_SET_ID_14, strlen(TX_SET_ID_14)) == 0) {
343                 if (vtt_class::load_all_14(in, buffer))
344                     tx_note("Error while reading set.", true);
345             } else {
346                 tx_note("This file is not a terminatorX set-file.", true);
347                 fclose(in);
348                 return;
349             }
350             fclose(in);
351         } else {
352             char message[PATH_MAX + 256];
353             sprintf(message, "Failed to acesss file: \"%s\"", globals.tables_filename);
354             tx_note(message, true);
355
356             return;
357         }
358     }
359 #else
360     else {
361         char message[PATH_MAX + 256];
362         sprintf(message, "Failed to acesss file: \"%s\"", globals.tables_filename);
363         tx_note(message, true);
364
365         return;
366     }
367 #endif
368
369     tx_mg_have_setname = true;
370     strcpy(tx_mg_current_setname, buffer);
371
372     tX_seqpar ::init_all_graphics();
373     vg_init_all_non_seqpars();
374
375     gtk_adjustment_set_value(volume_adj, globals.volume);
376     gtk_adjustment_set_value(pitch_adj, globals.pitch);
377     sprintf(wbuf, "terminatorX - %s", strip_path(buffer));
378 #ifdef USE_ALSA_MIDI_IN
379     if (globals.auto_assign_midi)
380         tX_midiin::auto_assign_midi_mappings(NULL, NULL);
381 #endif
382     gtk_window_set_title(GTK_WINDOW(main_window), wbuf);
383 }
384
385 GCallback load_tables() {
386     GtkWidget* dialog = gtk_file_chooser_dialog_new("Open Set File",
387         GTK_WINDOW(main_window), GTK_FILE_CHOOSER_ACTION_OPEN,
388         "_Cancel", GTK_RESPONSE_CANCEL,
389         "_Open", GTK_RESPONSE_ACCEPT, NULL);
390
391     GtkFileFilter* filter = gtk_file_filter_new();
392     gtk_file_filter_add_pattern(filter, "*.tX");
393     gtk_file_filter_add_pattern(filter, "*.tx");
394     gtk_file_filter_set_name(filter, "terminatorX Set Files");
395     gtk_file_chooser_add_filter(GTK_FILE_CHOOSER(dialog), filter);
396
397     filter = gtk_file_filter_new();
398     gtk_file_filter_add_pattern(filter, "*");
399     gtk_file_filter_set_name(filter, "All Files");
400     gtk_file_chooser_add_filter(GTK_FILE_CHOOSER(dialog), filter);
401
402     if (strlen(globals.tables_filename)) {
403         gtk_file_chooser_set_filename(GTK_FILE_CHOOSER(dialog), globals.tables_filename);
404     }
405
406     if (gtk_dialog_run(GTK_DIALOG(dialog)) == GTK_RESPONSE_ACCEPT) {
407         char* filename = gtk_file_chooser_get_filename(GTK_FILE_CHOOSER(dialog));
408         gtk_widget_hide(dialog);
409         tX_cursor::set_cursor(tX_cursor::WAIT_CURSOR);
410         load_tt_part(filename);
411         tX_cursor::reset_cursor();
412     }
413
414     gtk_widget_destroy(dialog);
415
416     return NULL;
417 }
418
419 vtt_class* choose_vtt() {
420     GtkWidget* dialog = gtk_dialog_new_with_buttons("Select Turntable",
421         GTK_WINDOW(main_window), GTK_DIALOG_MODAL, "_Cancel", GTK_RESPONSE_REJECT,
422         "_OK", GTK_RESPONSE_ACCEPT, NULL);
423
424     GtkWidget* label = gtk_label_new("Select turntable to load audio file to:");
425     gtk_widget_show(label);
426
427     gtk_container_add(GTK_CONTAINER(gtk_dialog_get_content_area(GTK_DIALOG(dialog))), label);
428     gtk_container_set_border_width(GTK_CONTAINER(gtk_dialog_get_content_area(GTK_DIALOG(dialog))), 10);
429
430     list<GtkWidget*> radio_buttons;
431     list<vtt_class*>::iterator iter;
432     bool first = true;
433     GtkWidget* radio;
434     vtt_class* res_vtt = NULL;
435
436     pthread_mutex_lock(&vtt_class::render_lock);
437
438     for (iter = vtt_class::main_list.begin(); iter != vtt_class::main_list.end(); iter++) {
439         if (first) {
440             first = false;
441             radio = gtk_radio_button_new_with_label(NULL, (*iter)->name);
442         } else {
443             radio = gtk_radio_button_new_with_label_from_widget(GTK_RADIO_BUTTON(radio), (*iter)->name);
444         }
445         g_object_set_data(G_OBJECT(radio), "tX_vtt", (*iter));
446         gtk_container_add(GTK_CONTAINER(gtk_dialog_get_content_area(GTK_DIALOG(dialog))), radio);
447         gtk_widget_show(radio);
448         radio_buttons.push_back(radio);
449     }
450
451     pthread_mutex_unlock(&vtt_class::render_lock);
452
453     // Giving up the lock here is necessary if we want audio to keep running
454     // however it is also wrong. Anyway as it is a modal dialog not too many
455     // evil things can happen. This sounds like some famous last words...
456
457     gint result = gtk_dialog_run(GTK_DIALOG(dialog));
458
459     if (result == GTK_RESPONSE_ACCEPT) {
460         list<GtkWidget*>::iterator radio;
461
462         for (radio = radio_buttons.begin(); radio != radio_buttons.end(); radio++) {
463             if (gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON((*radio)))) {
464                 res_vtt = (vtt_class*)g_object_get_data(G_OBJECT(*radio), "tX_vtt");
465             }
466         }
467     }
468
469     gtk_widget_destroy(dialog);
470
471     return res_vtt;
472 }
473
474 GCallback load_audio() {
475     vtt_class* vtt = NULL;
476
477     pthread_mutex_lock(&vtt_class::render_lock);
478     if (vtt_class::main_list.size() == 1) {
479         vtt = (*vtt_class::main_list.begin());
480     }
481     pthread_mutex_unlock(&vtt_class::render_lock);
482
483     if (vtt == NULL) {
484         vtt = choose_vtt();
485     }
486
487     if (vtt != NULL)
488         load_file(NULL, vtt);
489
490     return NULL;
491 }
492
493 GCallback drop_set(GtkWidget* widget, GdkDragContext* context,
494     gint x, gint y, GtkSelectionData* selection_data,
495     guint info, guint time, vtt_class* vtt) {
496     char filename[PATH_MAX];
497     char* fn;
498
499     strncpy(filename, (char*)gtk_selection_data_get_data(selection_data), (size_t)gtk_selection_data_get_length(selection_data));
500     filename[gtk_selection_data_get_length(selection_data)] = 0;
501
502     fn = strchr(filename, '\r');
503     *fn = 0;
504
505     fn = strchr(filename, ':');
506     if (fn)
507         fn++;
508     else
509         fn = (char*)gtk_selection_data_get_data(selection_data);
510
511     load_tt_part(fn);
512
513     strcpy(filename, "dont segfault workaround ;)");
514     return NULL;
515 }
516
517 /* save tables */
518 void do_save_tables(char* buffer) {
519     FILE* out;
520     gzFile zout;
521     char wbuf[PATH_MAX];
522     char* ext;
523
524     ext = strrchr(buffer, '.');
525
526     if (ext) {
527         if (strcmp(ext, ".tX"))
528             strcat(buffer, ".tX");
529     } else {
530         strcat(buffer, ".tX");
531     }
532
533     tx_mg_have_setname = true;
534     strcpy(tx_mg_current_setname, buffer);
535     strcpy(globals.tables_filename, buffer);
536
537     if (globals.compress_set_files) {
538         _store_compress_xml = 1;
539         out = NULL;
540         zout = gzopen(buffer, "w");
541     } else {
542         _store_compress_xml = 0;
543         out = fopen(buffer, "w");
544         zout = NULL;
545     }
546
547     if (out || zout) {
548         if (vtt_class::save_all(out, zout))
549             tx_note("Error while saving set.", true);
550         if (out)
551             fclose(out);
552         else if (zout)
553             gzclose(zout);
554         sprintf(wbuf, "terminatorX - %s", strip_path(buffer));
555         gtk_window_set_title(GTK_WINDOW(main_window), wbuf);
556     } else {
557         tx_note("Failed to open file for write access.", true);
558     }
559 }
560
561 GCallback save_tables_as() {
562     GtkWidget* dialog = gtk_file_chooser_dialog_new("Save Set",
563         GTK_WINDOW(main_window), GTK_FILE_CHOOSER_ACTION_SAVE,
564         "_Cancel", GTK_RESPONSE_CANCEL,
565         "_Save", GTK_RESPONSE_ACCEPT, NULL);
566
567     if (tx_mg_have_setname) {
568         gtk_file_chooser_set_filename(GTK_FILE_CHOOSER(dialog), tx_mg_current_setname);
569     }
570
571     if (gtk_dialog_run(GTK_DIALOG(dialog)) == GTK_RESPONSE_ACCEPT) {
572         char buffer[PATH_MAX];
573         char* filename = gtk_file_chooser_get_filename(GTK_FILE_CHOOSER(dialog));
574         strcpy(buffer, filename);
575         g_free(filename);
576         do_save_tables(buffer);
577     }
578
579     gtk_widget_destroy(dialog);
580
581     return NULL;
582 }
583
584 GCallback save_tables() {
585     if (!tx_mg_have_setname) {
586         save_tables_as();
587     } else {
588         do_save_tables(tx_mg_current_setname);
589     }
590
591     return NULL;
592 }
593
594 GCallback main_volume_changed(GtkWidget* wid, void* d) {
595     sp_main_volume.receive_gui_value((float)gtk_adjustment_get_value(GTK_ADJUSTMENT(wid)));
596     return NULL;
597 }
598
599 GCallback main_pitch_changed(GtkWidget* wid, void* d) {
600     sp_main_pitch.receive_gui_value((float)gtk_adjustment_get_value(GTK_ADJUSTMENT(wid)));
601     return NULL;
602 }
603
604 void mg_enable_critical_buttons(int enable) {
605     gtk_widget_set_sensitive(seq_rec_btn, enable);
606     gtk_widget_set_sensitive(seq_play_btn, enable);
607     gtk_widget_set_sensitive(seq_slider, enable);
608
609     gtk_widget_set_sensitive(rec_menu_item, enable);
610     gtk_widget_set_sensitive(delete_all_item, enable);
611     gtk_widget_set_sensitive(delete_all_vtt_item, enable);
612
613     vg_enable_critical_buttons(enable);
614 }
615
616 GCallback seq_stop(GtkWidget* w, void*);
617
618 static bool stop_override = false;
619
620 GCallback audio_on(GtkWidget* w, void* d) {
621     tX_engine_error res;
622
623     if (gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(w))) {
624         sequencer_ready = 0;
625         mg_enable_critical_buttons(0);
626         res = tX_engine::get_instance()->run();
627
628         if (res != NO_ERROR) {
629             mg_enable_critical_buttons(1);
630             stop_override = true;
631             gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(w), 0);
632             stop_override = false;
633
634             switch (res) {
635             case ERROR_BUSY:
636                 tx_note("Error starting engine: engine is already running.", true);
637                 break;
638             case ERROR_AUDIO:
639                 tx_note("Error starting engine: failed to access audiodevice.\nPlease check the audio device settings in the \"Preferences\" dialog.", true);
640                 break;
641             case ERROR_BACKEND:
642                 char audio_backend[256];
643                 char message[4096];
644
645                 switch (globals.audiodevice_type) {
646                 case OSS:
647                     strcpy(audio_backend, "the Open Sound System (OSS)");
648                     break;
649                 case ALSA:
650                     strcpy(audio_backend, "the Advanced Linux Sound Architecture (ALSA)");
651                     break;
652                 case JACK:
653                     strcpy(audio_backend, "the JACK Audio Connection Kit");
654                     break;
655                 case PULSE:
656                     strcpy(audio_backend, "PulseAudio");
657                     break;
658                 default:
659                     strcpy(audio_backend, "unkown");
660                 }
661
662                 sprintf(message, "Error starting engine: couldn't start audio driver.\nThis terminatorX binary was compiled without support for %s.\nPlease check the audio device settings in the \"Preferences\" dialog.", audio_backend);
663                 tx_note(message, true);
664                 break;
665             case ERROR_TAPE:
666                 tx_note("Error starting engine: failed to open the recording file.", true);
667                 break;
668             default:
669                 tx_note("Error starting engine: Unknown error.", true);
670             }
671
672             return 0;
673         }
674
675         sequencer_ready = 1;
676         stop_update = 0;
677         audioon = 1;
678         update_delay = globals.update_delay;
679         update_tag = g_timeout_add(globals.update_idle, (GSourceFunc)pos_update, NULL);
680         gtk_widget_set_sensitive(grab_button, 1);
681     } else {
682         if (stop_override)
683             return NULL;
684         if (!sequencer_ready)
685             return NULL;
686         gtk_widget_set_sensitive(grab_button, 0);
687         tX_engine::get_instance()->stop();
688         stop_update = 1;
689         audioon = 0;
690         if (tX_engine::get_instance()->get_recording_request()) {
691             tX_engine::get_instance()->set_recording_request(false);
692             rec_dont_care = 1;
693             gtk_check_menu_item_set_active(GTK_CHECK_MENU_ITEM(rec_menu_item), 0);
694             rec_dont_care = 0;
695         }
696
697         seq_stop(NULL, NULL);
698         mg_enable_critical_buttons(1);
699
700         if (tX_engine::get_instance()->get_runtime_error()) {
701             tx_note("Fatal: The audio device broke down while playing\nback audio. Note that that some audio devices can not\nrecover from such a breakdown.", true);
702         }
703         if (tX_engine::get_instance()->get_overload_error()) {
704             tx_note("Fatal: The audio engine was stopped due to an overload\ncondition. Try reducing the amount of plugins or\nturntables.", true);
705         }
706     }
707
708     return NULL;
709 }
710
711 GCallback cancel_rec(GtkWidget* wid) {
712     gtk_widget_destroy(rec_dialog);
713     rec_dialog = NULL;
714     rec_dialog_win = NULL;
715     rec_dont_care = 0;
716     return (0);
717 }
718
719 void do_rec(GtkWidget* wid) {
720     char buffer[PATH_MAX];
721
722     strcpy(buffer, gtk_file_chooser_get_filename(GTK_FILE_CHOOSER(rec_dialog)));
723
724     if (strlen(buffer)) {
725         strcpy(globals.record_filename, buffer);
726         tX_engine::get_instance()->set_recording_request(true);
727         gtk_check_menu_item_set_active(GTK_CHECK_MENU_ITEM(rec_menu_item), 1);
728     }
729
730     rec_dont_care = 0;
731
732     gtk_widget_destroy(rec_dialog);
733
734     rec_dialog = NULL;
735     rec_dialog_win = NULL;
736 }
737
738 GCallback select_rec_file() {
739     GtkWidget* dialog = gtk_file_chooser_dialog_new("Record To Disk",
740         GTK_WINDOW(main_window), GTK_FILE_CHOOSER_ACTION_SAVE,
741         "_Cancel", GTK_RESPONSE_CANCEL,
742         "_Save", GTK_RESPONSE_ACCEPT, NULL);
743
744     if (strlen(globals.record_filename)) {
745         gtk_file_chooser_set_filename(GTK_FILE_CHOOSER(dialog), globals.record_filename);
746     }
747
748     if (gtk_dialog_run(GTK_DIALOG(dialog)) == GTK_RESPONSE_ACCEPT) {
749         char* filename = gtk_file_chooser_get_filename(GTK_FILE_CHOOSER(dialog));
750         strncpy(globals.record_filename, filename, sizeof(globals.record_filename) - 1);
751         g_free(filename);
752
753         tX_engine::get_instance()->set_recording_request(true);
754         gtk_check_menu_item_set_active(GTK_CHECK_MENU_ITEM(rec_menu_item), 1);
755     } else {
756         gtk_check_menu_item_set_active(GTK_CHECK_MENU_ITEM(rec_menu_item), 0);
757     }
758
759     rec_dont_care = 0;
760
761     gtk_widget_destroy(dialog);
762
763     return NULL;
764 }
765
766 GCallback tape_on(GtkWidget* w, void* d) {
767     if (rec_dont_care)
768         return 0;
769
770     if (gtk_check_menu_item_get_active(GTK_CHECK_MENU_ITEM(w))) {
771         rec_dont_care = 1;
772         gtk_check_menu_item_set_active(GTK_CHECK_MENU_ITEM(w), 0);
773         select_rec_file();
774     } else {
775         tX_engine::get_instance()->set_recording_request(false);
776     }
777
778     return NULL;
779 }
780
781 void grab_on(GtkWidget* w, void* d) {
782     if (gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(w))) {
783         if (mouse.grab() != 0) {
784             //tX_engine::get_instance()->set_grab_request();
785             // TODO: handle error
786         }
787     }
788     grab_status = 1;
789 }
790
791 void grab_off() {
792     grab_status = 0;
793 }
794
795 gboolean quit() {
796     if (globals.quit_confirm) {
797         GtkWidget* dialog = gtk_message_dialog_new(GTK_WINDOW(main_window),
798             GTK_DIALOG_DESTROY_WITH_PARENT, GTK_MESSAGE_QUESTION, GTK_BUTTONS_YES_NO,
799             "Exit terminatorX and lose all unsaved data?");
800
801         int res = gtk_dialog_run(GTK_DIALOG(dialog));
802         gtk_widget_destroy(dialog);
803
804         if (res != GTK_RESPONSE_YES) {
805             return TRUE;
806         }
807     }
808
809     tX_shutdown = true;
810
811     turn_audio_off();
812     vtt_class::delete_all();
813
814     if (update_tag) {
815         g_source_remove(update_tag);
816     }
817     GtkAllocation allocation;
818     gtk_widget_get_allocation(GTK_WIDGET(main_window), &allocation);
819     globals.width = allocation.width;
820     globals.height = allocation.height;
821
822     gtk_main_quit();
823
824     return true;
825 }
826
827 void mplcfitx()
828 /* Most Probably Least Called Function In terminatorX :) */
829 {
830     show_about(0);
831 }
832
833 GCallback seq_play(GtkWidget* w, void*) {
834     if ((sequencer.is_empty()) && (!gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(seq_rec_btn)))) {
835         if (gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(w))) {
836             tx_note("Sequencer playback triggered - but no events recorded yet - nothing to playback!\n\nTo perform live with terminatorX just activate the audio engine with the \"Power\" button.");
837             gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(w), 0);
838         }
839     } else {
840         if (seq_stop_override)
841             return NULL;
842
843         seq_adj_care = 0;
844         gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(w), 1);
845         sequencer.trig_play();
846
847         gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(engine_btn), 1);
848     }
849
850     return NULL;
851 }
852
853 GCallback seq_stop(GtkWidget* w, void*) {
854     if (!sequencer_ready)
855         return NULL;
856     sequencer.trig_stop();
857     seq_adj_care = 1;
858     seq_stop_override = 1;
859     gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(seq_play_btn), 0);
860     gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(seq_rec_btn), 0);
861     while (gtk_events_pending())
862         gtk_main_iteration();
863     seq_stop_override = 0;
864     gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(engine_btn), 0);
865     gtk_widget_set_sensitive(seq_slider, 1);
866     gtk_widget_set_sensitive(engine_btn, 1);
867     gtk_widget_set_sensitive(seq_rec_btn, 1);
868
869     return NULL;
870 }
871
872 GCallback seq_rec(GtkWidget* w, void*) {
873     seq_adj_care = 0;
874     gtk_widget_set_sensitive(seq_slider, 0);
875
876     if (seq_stop_override)
877         return NULL;
878     gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(w), 1);
879     gtk_widget_set_sensitive(engine_btn, 0);
880     gtk_widget_set_sensitive(seq_rec_btn, 0);
881     sequencer.trig_rec();
882
883     return NULL;
884 }
885
886 void seq_update_entry(const guint32 timestamp) {
887     char buffer[20];
888     guint32 samples;
889     guint32 minu, sec, hun;
890     guint32 sr;
891
892     samples = timestamp * vtt_class::get_mix_buffer_size();
893     sr = vtt_class::get_last_sample_rate();
894
895     if (samples > 0) {
896         minu = samples / (sr * 60);
897         samples -= (sr * 60) * minu;
898
899         sec = samples / sr;
900         samples -= sr * sec;
901
902         hun = samples / (sr / 100);
903     } else {
904         minu = sec = hun = 0;
905     }
906
907     sprintf(buffer, "%02i:%02i.%02i", minu, sec, hun);
908     gtk_entry_set_text(GTK_ENTRY(seq_entry), buffer);
909 }
910
911 void seq_update() {
912     seq_update_entry(sequencer.get_timestamp());
913     gtk_adjustment_set_value(seq_adj, sequencer.get_timestamp_as_float());
914 }
915 gboolean seq_slider_released(GtkWidget* wid, void* d) {
916     seq_adj_care = 0;
917     gtk_widget_set_sensitive(seq_slider, 0);
918     sequencer.forward_to_start_timestamp(0);
919     gtk_widget_set_sensitive(seq_slider, 1);
920     seq_adj_care = 1;
921
922     return FALSE;
923 }
924 void sequencer_move(GtkWidget* wid, void* d) {
925     guint32 pos;
926
927     if (seq_adj_care) {
928         pos = sequencer.set_start_timestamp((float)gtk_adjustment_get_value(GTK_ADJUSTMENT(wid)));
929         seq_update_entry(pos);
930     }
931 }
932
933 #define add_sep()                                            \
934     ;                                                        \
935     dummy = gtk_separator_new(GTK_ORIENTATION_HORIZONTAL);   \
936     gtk_box_pack_start(GTK_BOX(right_hbox), dummy, WID_FIX); \
937     gtk_widget_show(dummy);
938
939 #define add_sep2()                                         \
940     ;                                                      \
941     dummy = gtk_separator_new(GTK_ORIENTATION_HORIZONTAL); \
942     gtk_box_pack_end(GTK_BOX(status_box), dummy, WID_FIX); \
943     gtk_widget_show(dummy);
944
945 #ifdef USE_X11
946 void fullscreen_toggle(GtkCheckMenuItem* item, gpointer data);
947 #endif
948
949 void display_help();
950 void display_browser();
951
952 tX_seqpar* del_sp = NULL;
953 vtt_class* del_vtt = NULL;
954 tx_menu_del_mode menu_del_mode = ALL_EVENTS_ALL_TURNTABLES;
955
956 GtkWidget* del_dialog = NULL;
957
958 GCallback menu_delete_all_events(GtkWidget*, void* param) {
959     del_dialog = create_tx_del_mode();
960     tX_set_icon(del_dialog);
961
962     GtkWidget* label = lookup_widget(del_dialog, "delmode_label");
963
964     menu_del_mode = ALL_EVENTS_ALL_TURNTABLES;
965
966     gtk_label_set_markup(GTK_LABEL(label), "Delete <b>all</b> events for <b>all</b> turntables.");
967     gtk_widget_show(del_dialog);
968
969     return NULL;
970 }
971
972 GCallback menu_delete_all_events_for_vtt(GtkWidget*, vtt_class* vtt) {
973     if (!vtt) {
974         tX_error("No vtt passed to menu_delete_all_events_for_vtt().");
975         return FALSE;
976     }
977
978     char label_str[512];
979
980     del_dialog = create_tx_del_mode();
981     tX_set_icon(del_dialog);
982
983     del_vtt = vtt;
984     GtkWidget* label = lookup_widget(del_dialog, "delmode_label");
985
986     menu_del_mode = ALL_EVENTS_FOR_TURNTABLE;
987
988     sprintf(label_str, "Delete <b>all</b> events for turntable <b>%s</b>.", vtt->name);
989     gtk_label_set_markup(GTK_LABEL(label), label_str);
990     gtk_widget_show(del_dialog);
991
992     return NULL;
993 }
994
995 GCallback menu_delete_all_events_for_sp(GtkWidget*, tX_seqpar* sp) {
996     if (!sp) {
997         tX_error("No sp passed to menu_delete_all_events_for_sp().");
998         return FALSE;
999     }
1000
1001     char label_str[512];
1002
1003     del_dialog = create_tx_del_mode();
1004     tX_set_icon(del_dialog);
1005
1006     GtkWidget* label = lookup_widget(del_dialog, "delmode_label");
1007
1008     menu_del_mode = ALL_EVENTS_FOR_SP;
1009     del_sp = sp;
1010     sprintf(label_str, "Delete all <b>%s</b> events for turntable '%s'.", sp->get_name(), sp->get_vtt_name());
1011     gtk_label_set_markup(GTK_LABEL(label), label_str);
1012     gtk_widget_show(del_dialog);
1013
1014     return NULL;
1015 }
1016
1017 static GtkWidget* table_menu = NULL;
1018 static GtkWidget* table_menu_item = NULL;
1019
1020 GCallback create_table_sequencer_menu(GtkWidget* widget, void* param) {
1021     char label[328];
1022     table_menu = gtk_menu_new();
1023
1024     list<vtt_class*>::iterator vtt;
1025
1026     for (vtt = vtt_class::main_list.begin(); vtt != vtt_class::main_list.end(); vtt++) {
1027         GtkWidget* menu_item = gtk_menu_item_new_with_label((*vtt)->name);
1028         gtk_container_add(GTK_CONTAINER(table_menu), menu_item);
1029         gtk_widget_show(menu_item);
1030
1031         GtkWidget* seqpar_menu = gtk_menu_new();
1032         list<tX_seqpar*>::iterator sp;
1033
1034         GtkWidget* all = gtk_menu_item_new_with_label("Delete All Events");
1035         gtk_container_add(GTK_CONTAINER(seqpar_menu), all);
1036         g_signal_connect(all, "activate", (GCallback)menu_delete_all_events_for_vtt, (*vtt));
1037         gtk_widget_show(all);
1038
1039         GtkWidget* sep = gtk_menu_item_new();
1040         gtk_widget_show(sep);
1041         gtk_container_add(GTK_CONTAINER(seqpar_menu), sep);
1042         gtk_widget_set_sensitive(sep, FALSE);
1043
1044         for (sp = tX_seqpar::all->begin(); sp != tX_seqpar::all->end(); sp++) {
1045             if ((*sp)->vtt == (*vtt)) {
1046                 sprintf(label, "Delete '%s' Events", (*sp)->get_name());
1047                 GtkWidget* menu_item = gtk_menu_item_new_with_label(label);
1048                 g_signal_connect(menu_item, "activate", (GCallback)menu_delete_all_events_for_sp, (*sp));
1049                 gtk_container_add(GTK_CONTAINER(seqpar_menu), menu_item);
1050                 gtk_widget_show(menu_item);
1051             }
1052         }
1053         gtk_menu_item_set_submenu(GTK_MENU_ITEM(menu_item), seqpar_menu);
1054     }
1055
1056     gtk_menu_item_set_submenu(GTK_MENU_ITEM(table_menu_item), table_menu);
1057
1058     return NULL;
1059 }
1060
1061 GCallback toggle_confirm_events(GtkWidget* widget, void* dummy) {
1062     globals.confirm_events = gtk_check_menu_item_get_active(GTK_CHECK_MENU_ITEM(widget));
1063     return NULL;
1064 }
1065
1066 GCallback toggle_auto_assign(GtkWidget* widget, void* dummy) {
1067     globals.auto_assign_midi = gtk_check_menu_item_get_active(GTK_CHECK_MENU_ITEM(widget));
1068 #ifdef USE_ALSA_MIDI_IN
1069     tX_midiin::auto_assign_midi_mappings(NULL, NULL);
1070 #endif
1071     return NULL;
1072 }
1073
1074 void create_main_menu() {
1075     GtkWidget* menu_item;
1076     GtkWidget* sub_menu;
1077
1078     GtkAccelGroup* accel_group = gtk_accel_group_new();
1079     gtk_window_add_accel_group(GTK_WINDOW(main_window), accel_group);
1080
1081     /* FILE */
1082     menu_item = gtk_menu_item_new_with_mnemonic("_File");
1083     gtk_widget_show(menu_item);
1084     gtk_container_add(GTK_CONTAINER(main_menubar), menu_item);
1085
1086     sub_menu = gtk_menu_new();
1087     gtk_menu_item_set_submenu(GTK_MENU_ITEM(menu_item), sub_menu);
1088
1089     menu_item = gtk_menu_item_new_with_mnemonic("Load _Audio File");
1090     gtk_widget_show(menu_item);
1091     gtk_container_add(GTK_CONTAINER(sub_menu), menu_item);
1092     g_signal_connect(menu_item, "activate", (GCallback)load_audio, NULL);
1093
1094     menu_item = gtk_separator_menu_item_new();
1095     gtk_widget_show(menu_item);
1096     gtk_container_add(GTK_CONTAINER(sub_menu), menu_item);
1097
1098     menu_item = gtk_menu_item_new_with_mnemonic("_New Set");
1099     gtk_widget_show(menu_item);
1100     gtk_container_add(GTK_CONTAINER(sub_menu), menu_item);
1101     g_signal_connect(menu_item, "activate", (GCallback)new_tables, NULL);
1102
1103     menu_item = gtk_menu_item_new_with_mnemonic("_Open Set");
1104     gtk_widget_show(menu_item);
1105     gtk_container_add(GTK_CONTAINER(sub_menu), menu_item);
1106     g_signal_connect(menu_item, "activate", (GCallback)load_tables, NULL);
1107
1108     menu_item = gtk_menu_item_new_with_mnemonic("_Save Set");
1109     gtk_widget_show(menu_item);
1110     gtk_container_add(GTK_CONTAINER(sub_menu), menu_item);
1111     g_signal_connect(menu_item, "activate", (GCallback)save_tables, NULL);
1112
1113     menu_item = gtk_menu_item_new_with_label("Save Set As");
1114     gtk_widget_show(menu_item);
1115     gtk_container_add(GTK_CONTAINER(sub_menu), menu_item);
1116     g_signal_connect(menu_item, "activate", (GCallback)save_tables_as, NULL);
1117
1118     menu_item = gtk_separator_menu_item_new();
1119     gtk_widget_show(menu_item);
1120     gtk_container_add(GTK_CONTAINER(sub_menu), menu_item);
1121
1122     menu_item = gtk_menu_item_new_with_mnemonic("_Quit");
1123     gtk_widget_show(menu_item);
1124     gtk_container_add(GTK_CONTAINER(sub_menu), menu_item);
1125     g_signal_connect(menu_item, "activate", (GCallback)quit, NULL);
1126
1127     /* Turntables */
1128     menu_item = gtk_menu_item_new_with_mnemonic("_Turntables");
1129     gtk_widget_show(menu_item);
1130     gtk_container_add(GTK_CONTAINER(main_menubar), menu_item);
1131
1132     sub_menu = gtk_menu_new();
1133     gtk_menu_item_set_submenu(GTK_MENU_ITEM(menu_item), sub_menu);
1134
1135     menu_item = gtk_menu_item_new_with_mnemonic("_Add Turntable");
1136     gtk_widget_show(menu_item);
1137     gtk_container_add(GTK_CONTAINER(sub_menu), menu_item);
1138     gtk_widget_add_accelerator(menu_item, "activate", accel_group, GDK_KEY_A, GDK_CONTROL_MASK, GTK_ACCEL_VISIBLE);
1139     g_signal_connect(menu_item, "activate", (GCallback)new_table, NULL);
1140
1141     menu_item = gtk_separator_menu_item_new();
1142     gtk_widget_show(menu_item);
1143     gtk_container_add(GTK_CONTAINER(sub_menu), menu_item);
1144
1145     menu_item = gtk_menu_item_new_with_mnemonic("Assign _Default MIDI Mappings");
1146     gtk_widget_show(menu_item);
1147     gtk_container_add(GTK_CONTAINER(sub_menu), menu_item);
1148     gtk_widget_add_accelerator(menu_item, "activate", accel_group, GDK_KEY_M, GDK_CONTROL_MASK, GTK_ACCEL_VISIBLE);
1149
1150 #ifdef USE_ALSA_MIDI_IN
1151     g_signal_connect(menu_item, "activate", G_CALLBACK(tX_midiin::auto_assign_midi_mappings), (void*)true);
1152 #else
1153     gtk_widget_set_sensitive(menu_item, FALSE);
1154 #endif
1155
1156     menu_item = gtk_check_menu_item_new_with_mnemonic("A_uto Assign Default MIDI Mappings");
1157     gtk_widget_show(menu_item);
1158     gtk_container_add(GTK_CONTAINER(sub_menu), menu_item);
1159     gtk_check_menu_item_set_active(GTK_CHECK_MENU_ITEM(menu_item), globals.auto_assign_midi);
1160 #ifdef USE_ALSA_MIDI_IN
1161     g_signal_connect(menu_item, "activate", (GCallback)toggle_auto_assign, NULL);
1162 #else
1163     gtk_widget_set_sensitive(menu_item, FALSE);
1164 #endif
1165
1166     menu_item = gtk_menu_item_new_with_mnemonic("_Clear MIDI Mappings");
1167     gtk_widget_show(menu_item);
1168     gtk_container_add(GTK_CONTAINER(sub_menu), menu_item);
1169     gtk_widget_add_accelerator(menu_item, "activate", accel_group, GDK_KEY_C, GDK_CONTROL_MASK, GTK_ACCEL_VISIBLE);
1170
1171 #ifdef USE_ALSA_MIDI_IN
1172     g_signal_connect(menu_item, "activate", G_CALLBACK(tX_midiin::clear_midi_mappings), (void*)true);
1173 #else
1174     gtk_widget_set_sensitive(menu_item, FALSE);
1175 #endif
1176
1177     menu_item = gtk_separator_menu_item_new();
1178     gtk_widget_show(menu_item);
1179     gtk_container_add(GTK_CONTAINER(sub_menu), menu_item);
1180
1181     menu_item = gtk_check_menu_item_new_with_mnemonic("_Record Audio To Disk");
1182     rec_menu_item = menu_item;
1183     gtk_widget_add_accelerator(menu_item, "activate", accel_group, GDK_KEY_R, GDK_CONTROL_MASK, GTK_ACCEL_VISIBLE);
1184     gtk_widget_show(menu_item);
1185     gtk_container_add(GTK_CONTAINER(sub_menu), menu_item);
1186     g_signal_connect(menu_item, "activate", (GCallback)tape_on, NULL);
1187
1188     /* Sequencer */
1189
1190     menu_item = gtk_menu_item_new_with_mnemonic("_Sequencer");
1191     gtk_widget_show(menu_item);
1192     gtk_container_add(GTK_CONTAINER(main_menubar), menu_item);
1193
1194     sub_menu = gtk_menu_new();
1195     gtk_menu_item_set_submenu(GTK_MENU_ITEM(menu_item), sub_menu);
1196
1197     table_menu = gtk_menu_new();
1198     menu_item = gtk_menu_item_new_with_mnemonic("Delete _Events");
1199     delete_all_vtt_item = menu_item;
1200     table_menu_item = menu_item;
1201     gtk_menu_item_set_submenu(GTK_MENU_ITEM(table_menu_item), table_menu);
1202
1203     gtk_widget_show(menu_item);
1204     gtk_container_add(GTK_CONTAINER(sub_menu), menu_item);
1205     g_signal_connect_swapped(G_OBJECT(menu_item), "select", G_CALLBACK(create_table_sequencer_menu), NULL);
1206
1207     menu_item = gtk_menu_item_new_with_mnemonic("Delete _All Events");
1208     delete_all_item = menu_item;
1209     gtk_widget_show(menu_item);
1210     gtk_container_add(GTK_CONTAINER(sub_menu), menu_item);
1211     g_signal_connect(menu_item, "activate", (GCallback)menu_delete_all_events, NULL);
1212
1213     menu_item = gtk_separator_menu_item_new();
1214     gtk_widget_show(menu_item);
1215     gtk_container_add(GTK_CONTAINER(sub_menu), menu_item);
1216
1217     menu_item = gtk_check_menu_item_new_with_mnemonic("_Confirm Recorded Events");
1218     //rec_menu_item = menu_item;
1219     gtk_widget_show(menu_item);
1220     gtk_container_add(GTK_CONTAINER(sub_menu), menu_item);
1221     gtk_check_menu_item_set_active(GTK_CHECK_MENU_ITEM(menu_item), globals.confirm_events);
1222     g_signal_connect(menu_item, "activate", (GCallback)toggle_confirm_events, NULL);
1223
1224     /* Options */
1225     menu_item = gtk_menu_item_new_with_mnemonic("_Options");
1226     gtk_widget_show(menu_item);
1227     gtk_container_add(GTK_CONTAINER(main_menubar), menu_item);
1228
1229     sub_menu = gtk_menu_new();
1230     gtk_menu_item_set_submenu(GTK_MENU_ITEM(menu_item), sub_menu);
1231
1232     menu_item = gtk_check_menu_item_new_with_mnemonic("_Fullscreen");
1233     fullscreen_item = menu_item;
1234     gtk_widget_show(menu_item);
1235     gtk_container_add(GTK_CONTAINER(sub_menu), menu_item);
1236
1237     gtk_check_menu_item_set_active(GTK_CHECK_MENU_ITEM(menu_item), globals.fullscreen_enabled);
1238     gtk_widget_add_accelerator(menu_item, "activate", accel_group, GDK_KEY_F11, (GdkModifierType)0, GTK_ACCEL_VISIBLE);
1239     g_signal_connect(menu_item, "activate", (GCallback)fullscreen_toggle, NULL);
1240
1241 #ifndef USE_X11
1242     gtk_widget_set_sensitive(menu_item, False);
1243 #endif
1244     menu_item = gtk_separator_menu_item_new();
1245     gtk_widget_show(menu_item);
1246     gtk_container_add(GTK_CONTAINER(sub_menu), menu_item);
1247
1248     menu_item = gtk_menu_item_new_with_mnemonic("_Preferences");
1249     gtk_widget_show(menu_item);
1250     gtk_container_add(GTK_CONTAINER(sub_menu), menu_item);
1251     g_signal_connect(menu_item, "activate", (GCallback)display_options, NULL);
1252
1253     /* HELP */
1254     menu_item = gtk_menu_item_new_with_mnemonic("_Help");
1255     gtk_widget_show(menu_item);
1256     gtk_container_add(GTK_CONTAINER(main_menubar), menu_item);
1257
1258     sub_menu = gtk_menu_new();
1259     gtk_menu_item_set_submenu(GTK_MENU_ITEM(menu_item), sub_menu);
1260
1261     menu_item = gtk_menu_item_new_with_mnemonic("_Contents");
1262     gtk_widget_show(menu_item);
1263     gtk_container_add(GTK_CONTAINER(sub_menu), menu_item);
1264     g_signal_connect(menu_item, "activate", (GCallback)display_help, NULL);
1265     gtk_widget_add_accelerator(menu_item, "activate", accel_group, GDK_KEY_F1, (GdkModifierType)0, GTK_ACCEL_VISIBLE);
1266
1267     menu_item = gtk_menu_item_new_with_mnemonic("_About");
1268     gtk_widget_show(menu_item);
1269     gtk_container_add(GTK_CONTAINER(sub_menu), menu_item);
1270     g_signal_connect(menu_item, "activate", (GCallback)mplcfitx, NULL);
1271
1272     menu_item = gtk_separator_menu_item_new();
1273     gtk_widget_show(menu_item);
1274     gtk_container_add(GTK_CONTAINER(sub_menu), menu_item);
1275
1276     menu_item = gtk_menu_item_new_with_mnemonic("_Visit terminatorX.org");
1277     gtk_widget_show(menu_item);
1278     gtk_container_add(GTK_CONTAINER(sub_menu), menu_item);
1279     g_signal_connect(menu_item, "activate", (GCallback)display_browser, NULL);
1280 }
1281
1282 void create_maingui(int x, int y) {
1283     GtkWidget* mother_of_all_boxen;
1284     GtkWidget* main_vbox;
1285     GtkWidget* right_hbox;
1286     GtkWidget* left_hbox;
1287     GtkWidget* control_box;
1288     GtkAdjustment* dumadj;
1289     GtkWidget* dummy;
1290     GtkWidget* main_vol_box;
1291     GtkWidget* status_box;
1292     GtkWidget* wrapbox;
1293
1294     main_window = gtk_window_new(GTK_WINDOW_TOPLEVEL);
1295
1296     gtk_window_set_title(GTK_WINDOW(main_window), "terminatorX");
1297
1298     gtk_window_set_default_size(GTK_WINDOW(main_window), x, y);
1299     gtk_widget_realize(main_window);
1300
1301     GtkWidget* posLabel = gtk_label_new("Pos:");
1302     PangoRectangle ink_rect;
1303     PangoRectangle logical_rect;
1304     pango_layout_get_pixel_extents(gtk_label_get_layout(GTK_LABEL(posLabel)), &ink_rect, &logical_rect);
1305     tX_debug("ink extent: x: %i, y: %i, width: %i, height %i ", ink_rect.x, ink_rect.y, ink_rect.width, ink_rect.height);
1306     tX_debug("logical extent: x: %i, y: %i, width: %i, height %i ", logical_rect.x, logical_rect.y, logical_rect.width, logical_rect.height);
1307     fontHeight = logical_rect.height - logical_rect.y;
1308
1309     tx_icons_init(fontHeight);
1310
1311     wrapbox = gtk_box_new(GTK_ORIENTATION_VERTICAL, 5);
1312     gtk_container_add(GTK_CONTAINER(main_window), wrapbox);
1313     gtk_widget_show(wrapbox);
1314
1315     main_menubar = gtk_menu_bar_new();
1316     gtk_box_pack_start(GTK_BOX(wrapbox), main_menubar, WID_FIX);
1317     gtk_widget_show(main_menubar);
1318
1319     mother_of_all_boxen = gtk_box_new(GTK_ORIENTATION_VERTICAL, 5);
1320     gtk_container_set_border_width(GTK_CONTAINER(mother_of_all_boxen), 5);
1321     gtk_widget_set_hexpand(mother_of_all_boxen, TRUE);
1322     gtk_widget_set_vexpand(mother_of_all_boxen, TRUE);
1323     gtk_container_add(GTK_CONTAINER(wrapbox), mother_of_all_boxen);
1324     gtk_widget_show(mother_of_all_boxen);
1325
1326     create_main_menu();
1327
1328     g_signal_connect(G_OBJECT(main_window), "motion_notify_event", G_CALLBACK(tx_mouse::motion_notify_wrap), &mouse);
1329     g_signal_connect(G_OBJECT(main_window), "button_press_event", G_CALLBACK(tx_mouse::button_press_wrap), &mouse);
1330     g_signal_connect(G_OBJECT(main_window), "button_release_event", G_CALLBACK(tx_mouse::button_release_wrap), &mouse);
1331     g_signal_connect(G_OBJECT(main_window), "key_press_event", G_CALLBACK(tx_mouse::key_press_wrap), &mouse);
1332     g_signal_connect(G_OBJECT(main_window), "key_release_event", G_CALLBACK(tx_mouse::key_release_wrap), &mouse);
1333
1334     main_vbox = gtk_box_new(GTK_ORIENTATION_HORIZONTAL, 5);
1335     gtk_box_pack_start(GTK_BOX(mother_of_all_boxen), main_vbox, WID_DYN);
1336     gtk_widget_show(main_vbox);
1337
1338     left_hbox = gtk_box_new(GTK_ORIENTATION_VERTICAL, 5);
1339     gtk_box_pack_start(GTK_BOX(main_vbox), left_hbox, WID_DYN);
1340     gtk_widget_show(left_hbox);
1341
1342     control_box = gtk_box_new(GTK_ORIENTATION_HORIZONTAL, 5);
1343     gtk_box_pack_start(GTK_BOX(left_hbox), control_box, WID_FIX);
1344     gtk_widget_show(control_box);
1345
1346     dummy = gtk_separator_new(GTK_ORIENTATION_HORIZONTAL);
1347     gtk_box_pack_start(GTK_BOX(left_hbox), dummy, WID_FIX);
1348     gtk_widget_show(dummy);
1349
1350     /* control_box contents */
1351
1352     dummy = tx_xpm_label_box(AUDIOENGINE, "Audio");
1353     gtk_box_pack_start(GTK_BOX(control_box), dummy, WID_FIX_BREATHE);
1354     gtk_widget_show(dummy);
1355
1356     dummy = tx_xpm_button_new(POWER, "Power ", 1);
1357     connect_button(dummy, audio_on, NULL);
1358     gtk_box_pack_start(GTK_BOX(control_box), dummy, WID_FIX);
1359     gui_set_tooltip(dummy, "Turn the audio engine on/off.");
1360     gtk_widget_show(dummy);
1361     engine_btn = dummy;
1362
1363     grab_button = tx_xpm_button_new(GRAB, "Mouse Grab ", 1);
1364     gtk_box_pack_start(GTK_BOX(control_box), grab_button, WID_FIX);
1365     connect_button(grab_button, grab_on, NULL);
1366     gui_set_tooltip(grab_button, "Enter the mouse grab mode operation. Press <ESCAPE> to exit grab mode.");
1367     gtk_widget_show(grab_button);
1368
1369     dummy = gtk_separator_new(GTK_ORIENTATION_VERTICAL);
1370     gtk_box_pack_start(GTK_BOX(control_box), dummy, WID_FIX);
1371     gtk_widget_show(dummy);
1372
1373     dummy = tx_xpm_label_box(SEQUENCER, "Seq.");
1374     gtk_box_pack_start(GTK_BOX(control_box), dummy, WID_FIX_BREATHE);
1375     gtk_widget_show(dummy);
1376
1377     dummy = tx_xpm_button_new(PLAY, "Play ", 1);
1378     connect_button(dummy, seq_play, NULL);
1379     seq_play_btn = dummy;
1380     gtk_box_pack_start(GTK_BOX(control_box), dummy, WID_FIX);
1381     gui_set_tooltip(dummy, "Playback previously recorded events from the sequencer. This will turn on the audio engine automagically.");
1382     gtk_widget_show(dummy);
1383
1384     dummy = tx_xpm_button_new(STOP, "Stop ", 0);
1385     seq_stop_btn = dummy;
1386     connect_button(dummy, seq_stop, NULL);
1387     gtk_box_pack_start(GTK_BOX(control_box), dummy, WID_FIX);
1388     gui_set_tooltip(dummy, "Stop the playback of sequencer events.");
1389     gtk_widget_show(dummy);
1390
1391     dummy = tx_xpm_button_new(RECORD, "Record ", 1);
1392     connect_button(dummy, seq_rec, NULL);
1393     seq_rec_btn = dummy;
1394     gtk_box_pack_start(GTK_BOX(control_box), dummy, WID_FIX);
1395     gui_set_tooltip(dummy, "Enable recording of *events* into the sequencer. All touched controls will be recorded. Existing events for the song-time recording will be overwritten for touched controls.");
1396     gtk_widget_show(dummy);
1397
1398     gtk_box_pack_start(GTK_BOX(control_box), posLabel, WID_FIX_BREATHE);
1399     gtk_widget_show(posLabel);
1400
1401     dummy = gtk_entry_new();
1402     gtk_entry_set_max_length(GTK_ENTRY(dummy), 12);
1403     seq_entry = dummy;
1404     //gtk_widget_set_usize(dummy, 65, 20);
1405     gtk_entry_set_text(GTK_ENTRY(dummy), "00:00.00");
1406 #if GTK_CHECK_VERSION(2, 4, 0)
1407     gtk_entry_set_alignment(GTK_ENTRY(dummy), 0.5);
1408 #endif
1409     gtk_entry_set_width_chars(GTK_ENTRY(dummy), 9);
1410     gtk_entry_set_max_width_chars(GTK_ENTRY(dummy), 9);
1411     gtk_box_pack_start(GTK_BOX(control_box), dummy, WID_FIX);
1412     gtk_widget_show(dummy);
1413
1414     dumadj = (GtkAdjustment*)gtk_adjustment_new(0, 0, 100, 0.1, 1, 1);
1415     seq_adj = dumadj;
1416     connect_adj(dumadj, sequencer_move, NULL);
1417     dummy = gtk_scale_new(GTK_ORIENTATION_HORIZONTAL, dumadj);
1418     gtk_widget_set_size_request(dummy, 65, 20);
1419     seq_slider = dummy;
1420     g_signal_connect(G_OBJECT(seq_slider), "button-release-event", (GCallback)seq_slider_released, NULL);
1421     gtk_scale_set_draw_value(GTK_SCALE(dummy), FALSE);
1422
1423     gui_set_tooltip(dummy, "Select the start position for the sequencer in song-time.");
1424     gtk_box_pack_start(GTK_BOX(control_box), dummy, WID_DYN);
1425     gtk_widget_show(dummy);
1426
1427     dummy = gtk_box_new(GTK_ORIENTATION_HORIZONTAL, 2);
1428     gtk_box_pack_start(GTK_BOX(left_hbox), dummy, WID_DYN);
1429     gtk_widget_show(dummy);
1430
1431     tt_parent = dummy;
1432
1433     panel_bar = gtk_box_new(GTK_ORIENTATION_HORIZONTAL, 0);
1434     gtk_box_set_homogeneous(GTK_BOX(panel_bar), TRUE);
1435     gtk_box_pack_start(GTK_BOX(left_hbox), panel_bar, WID_FIX);
1436
1437     control_parent = gtk_box_new(GTK_ORIENTATION_HORIZONTAL, 4);
1438     gtk_box_pack_start(GTK_BOX(tt_parent), control_parent, WID_FIX);
1439     gtk_widget_show(control_parent);
1440
1441     dummy = gtk_separator_new(GTK_ORIENTATION_VERTICAL);
1442     gtk_box_pack_start(GTK_BOX(tt_parent), dummy, WID_FIX);
1443     gtk_widget_show(dummy);
1444
1445     audio_parent = gtk_box_new(GTK_ORIENTATION_VERTICAL, 2);
1446     gtk_box_pack_start(GTK_BOX(tt_parent), audio_parent, WID_DYN);
1447     gtk_widget_show(audio_parent);
1448
1449     dummy = gtk_separator_new(GTK_ORIENTATION_VERTICAL);
1450     gtk_box_pack_start(GTK_BOX(main_vbox), dummy, WID_FIX);
1451     gtk_widget_show(dummy);
1452
1453     right_hbox = gtk_box_new(GTK_ORIENTATION_VERTICAL, 5);
1454     gtk_box_pack_start(GTK_BOX(main_vbox), right_hbox, WID_FIX);
1455     gtk_widget_show(right_hbox);
1456
1457     /* Main */
1458
1459     dummy = gtk_label_new(NULL);
1460     gtk_label_set_markup(GTK_LABEL(dummy), "<b>Main</b>");
1461     gtk_box_pack_start(GTK_BOX(right_hbox), dummy, WID_FIX);
1462     gtk_widget_show(dummy);
1463
1464     dummy = gtk_separator_new(GTK_ORIENTATION_HORIZONTAL);
1465     gtk_box_pack_start(GTK_BOX(right_hbox), dummy, WID_FIX);
1466     gtk_widget_show(dummy);
1467
1468     /* Pitch */
1469
1470     dumadj = (GtkAdjustment*)gtk_adjustment_new(globals.pitch, -3, 3, 0.001, 0.001, 0.01);
1471     pitch_adj = dumadj;
1472     connect_adj(dumadj, main_pitch_changed, NULL);
1473
1474     tX_extdial* pdial = new tX_extdial("Pitch", pitch_adj, &sp_main_pitch, true);
1475     gtk_box_pack_start(GTK_BOX(right_hbox), pdial->get_widget(), WID_FIX);
1476     gui_set_tooltip(pdial->get_entry(), "Use this dial to adjust the main pitch (affecting *all* turntables).");
1477
1478     dummy = gtk_separator_new(GTK_ORIENTATION_HORIZONTAL);
1479     gtk_box_pack_start(GTK_BOX(right_hbox), dummy, WID_FIX);
1480     gtk_widget_show(dummy);
1481
1482     /* Volume */
1483     main_vol_box = gtk_box_new(GTK_ORIENTATION_HORIZONTAL, 5);
1484     gtk_box_pack_start(GTK_BOX(right_hbox), main_vol_box, WID_DYN);
1485     gtk_widget_show(main_vol_box);
1486
1487     dumadj = (GtkAdjustment*)gtk_adjustment_new(globals.volume, 0, 2, 0.01, 0.05, 0.000);
1488     volume_adj = dumadj;
1489
1490     connect_adj(dumadj, main_volume_changed, NULL);
1491     dummy = gtk_scale_new(GTK_ORIENTATION_VERTICAL, dumadj);
1492     gtk_range_set_inverted(GTK_RANGE(dummy), TRUE);
1493     gtk_scale_set_draw_value(GTK_SCALE(dummy), FALSE);
1494     g_signal_connect(G_OBJECT(dummy), "button_press_event", (GCallback)tX_seqpar::tX_seqpar_press, &sp_main_volume);
1495
1496     gtk_box_pack_end(GTK_BOX(main_vol_box), dummy, WID_FIX);
1497     gtk_widget_show(dummy);
1498     gui_set_tooltip(dummy, "Adjust the main volume. This parameter will effect *all* turntables in the set.");
1499
1500     main_flash = gtk_tx_flash_new();
1501     gtk_box_pack_end(GTK_BOX(main_vol_box), main_flash, WID_DYN);
1502     gtk_widget_show(main_flash);
1503
1504     dummy = gtk_label_new("Volume");
1505     gtk_box_pack_start(GTK_BOX(right_hbox), dummy, WID_FIX);
1506     gtk_widget_show(dummy);
1507
1508     /* STATUS BOX */
1509     dummy = gtk_separator_new(GTK_ORIENTATION_HORIZONTAL);
1510     gtk_box_pack_start(GTK_BOX(right_hbox), dummy, WID_FIX);
1511     gtk_widget_show(dummy);
1512
1513     status_box = gtk_box_new(GTK_ORIENTATION_VERTICAL, 0);
1514     gtk_box_pack_start(GTK_BOX(right_hbox), status_box, WID_FIX);
1515     gtk_widget_show(status_box);
1516
1517     dummy = gtk_label_new("0");
1518     used_mem = dummy;
1519     gtk_widget_set_halign(dummy, GTK_ALIGN_END);
1520     gtk_box_pack_end(GTK_BOX(status_box), dummy, WID_FIX);
1521     gtk_widget_show(dummy);
1522
1523     add_sep2();
1524
1525     dummy = gtk_label_new("v" VERSION);
1526     gtk_widget_set_halign(dummy, GTK_ALIGN_END);
1527     gtk_box_pack_end(GTK_BOX(status_box), dummy, WID_FIX);
1528     gtk_widget_show(dummy);
1529
1530     add_sep2();
1531
1532     dummy = gtk_label_new(NULL);
1533     gtk_label_set_markup(GTK_LABEL(dummy), "<b>Status</b>");
1534     gtk_box_pack_end(GTK_BOX(status_box), dummy, WID_FIX);
1535     gtk_widget_show(dummy);
1536
1537     /* END GUI */
1538
1539     gtk_widget_set_sensitive(grab_button, 0);
1540
1541     new_table(NULL, NULL); // to give the user something to start with ;)
1542
1543     g_signal_connect(G_OBJECT(main_window), "delete-event", (GCallback)quit, NULL);
1544
1545     //  if (globals.tooltips) gtk_tooltips_enable(gui_tooltips);
1546     //  else gtk_tooltips_disable(gui_tooltips);
1547     //  TODO: Check for global enable/disable of tooltips
1548 }
1549
1550 gfloat old_percent = -1;
1551
1552 void note_destroy(GtkWidget* widget, GtkWidget* mbox) {
1553     gtk_widget_destroy(GTK_WIDGET(mbox));
1554 }
1555
1556 void tx_note(const char* message, bool isError, GtkWindow* window) {
1557     if (!window)
1558         window = GTK_WINDOW(main_window);
1559
1560     GtkWidget* dialog = gtk_message_dialog_new_with_markup(window,
1561         GTK_DIALOG_DESTROY_WITH_PARENT,
1562         isError ? GTK_MESSAGE_ERROR : GTK_MESSAGE_INFO, GTK_BUTTONS_CLOSE, "%s", "");
1563     gtk_message_dialog_set_markup(GTK_MESSAGE_DIALOG(dialog), message);
1564     gtk_dialog_run(GTK_DIALOG(dialog));
1565     gtk_widget_destroy(dialog);
1566 }
1567
1568 void tx_l_note(const char* message) {
1569     char buffer[4096] = "Plugin info:\n\n";
1570     strcat(buffer, message);
1571
1572     GtkWidget* dialog = gtk_message_dialog_new(GTK_WINDOW(main_window),
1573         GTK_DIALOG_DESTROY_WITH_PARENT, GTK_MESSAGE_INFO, GTK_BUTTONS_CLOSE, "%s", message);
1574     gtk_dialog_run(GTK_DIALOG(dialog));
1575     gtk_widget_destroy(dialog);
1576 }
1577
1578 void add_to_panel_bar(GtkWidget* button) {
1579     buttons_on_panel_bar++;
1580     gtk_box_pack_start(GTK_BOX(panel_bar), button, WID_DYN);
1581     gtk_widget_show(panel_bar);
1582 }
1583
1584 void remove_from_panel_bar(GtkWidget* button) {
1585     buttons_on_panel_bar--;
1586     gtk_container_remove(GTK_CONTAINER(panel_bar), button);
1587     if (buttons_on_panel_bar == 0)
1588         gtk_widget_hide(panel_bar);
1589 }
1590
1591 #ifdef USE_X11
1592 /* Fullscreen code... */
1593 #define _WIN_LAYER_TOP -1
1594 #define _WIN_LAYER_NORMAL 4
1595 #define _NET_WM_STATE_REMOVE 0
1596 #define _NET_WM_STATE_ADD 1
1597 #define _NET_WM_STATE_TOGGLE 2
1598
1599 void fullscreen_toggle(GtkCheckMenuItem* item, gpointer data) {
1600     XEvent xev;
1601     Window win = GDK_WINDOW_XID(gtk_widget_get_window(main_window));
1602     Display* disp = GDK_WINDOW_XDISPLAY(gtk_widget_get_window(main_window));
1603
1604     globals.fullscreen_enabled = gtk_check_menu_item_get_active(GTK_CHECK_MENU_ITEM(fullscreen_item));
1605
1606     /* Top layer.. */
1607     xev.xclient.type = ClientMessage;
1608     xev.xclient.serial = 0;
1609     xev.xclient.send_event = True;
1610     xev.xclient.display = disp;
1611     xev.xclient.window = win;
1612     xev.xclient.message_type = gdk_x11_get_xatom_by_name("_WIN_LAYER");
1613     xev.xclient.format = 32;
1614     xev.xclient.data.l[0] = globals.fullscreen_enabled ? _WIN_LAYER_TOP : _WIN_LAYER_NORMAL;
1615     XSendEvent(disp, GDK_WINDOW_XID(gdk_get_default_root_window()),
1616         False, SubstructureRedirectMask | SubstructureNotifyMask,
1617         &xev);
1618
1619     /* Fullscreen */
1620     xev.xclient.type = ClientMessage;
1621     xev.xclient.serial = 0;
1622     xev.xclient.send_event = True;
1623     xev.xclient.display = disp;
1624     xev.xclient.window = win;
1625     xev.xclient.message_type = gdk_x11_get_xatom_by_name("_NET_WM_STATE");
1626     xev.xclient.format = 32;
1627     xev.xclient.data.l[0] = globals.fullscreen_enabled ? _NET_WM_STATE_ADD : _NET_WM_STATE_REMOVE;
1628     xev.xclient.data.l[1] = gdk_x11_atom_to_xatom(gdk_atom_intern("_NET_WM_STATE_FULLSCREEN", TRUE));
1629     xev.xclient.data.l[2] = gdk_x11_atom_to_xatom(GDK_NONE);
1630     XSendEvent(gdk_x11_get_default_xdisplay(), GDK_WINDOW_XID(gdk_get_default_root_window()),
1631         False, SubstructureRedirectMask | SubstructureNotifyMask,
1632         &xev);
1633 }
1634
1635 void fullscreen_setup() {
1636     if (globals.fullscreen_enabled) {
1637         fullscreen_toggle(NULL, NULL);
1638     }
1639 }
1640 #endif
1641
1642 void display_maingui() {
1643     GtkWidget* top;
1644     gtk_widget_realize(main_window);
1645     tX_set_icon(main_window);
1646     load_knob_pixs(fontHeight, gdk_window_get_scale_factor(gtk_widget_get_window(GTK_WIDGET(main_window))));
1647     gtk_widget_show(main_window);
1648     top = gtk_widget_get_toplevel(main_window);
1649     top_window = GDK_WINDOW(gtk_widget_get_window(top));
1650
1651 #ifdef USE_X11
1652     fullscreen_setup();
1653     x_window = gdk_x11_window_get_xid(gtk_widget_get_window(top));
1654 #endif
1655 }
1656
1657 pid_t help_child = 0;
1658 GTimer* help_timer = NULL;
1659 int help_tag = -1;
1660
1661 gboolean help_checker() {
1662     gdouble time;
1663     gulong ms;
1664     int status;
1665     int result = waitpid(help_child, &status, WNOHANG);
1666
1667     if (result == 0) {
1668         time = g_timer_elapsed(help_timer, &ms);
1669         if (time > 5) {
1670             /* 5 seconds and it's still running - so we assume everything's OK. */
1671             tX_debug("No longer waiting for gnome-help..");
1672             g_source_remove(help_tag);
1673             help_tag = -1;
1674         }
1675     } else {
1676         //yelp waitpid status does not allow determining success
1677         //printf("%i %i\n", WIFEXITED(status), WEXITSTATUS(status));
1678         ///* We are still here and the child exited - that could mean trouble. */
1679         //tx_note("Couldn't run the gnome-help command (alias \"yelp\") to display the terminatorX manual. Please ensure that \"yelp\" is installed.", true);
1680
1681         g_source_remove(help_tag);
1682         help_tag = -1;
1683     }
1684     return TRUE;
1685 }
1686
1687 #ifndef INSTALL_PREFIX
1688 #define INSTALL_PREFIX "/usr/local/share"
1689 #endif
1690
1691 void display_help() {
1692     help_child = fork();
1693
1694     if (help_tag != -1) {
1695         g_source_remove(help_tag);
1696         if (help_timer)
1697             g_timer_destroy(help_timer);
1698         help_child = 0;
1699         help_tag = -1;
1700         help_timer = NULL;
1701     }
1702
1703     if (help_child == 0) {
1704         // child
1705         execlp("gnome-help", "gnome-help", "help:terminatorX-manual", NULL);
1706         _exit(-1);
1707     } else if (help_child == -1) {
1708         tx_note("System error: couldn't fork() to run the help process.", true);
1709     } else {
1710         help_timer = g_timer_new();
1711         g_timer_start(help_timer);
1712
1713         help_tag = g_idle_add((GSourceFunc)help_checker, NULL);
1714     }
1715 }
1716
1717 pid_t browser_child = 0;
1718 GTimer* browser_timer = NULL;
1719 int browser_tag = -1;
1720
1721 void display_browser() {
1722     browser_child = fork();
1723
1724     if (browser_child == 0) {
1725         // child
1726         execlp(BROWSER1, BROWSER1, "https://terminatorX.org", NULL);
1727         execlp(BROWSER2, BROWSER2, "https://terminatorX.org", NULL);
1728         execlp(BROWSER3, BROWSER3, "https://terminatorX.org", NULL);
1729         _exit(-1);
1730     } else if (browser_child == -1) {
1731         tx_note("System error: couldn't fork() to run the browser process.", true);
1732     }
1733 }
1734
1735 GdkCursor* tX_cursor::cursors[MAX_CURSOR] = { NULL, NULL, NULL };
1736 tX_cursor::cursor_shape tX_cursor::current_shape = tX_cursor::DEFAULT_CURSOR;
1737
1738 void tX_cursor::set_cursor(cursor_shape shape) {
1739     GdkDisplay* display = gdk_window_get_display(gtk_widget_get_window(main_window));
1740     switch (shape) {
1741     case DEFAULT_CURSOR:
1742         cursors[shape] = NULL;
1743         break;
1744
1745     case WAIT_CURSOR:
1746         if (!cursors[shape])
1747             cursors[shape] = gdk_cursor_new_for_display(display, GDK_WATCH);
1748         break;
1749
1750     case WAIT_A_SECOND_CURSOR:
1751         /* FIXME: What's that short-time wait cursor's id? */
1752         if (!cursors[shape])
1753             cursors[shape] = gdk_cursor_new_for_display(display, GDK_WATCH);
1754         break;
1755
1756     default:
1757         tX_debug("No such cursor shape.");
1758         return;
1759     }
1760
1761     /* Still here? Ok... */
1762     current_shape = shape;
1763
1764     gdk_window_set_cursor(gtk_widget_get_window(main_window), cursors[shape]);
1765 }
1766
1767 GdkCursor* tX_cursor::get_cursor() {
1768     return cursors[current_shape];
1769 }
1770
1771 void tX_cursor::reset_cursor() {
1772     current_shape = DEFAULT_CURSOR;
1773     gdk_window_set_cursor(gtk_widget_get_window(main_window), cursors[current_shape]);
1774 }