Rebase onto master
[xonotic/netradiant.git] / libs / uilib / uilib.cpp
1 #include "uilib.h"
2
3 #include <tuple>
4
5 #include <gtk/gtk.h>
6
7 #include "gtkutil/dialog.h"
8 #include "gtkutil/filechooser.h"
9 #include "gtkutil/messagebox.h"
10 #include "gtkutil/window.h"
11
12 namespace ui {
13
14     bool init(int *argc, char **argv[], char const *parameter_string, char const **error)
15     {
16         gtk_disable_setlocale();
17         static GOptionEntry entries[] = {{NULL}};
18         char const *translation_domain = NULL;
19         GError *gerror = NULL;
20         bool ret = gtk_init_with_args(argc, argv, parameter_string, entries, translation_domain, &gerror) != 0;
21         if (!ret) {
22             *error = gerror->message;
23         }
24         return ret;
25     }
26
27     void main()
28     {
29         gtk_main();
30     }
31
32     void process()
33     {
34         while (gtk_events_pending()) {
35             gtk_main_iteration();
36         }
37     }
38
39     Widget root;
40
41 #define IMPL(T, F) template<> _IMPL(T, F)
42 #define _IMPL(T, F) struct verify<T *> { using self = T; static self test(self it) { return self(F(it)); } }
43
44     template<class T>
45     struct verify;
46
47     template<class T> _IMPL(T,);
48
49 #define this (verify<self>::test(*static_cast<self>(this)))
50
51     IMPL(Editable, GTK_EDITABLE);
52
53     void IEditable::editable(bool value)
54     {
55         gtk_editable_set_editable(GTK_EDITABLE(this), value);
56     }
57
58     IMPL(Widget, GTK_WIDGET);
59
60     Widget::Widget() : Widget(nullptr)
61     {}
62
63     alert_response IWidget::alert(std::string text, std::string title, alert_type type, alert_icon icon)
64     {
65         auto ret = gtk_MessageBox(this, text.c_str(),
66                                   title.c_str(),
67                                   type == alert_type::OK ? eMB_OK :
68                                   type == alert_type::OKCANCEL ? eMB_OKCANCEL :
69                                   type == alert_type::YESNO ? eMB_YESNO :
70                                   type == alert_type::YESNOCANCEL ? eMB_YESNOCANCEL :
71                                   type == alert_type::NOYES ? eMB_NOYES :
72                                   eMB_OK,
73                                   icon == alert_icon::Default ? eMB_ICONDEFAULT :
74                                   icon == alert_icon::Error ? eMB_ICONERROR :
75                                   icon == alert_icon::Warning ? eMB_ICONWARNING :
76                                   icon == alert_icon::Question ? eMB_ICONQUESTION :
77                                   icon == alert_icon::Asterisk ? eMB_ICONASTERISK :
78                                   eMB_ICONDEFAULT
79         );
80         return
81                 ret == eIDOK ? alert_response::OK :
82                 ret == eIDCANCEL ? alert_response::CANCEL :
83                 ret == eIDYES ? alert_response::YES :
84                 ret == eIDNO ? alert_response::NO :
85                 alert_response::OK;
86     }
87
88     const char *
89     IWidget::file_dialog(bool open, const char *title, const char *path, const char *pattern, bool want_load,
90                          bool want_import, bool want_save)
91     {
92         return ::file_dialog(this, open, title, path, pattern, want_load, want_import, want_save);
93     }
94
95     void IWidget::show()
96     {
97         gtk_widget_show(this);
98     }
99
100     IMPL(Container, GTK_CONTAINER);
101
102     void IContainer::add(Widget widget)
103     {
104         gtk_container_add(this, widget);
105     }
106
107     void IContainer::remove(Widget widget)
108     {
109         gtk_container_remove(this, widget);
110     }
111
112     IMPL(Bin, GTK_BIN);
113
114     IMPL(Window, GTK_WINDOW);
115
116     Window::Window(window_type type) : Window(GTK_WINDOW(gtk_window_new(
117             type == window_type::TOP ? GTK_WINDOW_TOPLEVEL :
118             type == window_type::POPUP ? GTK_WINDOW_POPUP :
119             GTK_WINDOW_TOPLEVEL
120     )))
121     {}
122
123     Window IWindow::create_dialog_window(const char *title, void func(), void *data, int default_w, int default_h)
124     {
125         return Window(::create_dialog_window(this, title, func, data, default_w, default_h));
126     }
127
128     Window IWindow::create_modal_dialog_window(const char *title, ModalDialog &dialog, int default_w, int default_h)
129     {
130         return Window(::create_modal_dialog_window(this, title, dialog, default_w, default_h));
131     }
132
133     Window IWindow::create_floating_window(const char *title)
134     {
135         return Window(::create_floating_window(title, this));
136     }
137
138     std::uint64_t IWindow::on_key_press(bool (*f)(Widget widget, _GdkEventKey *event, void *extra), void *extra)
139     {
140         using f_t = decltype(f);
141         struct user_data {
142             f_t f;
143             void *extra;
144         } *pass = new user_data{f, extra};
145         auto dtor = [](user_data *data, GClosure *) {
146             delete data;
147         };
148         auto func = [](_GtkWidget *widget, GdkEventKey *event, user_data *args) -> bool {
149             return args->f(Widget(widget), event, args->extra);
150         };
151         auto clos = g_cclosure_new(G_CALLBACK(+func), pass, reinterpret_cast<GClosureNotify>(+dtor));
152         return g_signal_connect_closure(G_OBJECT(this), "key-press-event", clos, false);
153     }
154
155     void IWindow::add_accel_group(AccelGroup group)
156     {
157         gtk_window_add_accel_group(this, group);
158     }
159
160     IMPL(Alignment, GTK_ALIGNMENT);
161
162     Alignment::Alignment(float xalign, float yalign, float xscale, float yscale)
163             : Alignment(GTK_ALIGNMENT(gtk_alignment_new(xalign, yalign, xscale, yscale)))
164     {}
165
166     IMPL(Frame, GTK_FRAME);
167
168     Frame::Frame(const char *label) : Frame(GTK_FRAME(gtk_frame_new(label)))
169     {}
170
171     IMPL(Button, GTK_BUTTON);
172
173     Button::Button() : Button(GTK_BUTTON(gtk_button_new()))
174     {}
175
176     Button::Button(const char *label) : Button(GTK_BUTTON(gtk_button_new_with_label(label)))
177     {}
178
179     IMPL(ToggleButton, GTK_TOGGLE_BUTTON);
180
181     bool IToggleButton::active()
182     {
183         return gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(this)) != 0;
184     }
185
186     IMPL(CheckButton, GTK_CHECK_BUTTON);
187
188     CheckButton::CheckButton(const char *label) : CheckButton(GTK_CHECK_BUTTON(gtk_check_button_new_with_label(label)))
189     {}
190
191     IMPL(MenuItem, GTK_MENU_ITEM);
192
193     MenuItem::MenuItem() : MenuItem(GTK_MENU_ITEM(gtk_menu_item_new()))
194     {}
195
196     MenuItem::MenuItem(const char *label, bool mnemonic) : MenuItem(
197             GTK_MENU_ITEM((mnemonic ? gtk_menu_item_new_with_mnemonic : gtk_menu_item_new_with_label)(label)))
198     {}
199
200     IMPL(TearoffMenuItem, GTK_TEAROFF_MENU_ITEM);
201
202     TearoffMenuItem::TearoffMenuItem() : TearoffMenuItem(GTK_TEAROFF_MENU_ITEM(gtk_tearoff_menu_item_new()))
203     {}
204
205     IMPL(ComboBoxText, GTK_COMBO_BOX_TEXT);
206
207     ComboBoxText::ComboBoxText() : ComboBoxText(GTK_COMBO_BOX_TEXT(gtk_combo_box_text_new()))
208     {}
209
210     IMPL(ScrolledWindow, GTK_SCROLLED_WINDOW);
211
212     ScrolledWindow::ScrolledWindow() : ScrolledWindow(GTK_SCROLLED_WINDOW(gtk_scrolled_window_new(nullptr, nullptr)))
213     {}
214
215     IMPL(VBox, GTK_VBOX);
216
217     VBox::VBox(bool homogenous, int spacing) : VBox(GTK_VBOX(gtk_vbox_new(homogenous, spacing)))
218     {}
219
220     IMPL(HBox, GTK_HBOX);
221
222     HBox::HBox(bool homogenous, int spacing) : HBox(GTK_HBOX(gtk_hbox_new(homogenous, spacing)))
223     {}
224
225     IMPL(HPaned, GTK_HPANED);
226
227     HPaned::HPaned() : HPaned(GTK_HPANED(gtk_hpaned_new()))
228     {}
229
230     IMPL(VPaned, GTK_VPANED);
231
232     VPaned::VPaned() : VPaned(GTK_VPANED(gtk_vpaned_new()))
233     {}
234
235     IMPL(Menu, GTK_MENU);
236
237     Menu::Menu() : Menu(GTK_MENU(gtk_menu_new()))
238     {}
239
240     IMPL(Table, GTK_TABLE);
241
242     Table::Table(std::size_t rows, std::size_t columns, bool homogenous) : Table(
243             GTK_TABLE(gtk_table_new(rows, columns, homogenous))
244     )
245     {}
246
247     IMPL(TextView, GTK_TEXT_VIEW);
248
249     TextView::TextView() : TextView(GTK_TEXT_VIEW(gtk_text_view_new()))
250     {}
251
252     TreeView::TreeView() : TreeView(GTK_TREE_VIEW(gtk_tree_view_new()))
253     {}
254
255     TreeView::TreeView(TreeModel model) : TreeView(GTK_TREE_VIEW(gtk_tree_view_new_with_model(model)))
256     {}
257
258     IMPL(Label, GTK_LABEL);
259
260     Label::Label(const char *label) : Label(GTK_LABEL(gtk_label_new(label)))
261     {}
262
263     IMPL(Image, GTK_IMAGE);
264
265     Image::Image() : Image(GTK_IMAGE(gtk_image_new()))
266     {}
267
268     IMPL(Entry, GTK_ENTRY);
269
270     Entry::Entry() : Entry(GTK_ENTRY(gtk_entry_new()))
271     {}
272
273     Entry::Entry(std::size_t max_length) : Entry()
274     {
275         gtk_entry_set_max_length(this, static_cast<gint>(max_length));
276     }
277
278     IMPL(SpinButton, GTK_SPIN_BUTTON);
279
280     SpinButton::SpinButton(Adjustment adjustment, double climb_rate, std::size_t digits) : SpinButton(
281             GTK_SPIN_BUTTON(gtk_spin_button_new(adjustment, climb_rate, digits)))
282     {}
283
284     IMPL(HScale, GTK_HSCALE);
285
286     HScale::HScale(Adjustment adjustment) : HScale(GTK_HSCALE(gtk_hscale_new(adjustment)))
287     {}
288
289     HScale::HScale(double min, double max, double step) : HScale(GTK_HSCALE(gtk_hscale_new_with_range(min, max, step)))
290     {}
291
292     IMPL(Adjustment, GTK_ADJUSTMENT);
293
294     Adjustment::Adjustment(double value,
295                            double lower, double upper,
296                            double step_increment, double page_increment,
297                            double page_size)
298             : Adjustment(
299             GTK_ADJUSTMENT(gtk_adjustment_new(value, lower, upper, step_increment, page_increment, page_size)))
300     {}
301
302     IMPL(CellRendererText, GTK_CELL_RENDERER_TEXT);
303
304     CellRendererText::CellRendererText() : CellRendererText(GTK_CELL_RENDERER_TEXT(gtk_cell_renderer_text_new()))
305     {}
306
307     IMPL(TreeViewColumn, GTK_TREE_VIEW_COLUMN);
308
309     TreeViewColumn::TreeViewColumn(const char *title, CellRenderer renderer,
310                                    std::initializer_list<TreeViewColumnAttribute> attributes)
311             : TreeViewColumn(gtk_tree_view_column_new_with_attributes(title, renderer, nullptr))
312     {
313         for (auto &it : attributes) {
314             gtk_tree_view_column_add_attribute(this, renderer, it.attribute, it.column);
315         }
316     };
317
318     IMPL(AccelGroup, GTK_ACCEL_GROUP);
319
320     AccelGroup::AccelGroup() : AccelGroup(GTK_ACCEL_GROUP(gtk_accel_group_new()))
321     {}
322
323     IMPL(ListStore, GTK_LIST_STORE);
324
325     void IListStore::clear()
326     {
327         gtk_list_store_clear(this);
328     }
329
330     // IMPL(TreePath, GTK_TREE_PATH);
331
332     TreePath::TreePath() : TreePath(gtk_tree_path_new())
333     {}
334
335     TreePath::TreePath(const char *path) : TreePath(gtk_tree_path_new_from_string(path))
336     {}
337
338     // Custom
339
340     guint IGLArea::on_render(GCallback pFunction, void *data)
341     {
342 #if GTK_TARGET == 3
343         return this.connect("render", pFunction, data);
344 #endif
345 #if GTK_TARGET == 2
346         return this.connect("expose_event", pFunction, data);
347 #endif
348     };
349
350 }