always make log/error window visible when a non-info message is posted
[ardour.git] / libs / gtkmm2ext / gtkmm2ext / gtk_ui.h
1 /*
2     Copyright (C) 1999 Paul Barton-Davis 
3
4     This program is free software; you can redistribute it and/or modify
5     it under the terms of the GNU General Public License as published by
6     the Free Software Foundation; either version 2 of the License, or
7     (at your option) any later version.
8
9     This program is distributed in the hope that it will be useful,
10     but WITHOUT ANY WARRANTY; without even the implied warranty of
11     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12     GNU General Public License for more details.
13
14     You should have received a copy of the GNU General Public License
15     along with this program; if not, write to the Free Software
16     Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
17
18 */
19
20 #ifndef __pbd_gtk_ui_h__
21 #define __pbd_gtk_ui_h__
22
23 #include <string>
24 #include <map>
25
26 #include <stdint.h>
27 #include <setjmp.h>
28 #include <pthread.h>
29
30 #include <glibmm/thread.h>
31
32 #include <gtkmm/widget.h>
33 #include <gtkmm/style.h>
34 #ifndef GTK_NEW_TOOLTIP_API
35 #include <gtkmm/tooltips.h>
36 #endif
37 #include <gtkmm/textbuffer.h>
38 #include <gtkmm/main.h>
39 #include <gdkmm/color.h>
40 #include <pbd/abstract_ui.h>
41 #include <pbd/ringbufferNPT.h>
42  
43 #include <pbd/pool.h>
44 #include <pbd/error.h>
45 #include <pbd/receiver.h>
46
47 class Touchable;
48
49 namespace Gtkmm2ext {
50
51 class TextViewer;
52
53 extern BaseUI::RequestType NullMessage;
54 extern BaseUI::RequestType ErrorMessage;
55 extern BaseUI::RequestType CallSlot;
56 extern BaseUI::RequestType TouchDisplay;
57 extern BaseUI::RequestType StateChange;
58 extern BaseUI::RequestType SetTip;
59 extern BaseUI::RequestType AddIdle;
60 extern BaseUI::RequestType AddTimeout;
61
62 struct UIRequest : public BaseUI::BaseRequestObject {
63      
64      /* this once used anonymous unions to merge elements
65         that are never part of the same request. that makes
66         the creation of a legal copy constructor difficult
67         because of the semantics of the slot member.
68      */
69      
70     Touchable *display;
71     const char *msg;
72     Gtk::StateType new_state;
73     int (*function)(void *);
74     Gtk::Widget *widget;
75     Transmitter::Channel chn;
76     void *arg;
77     const char *msg2;
78
79     UIRequest () {
80             type = NullMessage;
81     }
82     
83     ~UIRequest () { 
84             if (type == ErrorMessage && msg) {
85                     /* msg was strdup()'ed */
86                     free ((char *)msg);
87             }
88     }
89 };
90
91 class UI : public Receiver, public AbstractUI<UIRequest>
92 {
93   public:
94         UI (std::string name, int *argc, char **argv[]);
95         virtual ~UI ();
96
97         static UI *instance() { return theGtkUI; }
98
99         /* receiver interface */
100
101         void receive (Transmitter::Channel, const char *);
102
103         /* Abstract UI interfaces */
104
105         bool caller_is_ui_thread ();
106
107         static Glib::Thread* thread_id() { return gui_thread; }
108
109         /* Gtk-UI specific interfaces */
110
111         bool running ();
112         void quit    ();
113         int  load_rcfile (std::string, bool themechange = false);
114         void run (Receiver &old_receiver);
115
116         void set_state (Gtk::Widget *w, Gtk::StateType state);
117         void popup_error (const std::string& text);
118         void flush_pending ();
119         void toggle_errors ();
120         void show_errors ();
121         void touch_display (Touchable *);
122         void set_tip (Gtk::Widget &w, const gchar *tip);
123         void set_tip (Gtk::Widget &w, const std::string &tip);
124         void set_tip (Gtk::Widget *w, const gchar *tip, const gchar *hlp="");
125         void idle_add (int (*func)(void *), void *arg);
126
127         Gtk::Main& main() const { return *theMain; }
128
129         template<class T> static bool idle_delete (T *obj) { delete obj; return false; }
130         template<class T> static void delete_when_idle (T *obj) {
131                 Glib::signal_idle().connect (bind (slot (&UI::idle_delete<T>), obj));
132         }
133
134         template<class T> void delete_in_self (T *obj) {
135                 call_slot (boost::bind (&UI::delete_in_self, this, obj));
136         }
137
138         Gdk::Color get_color (const std::string& prompt, bool& picked, const Gdk::Color *initial = 0);
139
140         /* starting is sent just before we enter the main loop,
141            stopping just after we return from it (at the top level)
142         */
143
144         sigc::signal<void> starting;
145         sigc::signal<void> stopping;
146
147         sigc::signal<void> theme_changed;
148
149         static bool just_hide_it (GdkEventAny *, Gtk::Window *);
150
151         static Glib::Thread* the_gui_thread() { return gui_thread; }
152
153   protected:
154         virtual void handle_fatal (const char *);
155         virtual void display_message (const char *prefix, gint prefix_len,
156                         Glib::RefPtr<Gtk::TextBuffer::Tag> ptag, Glib::RefPtr<Gtk::TextBuffer::Tag> mtag,
157                         const char *msg);
158
159   private:
160         static UI *theGtkUI;
161         static Glib::Thread* gui_thread;
162         bool _active;
163         Gtk::Main *theMain;
164 #ifndef GTK_NEW_TOOLTIP_API
165         Gtk::Tooltips *tips;
166 #endif
167         TextViewer *errors;
168         Glib::RefPtr<Gtk::TextBuffer::Tag> error_ptag;
169         Glib::RefPtr<Gtk::TextBuffer::Tag> error_mtag;
170         Glib::RefPtr<Gtk::TextBuffer::Tag> fatal_ptag;
171         Glib::RefPtr<Gtk::TextBuffer::Tag> fatal_mtag;
172         Glib::RefPtr<Gtk::TextBuffer::Tag> info_ptag;
173         Glib::RefPtr<Gtk::TextBuffer::Tag> info_mtag;
174         Glib::RefPtr<Gtk::TextBuffer::Tag> warning_ptag;
175         Glib::RefPtr<Gtk::TextBuffer::Tag> warning_mtag;
176
177         static void signal_pipe_callback (void *, gint, GdkInputCondition);
178         void process_error_message (Transmitter::Channel, const char *);
179         void do_quit ();
180
181         void color_selection_done (bool status);
182         bool color_selection_deleted (GdkEventAny *);
183         bool color_picked;
184
185         void do_request (UIRequest*);
186 };
187
188 } /* namespace */
189
190 #endif /* __pbd_gtk_ui_h__ */