#ifndef INCLUDED_UILIB_H #define INCLUDED_UILIB_H #include #include struct _GdkEventKey; struct _GtkAccelGroup; struct _GtkAdjustment; struct _GtkAlignment; struct _GtkBin; struct _GtkBox; struct _GtkButton; struct _GtkCellEditable; struct _GtkCellRenderer; struct _GtkCellRendererText; struct _GtkCheckButton; struct _GtkCheckMenuItem; struct _GtkComboBox; struct _GtkComboBoxText; struct _GtkContainer; struct _GtkDialog; struct _GtkEditable; struct _GtkEntry; struct _GtkEntryCompletion; struct _GtkFrame; struct _GtkHBox; struct _GtkHPaned; struct _GtkHScale; struct _GtkImage; struct _GtkItem; struct _GtkLabel; struct _GtkListStore; struct _GtkTreeIter; struct _GtkMenu; struct _GtkMenuBar; struct _GtkMenuItem; struct _GtkMenuShell; struct _GtkMisc; struct _GtkObject; struct _GtkPaned; struct _GtkRadioButton; struct _GtkRadioMenuItem; struct _GtkRadioToolButton; struct _GtkRange; struct _GtkScale; struct _GtkScrolledWindow; struct _GtkSpinButton; struct _GtkTable; struct _GtkTearoffMenuItem; struct _GtkTextView; struct _GtkToggleButton; struct _GtkToggleToolButton; struct _GtkToolbar; struct _GtkToolButton; struct _GtkToolItem; struct _GtkTreeModel; struct _GtkTreePath; struct _GtkTreeSelection; struct _GtkTreeStore; struct _GtkTreeView; struct _GtkTreeViewColumn; struct _GtkVBox; struct _GtkVPaned; struct _GtkWidget; struct _GtkWindow; struct _GTypeInstance; #if GTK_TARGET == 3 struct _GtkGLArea; #endif #if GTK_TARGET == 2 using _GtkGLArea = struct _GtkDrawingArea; #endif struct ModalDialog; namespace ui { bool init(int *argc, char **argv[], char const *parameter_string, char const **error); void main(); void process(); enum class window_type { TOP, POPUP }; enum class Shadow { NONE, IN, OUT, ETCHED_IN, ETCHED_OUT }; enum class Policy { ALWAYS, AUTOMATIC, NEVER }; namespace details { enum class Convert { Implicit, Explicit }; template struct Convertible; template struct Convertible { operator T() const { return reinterpret_cast(static_cast(this)->_handle); } }; template struct Convertible { explicit operator T() const { return reinterpret_cast(static_cast(this)->_handle); } }; template struct All : T ... { All() { }; }; template struct Mixin; template struct Mixin { using type = All; }; template struct Mixin { using type = All; }; } const struct Null {} null = {}; const struct New_t {} New = {}; class Object : public details::Convertible, public details::Convertible { public: using self = Object *; using native = _GtkObject *; native _handle; explicit Object(native h) : _handle(h) { } explicit operator bool() const { return _handle != nullptr; } explicit operator void *() const { return _handle; } void unref() { g_object_unref(_handle); } void ref() { g_object_ref(_handle); } template gulong connect(char const *detailed_signal, Lambda &&c_handler, void *data); template gulong connect(char const *detailed_signal, Lambda &&c_handler, Object data); }; static_assert(sizeof(Object) == sizeof(Object::native), "object slicing"); #define WRAP(name, super, T, interfaces, ctors, methods) \ class name; \ class I ## name : public details::Convertible { \ public: \ using self = name *; \ methods \ }; \ class name : public super, public I ## name, public details::Mixin::type { \ public: \ using self = name *; \ using native = T *; \ protected: \ explicit name(native h) noexcept : super(reinterpret_cast(h)) {} \ public: \ explicit name(Null n) noexcept : name((native) nullptr) {} \ explicit name(New_t); \ static name from(native h) { return name(h); } \ static name from(void *ptr) { return name((native) ptr); } \ ctors \ }; \ inline bool operator<(name self, name other) { return self._handle < other._handle; } \ static_assert(sizeof(name) == sizeof(super), "object slicing") // https://developer.gnome.org/gtk2/stable/ch01.html // GInterface WRAP(CellEditable, Object, _GtkCellEditable, (), , ); WRAP(Editable, Object, _GtkEditable, (), , void editable(bool value); ); WRAP(TreeModel, Object, _GtkTreeModel, (), , ); // GObject struct Dimensions { int width; int height; }; class Window; WRAP(Widget, Object, _GtkWidget, (), , Window window(); const char *file_dialog( bool open, const char *title, const char *path = nullptr, const char *pattern = nullptr, bool want_load = false, bool want_import = false, bool want_save = false ); bool visible(); void visible(bool shown); void show(); void hide(); Dimensions dimensions(); void dimensions(int width, int height); void destroy(); ); WRAP(Container, Widget, _GtkContainer, (), , void add(Widget widget); void remove(Widget widget); template void foreach(Lambda &&lambda); ); WRAP(Bin, Container, _GtkBin, (), , ); class AccelGroup; WRAP(Window, Bin, _GtkWindow, (), explicit Window(window_type type); , Window create_dialog_window( const char *title, void func(), void *data, int default_w = -1, int default_h = -1 ); Window create_modal_dialog_window( const char *title, ModalDialog &dialog, int default_w = -1, int default_h = -1 ); Window create_floating_window(const char *title); std::uint64_t on_key_press( bool (*f)(Widget widget, _GdkEventKey *event, void *extra), void *extra = nullptr ); void add_accel_group(AccelGroup group); ); WRAP(Dialog, Window, _GtkDialog, (), , ); WRAP(Alignment, Bin, _GtkAlignment, (), Alignment(float xalign, float yalign, float xscale, float yscale); , ); WRAP(Frame, Bin, _GtkFrame, (), explicit Frame(const char *label = nullptr); , ); WRAP(Button, Bin, _GtkButton, (), explicit Button(const char *label); , ); WRAP(ToggleButton, Button, _GtkToggleButton, (), , bool active() const; void active(bool value); ); WRAP(CheckButton, ToggleButton, _GtkCheckButton, (), explicit CheckButton(const char *label); , ); WRAP(RadioButton, CheckButton, _GtkRadioButton, (), , ); WRAP(Item, Bin, _GtkItem, (), , ); WRAP(MenuItem, Item, _GtkMenuItem, (), explicit MenuItem(const char *label, bool mnemonic = false); , ); WRAP(CheckMenuItem, MenuItem, _GtkCheckMenuItem, (), , ); WRAP(RadioMenuItem, CheckMenuItem, _GtkRadioMenuItem, (), , ); WRAP(TearoffMenuItem, MenuItem, _GtkTearoffMenuItem, (), , ); WRAP(ComboBox, Bin, _GtkComboBox, (), , ); WRAP(ComboBoxText, ComboBox, _GtkComboBoxText, (), , ); WRAP(ToolItem, Bin, _GtkToolItem, (), , ); WRAP(ToolButton, ToolItem, _GtkToolButton, (), , ); WRAP(ToggleToolButton, ToolButton, _GtkToggleToolButton, (), , ); WRAP(RadioToolButton, ToggleToolButton, _GtkRadioToolButton, (), , ); WRAP(ScrolledWindow, Bin, _GtkScrolledWindow, (), , void overflow(Policy x, Policy y); ); WRAP(Box, Container, _GtkBox, (), , void pack_start(ui::Widget child, bool expand, bool fill, unsigned int padding); void pack_end(ui::Widget child, bool expand, bool fill, unsigned int padding); ); WRAP(VBox, Box, _GtkVBox, (), VBox(bool homogenous, int spacing); , ); WRAP(HBox, Box, _GtkHBox, (), HBox(bool homogenous, int spacing); , ); WRAP(Paned, Container, _GtkPaned, (), , ); WRAP(HPaned, Paned, _GtkHPaned, (), , ); WRAP(VPaned, Paned, _GtkVPaned, (), , ); WRAP(MenuShell, Container, _GtkMenuShell, (), , ); WRAP(MenuBar, MenuShell, _GtkMenuBar, (), , ); WRAP(Menu, MenuShell, _GtkMenu, (), , ); struct TableAttach { unsigned int left, right, top, bottom; }; struct TableAttachOptions { // todo: type safety unsigned int x, y; }; struct TablePadding { unsigned int x, y; }; WRAP(Table, Container, _GtkTable, (), Table(std::size_t rows, std::size_t columns, bool homogenous); , // 5 = expand | fill void attach(Widget child, TableAttach attach, TableAttachOptions options = {5, 5}, TablePadding padding = {0, 0}); ); WRAP(TextView, Container, _GtkTextView, (), , void text(char const *str); ); WRAP(Toolbar, Container, _GtkToolbar, (), , ); class TreeModel; WRAP(TreeView, Widget, _GtkTreeView, (), TreeView(TreeModel model); , ); WRAP(Misc, Widget, _GtkMisc, (), , ); WRAP(Label, Widget, _GtkLabel, (), explicit Label(const char *label); , void text(char const *str); ); WRAP(Image, Widget, _GtkImage, (), , ); WRAP(Entry, Widget, _GtkEntry, (IEditable, ICellEditable), explicit Entry(std::size_t max_length); , char const *text(); void text(char const *str); ); class Adjustment; WRAP(SpinButton, Entry, _GtkSpinButton, (), SpinButton(Adjustment adjustment, double climb_rate, std::size_t digits); , ); WRAP(Range, Widget, _GtkRange, (), , ); WRAP(Scale, Range, _GtkScale, (), , ); WRAP(HScale, Scale, _GtkHScale, (), explicit HScale(Adjustment adjustment); HScale(double min, double max, double step); , ); WRAP(Adjustment, Object, _GtkAdjustment, (), Adjustment(double value, double lower, double upper, double step_increment, double page_increment, double page_size); , ); WRAP(CellRenderer, Object, _GtkCellRenderer, (), , ); WRAP(CellRendererText, CellRenderer, _GtkCellRendererText, (), , ); struct TreeViewColumnAttribute { const char *attribute; int column; }; WRAP(TreeViewColumn, Object, _GtkTreeViewColumn, (), TreeViewColumn(const char *title, CellRenderer renderer, std::initializer_list attributes); , ); WRAP(AccelGroup, Object, _GtkAccelGroup, (), , ); WRAP(EntryCompletion, Object, _GtkEntryCompletion, (), , ); WRAP(ListStore, Object, _GtkListStore, (ITreeModel), , void clear(); template void append(T ... args); void append(); ); WRAP(TreeStore, Object, _GtkTreeStore, (ITreeModel), , ); WRAP(TreeSelection, Object, _GtkTreeSelection, (), , ); // GBoxed WRAP(TreePath, Object, _GtkTreePath, (), explicit TreePath(const char *path); , ); // Custom WRAP(GLArea, Widget, _GtkGLArea, (), , guint on_render(GCallback pFunction, void *data); ); #undef WRAP // global enum class alert_response { OK, CANCEL, YES, NO, }; enum class alert_type { OK, OKCANCEL, YESNO, YESNOCANCEL, NOYES, }; enum class alert_icon { Default, Error, Warning, Question, Asterisk, }; extern class Window root; alert_response alert( Window parent, std::string text, std::string title = "WorldSpawn", alert_type type = alert_type::OK, alert_icon icon = alert_icon::Default ); // callbacks namespace { using GtkCallback = void (*)(_GtkWidget *, void *); extern "C" { void gtk_container_foreach(_GtkContainer *, GtkCallback, void *); } } #define this (*static_cast(this)) template gulong Object::connect(char const *detailed_signal, Lambda &&c_handler, void *data) { return g_signal_connect(G_OBJECT(this), detailed_signal, c_handler, data); } template gulong Object::connect(char const *detailed_signal, Lambda &&c_handler, Object data) { return g_signal_connect(G_OBJECT(this), detailed_signal, c_handler, (_GtkObject *) data); } template void IContainer::foreach(Lambda &&lambda) { GtkCallback cb = [](_GtkWidget *widget, void *data) -> void { using Function = typename std::decay::type; Function *f = static_cast(data); (*f)(Widget::from(widget)); }; gtk_container_foreach(this, cb, &lambda); } namespace { extern "C" { void gtk_list_store_insert_with_values(_GtkListStore *, _GtkTreeIter *, gint position, ...); } } template void IListStore::append(T... args) { static_assert(sizeof...(args) % 2 == 0, "received an odd number of arguments"); gtk_list_store_insert_with_values(this, NULL, -1, args ..., -1); } #undef this } #endif