update uwproj
[mizunara.git] / ui / gtk / container.c
1 /*
2  * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER.
3  *
4  * Copyright 2017 Olaf Wintermann. All rights reserved.
5  *
6  * Redistribution and use in source and binary forms, with or without
7  * modification, are permitted provided that the following conditions are met:
8  *
9  *   1. Redistributions of source code must retain the above copyright
10  *      notice, this list of conditions and the following disclaimer.
11  *
12  *   2. Redistributions in binary form must reproduce the above copyright
13  *      notice, this list of conditions and the following disclaimer in the
14  *      documentation and/or other materials provided with the distribution.
15  *
16  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
17  * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
18  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
19  * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
20  * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
21  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
22  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
23  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
24  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
25  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
26  * POSSIBILITY OF SUCH DAMAGE.
27  */
28
29 #include <stdio.h>
30 #include <stdlib.h>
31 #include <limits.h>
32
33 #include "container.h"
34 #include "toolkit.h"
35
36 #include "../common/context.h"
37 #include "../common/object.h"
38
39
40 void ui_container_begin_close(UiObject *obj) {
41     UiContainer *ct = uic_get_current_container(obj);
42     ct->close = 1;
43 }
44
45 int ui_container_finish(UiObject *obj) {
46     UiContainer *ct = uic_get_current_container(obj);
47     if(ct->close) {
48         ui_end(obj);
49         return 0;
50     }
51     return 1;
52 }
53
54 GtkWidget* ui_gtk_vbox_new(int spacing) {
55 #ifdef UI_GTK3
56     return gtk_box_new(GTK_ORIENTATION_VERTICAL, spacing);
57 #else
58     return gtk_vbox_new(FALSE, spacing);
59 #endif
60 }
61
62 GtkWidget* ui_gtk_hbox_new(int spacing) {
63 #ifdef UI_GTK3
64     return gtk_box_new(GTK_ORIENTATION_HORIZONTAL, spacing);
65 #else
66     return gtk_hbox_new(FALSE, spacing);
67 #endif
68 }
69
70 /* -------------------- Frame Container (deprecated) -------------------- */
71 UiContainer* ui_frame_container(UiObject *obj, GtkWidget *frame) {
72     UiContainer *ct = ucx_mempool_calloc(
73             obj->ctx->mempool,
74             1,
75             sizeof(UiContainer));
76     ct->widget = frame;
77     ct->add = ui_frame_container_add;
78     return ct;
79 }
80
81 void ui_frame_container_add(UiContainer *ct, GtkWidget *widget, UiBool fill) {
82     gtk_container_add(GTK_CONTAINER(ct->widget), widget);
83     ui_reset_layout(ct->layout);
84     ct->current = widget;
85 }
86
87
88 /* -------------------- Box Container -------------------- */
89 UiContainer* ui_box_container(UiObject *obj, GtkWidget *box) {
90     UiBoxContainer *ct = ucx_mempool_calloc(
91             obj->ctx->mempool,
92             1,
93             sizeof(UiBoxContainer));
94     ct->container.widget = box;
95     ct->container.add = ui_box_container_add;
96     return (UiContainer*)ct;
97 }
98
99 void ui_box_container_add(UiContainer *ct, GtkWidget *widget, UiBool fill) {
100     UiBoxContainer *bc = (UiBoxContainer*)ct;
101     if(ct->layout.fill != UI_LAYOUT_UNDEFINED) {
102         fill = ui_lb2bool(ct->layout.fill);
103     }
104     
105     if(bc->has_fill && fill) {
106         fprintf(stderr, "UiError: container has 2 filled widgets");
107         fill = FALSE;
108     }
109     if(fill) {
110         bc->has_fill = TRUE;
111     }
112     
113     UiBool expand = fill;
114     gtk_box_pack_start(GTK_BOX(ct->widget), widget, expand, fill, 0);
115     
116     ui_reset_layout(ct->layout);
117     ct->current = widget;
118 }
119
120 UiContainer* ui_grid_container(UiObject *obj, GtkWidget *grid) {
121     UiGridContainer *ct = ucx_mempool_calloc(
122             obj->ctx->mempool,
123             1,
124             sizeof(UiGridContainer));
125     ct->container.widget = grid;
126     ct->container.add = ui_grid_container_add;
127 #ifdef UI_GTK2
128     ct->width = 0;
129     ct->height = 1;
130 #endif
131     return (UiContainer*)ct;
132 }
133
134 #ifdef UI_GTK3
135 void ui_grid_container_add(UiContainer *ct, GtkWidget *widget, UiBool fill) {
136     UiGridContainer *grid = (UiGridContainer*)ct;
137     
138     if(ct->layout.newline) {
139         grid->x = 0;
140         grid->y++;
141         ct->layout.newline = FALSE;
142     }
143     
144     int hexpand = FALSE;
145     int vexpand = FALSE;
146     if(ct->layout.hexpand != UI_LAYOUT_UNDEFINED) {
147         hexpand = ct->layout.hexpand;
148     }
149     if(ct->layout.vexpand != UI_LAYOUT_UNDEFINED) {
150         vexpand = ct->layout.vexpand;
151     }
152     
153     if(hexpand) {
154         gtk_widget_set_hexpand(widget, TRUE);
155     }
156     if(vexpand) {
157         gtk_widget_set_vexpand(widget, TRUE);
158     }
159     
160     int gwidth = ct->layout.gridwidth > 0 ? ct->layout.gridwidth : 1;
161     
162     gtk_grid_attach(GTK_GRID(ct->widget), widget, grid->x, grid->y, gwidth, 1);
163     grid->x += gwidth;
164     
165     ui_reset_layout(ct->layout);
166     ct->current = widget;
167 }
168 #endif
169 #ifdef UI_GTK2
170 void ui_grid_container_add(UiContainer *ct, GtkWidget *widget, UiBool fill) {
171     UiGridContainer *grid = (UiGridContainer*)ct;
172     
173     if(ct->layout.newline) {
174         grid->x = 0;
175         grid->y++;
176         ct->layout.newline = FALSE;
177     }
178     
179     int hexpand = FALSE;
180     int vexpand = FALSE;
181     if(ct->layout.hexpand != UI_LAYOUT_UNDEFINED) {
182         hexpand = ct->layout.hexpand;
183     }
184     if(ct->layout.vexpand != UI_LAYOUT_UNDEFINED) {
185         vexpand = ct->layout.vexpand;
186     }
187     GtkAttachOptions xoptions = hexpand ? GTK_FILL | GTK_EXPAND : GTK_FILL;
188     GtkAttachOptions yoptions = vexpand ? GTK_FILL | GTK_EXPAND : GTK_FILL;
189     
190     gtk_table_attach(GTK_TABLE(ct->widget), widget, grid->x, grid->x+1, grid->y, grid->y+1, xoptions, yoptions, 0, 0);
191     grid->x++;
192     int nw = grid->x > grid->width ? grid->x : grid->width;
193     if(grid->x > grid->width || grid->y > grid->height) {
194         grid->width = nw;
195         grid->height = grid->y + 1;
196         gtk_table_resize(GTK_TABLE(ct->widget), grid->width, grid->height);
197     }
198     
199     ui_reset_layout(ct->layout);
200     ct->current = widget;
201 }
202 #endif
203
204 UiContainer* ui_scrolledwindow_container(UiObject *obj, GtkWidget *scrolledwindow) {
205     UiContainer *ct = ucx_mempool_calloc(
206             obj->ctx->mempool,
207             1,
208             sizeof(UiContainer));
209     ct->widget = scrolledwindow;
210     ct->add = ui_scrolledwindow_container_add;
211     return ct;
212 }
213
214 void ui_scrolledwindow_container_add(UiContainer *ct, GtkWidget *widget, UiBool fill) {
215     // TODO: check if the widget implements GtkScrollable
216 #ifdef UI_GTK3
217     gtk_container_add(GTK_CONTAINER(ct->widget), widget);
218 #else
219     gtk_scrolled_window_add_with_viewport(GTK_SCROLLED_WINDOW(ct->widget), widget);
220 #endif
221     ui_reset_layout(ct->layout);
222     ct->current = widget;
223 }
224
225 UiContainer* ui_tabview_container(UiObject *obj, GtkWidget *tabview) {
226     UiTabViewContainer *ct = ucx_mempool_calloc(
227             obj->ctx->mempool,
228             1,
229             sizeof(UiTabViewContainer));
230     ct->container.widget = tabview;
231     ct->container.add = ui_tabview_container_add;
232     return (UiContainer*)ct;
233 }
234
235 void ui_tabview_container_add(UiContainer *ct, GtkWidget *widget, UiBool fill) {
236     gtk_notebook_append_page(
237             GTK_NOTEBOOK(ct->widget),
238             widget,
239             gtk_label_new(ct->layout.label));
240     
241     ui_reset_layout(ct->layout);
242     ct->current = widget;
243 }
244
245
246 UIWIDGET ui_vbox(UiObject *obj) {
247     return ui_vbox_sp(obj, 0, 0);
248 }
249
250 UIWIDGET ui_hbox(UiObject *obj) {
251     return ui_hbox_sp(obj, 0, 0);
252 }
253
254 static GtkWidget* box_set_margin(GtkWidget *box, int margin) {
255     GtkWidget *ret = box;
256 #ifdef UI_GTK3
257 #if GTK_MAJOR_VERSION == 3 && GTK_MINOR_VERSION >= 12
258     gtk_widget_set_margin_start(box, margin);
259     gtk_widget_set_margin_end(box, margin);
260 #else
261     gtk_widget_set_margin_left(box, margin);
262     gtk_widget_set_margin_right(box, margin);
263 #endif
264     gtk_widget_set_margin_top(box, margin);
265     gtk_widget_set_margin_bottom(box, margin);
266 #elif defined(UI_GTK2)
267     GtkWidget *a = gtk_alignment_new(0.5, 0.5, 1, 1);
268     gtk_alignment_set_padding(GTK_ALIGNMENT(a), margin, margin, margin, margin);
269     gtk_container_add(GTK_CONTAINER(a), box);
270     ret = a;
271 #endif
272     return ret;
273 }
274
275 UIWIDGET ui_vbox_sp(UiObject *obj, int margin, int spacing) {
276     UiContainer *ct = uic_get_current_container(obj);
277     
278     GtkWidget *vbox = ui_gtk_vbox_new(spacing);   
279     GtkWidget *widget = margin > 0 ? box_set_margin(vbox, margin) : vbox;
280     ct->add(ct, widget, TRUE);
281     
282     UiObject *newobj = uic_object_new(obj, vbox);
283     newobj->container = ui_box_container(obj, vbox);
284     uic_obj_add(obj, newobj);
285     
286     return widget;
287 }
288
289 UIWIDGET ui_hbox_sp(UiObject *obj, int margin, int spacing) {
290     UiContainer *ct = uic_get_current_container(obj);
291     
292     GtkWidget *hbox = ui_gtk_hbox_new(spacing);
293     GtkWidget *widget = margin > 0 ? box_set_margin(hbox, margin) : hbox;
294     ct->add(ct, widget, TRUE);
295     
296     UiObject *newobj = uic_object_new(obj, hbox);
297     newobj->container = ui_box_container(obj, hbox);
298     uic_obj_add(obj, newobj);
299     
300     return widget;
301 }
302
303 UIWIDGET ui_grid(UiObject *obj) {
304     return ui_grid_sp(obj, 0, 0, 0);
305 }
306
307 UIWIDGET ui_grid_sp(UiObject *obj, int margin, int columnspacing, int rowspacing) {
308     UiContainer *ct = uic_get_current_container(obj);
309     GtkWidget *widget;
310     
311 #ifdef UI_GTK3
312     GtkWidget *grid = gtk_grid_new();
313     gtk_grid_set_column_spacing(GTK_GRID(grid), columnspacing);
314     gtk_grid_set_row_spacing(GTK_GRID(grid), rowspacing);
315 #if GTK_MAJOR_VERSION == 3 && GTK_MINOR_VERSION >= 12
316     gtk_widget_set_margin_start(grid, margin);
317     gtk_widget_set_margin_end(grid, margin);
318 #else
319     gtk_widget_set_margin_left(grid, margin);
320     gtk_widget_set_margin_right(grid, margin);
321 #endif
322     gtk_widget_set_margin_top(grid, margin);
323     gtk_widget_set_margin_bottom(grid, margin);
324     
325     widget = grid;
326 #elif defined(UI_GTK2)
327     GtkWidget *grid = gtk_table_new(1, 1, FALSE);
328     
329     gtk_table_set_col_spacings(GTK_TABLE(grid), columnspacing);
330     gtk_table_set_row_spacings(GTK_TABLE(grid), rowspacing);
331     
332     if(margin > 0) {
333         GtkWidget *a = gtk_alignment_new(0.5, 0.5, 1, 1);
334         gtk_alignment_set_padding(GTK_ALIGNMENT(a), margin, margin, margin, margin);
335         gtk_container_add(GTK_CONTAINER(a), grid);
336         widget = a;
337     } else {
338         widget = grid;
339     }
340 #endif
341     ct->add(ct, widget, TRUE);
342     
343     UiObject *newobj = uic_object_new(obj, grid);
344     newobj->container = ui_grid_container(obj, grid);
345     uic_obj_add(obj, newobj);
346     
347     return widget;
348 }
349
350 UIWIDGET ui_scrolledwindow(UiObject *obj) {
351     UiContainer *ct = uic_get_current_container(obj);
352     GtkWidget *sw = gtk_scrolled_window_new(NULL, NULL);
353     ct->add(ct, sw, TRUE);
354     
355     UiObject *newobj = uic_object_new(obj, sw);
356     newobj->container = ui_scrolledwindow_container(obj, sw);
357     uic_obj_add(obj, newobj);
358     
359     return sw;
360 }
361
362 UIWIDGET ui_tabview(UiObject *obj) {
363     GtkWidget *tabview = gtk_notebook_new();
364     gtk_notebook_set_show_border(GTK_NOTEBOOK(tabview), FALSE);
365     gtk_notebook_set_show_tabs(GTK_NOTEBOOK(tabview), FALSE);
366     
367     UiContainer *ct = uic_get_current_container(obj);
368     ct->add(ct, tabview, TRUE);
369     
370     UiObject *tabviewobj = uic_object_new(obj, tabview);
371     tabviewobj->container = ui_tabview_container(obj, tabview);
372     uic_obj_add(obj, tabviewobj);
373     
374     return tabview;
375 }
376
377 void ui_tab(UiObject *obj, char *title) {
378     UiContainer *ct = uic_get_current_container(obj);
379     ct->layout.label = title;
380     ui_vbox(obj);
381 }
382
383 void ui_select_tab(UIWIDGET tabview, int tab) {
384     gtk_notebook_set_current_page(GTK_NOTEBOOK(tabview), tab);
385 }
386
387 /* -------------------- Splitpane -------------------- */
388
389 static GtkWidget* create_paned(UiOrientation orientation) {
390 #ifdef UI_GTK3
391     switch(orientation) {
392         case UI_HORIZONTAL: return gtk_paned_new(GTK_ORIENTATION_HORIZONTAL);
393         case UI_VERTICAL: return gtk_paned_new(GTK_ORIENTATION_VERTICAL);
394     }
395 #else
396     switch(orientation) {
397         case UI_HORIZONTAL: return gtk_hpaned_new();
398         case UI_VERTICAL: return gtk_vpaned_new();
399     }
400 #endif
401     return NULL;
402 }
403
404 UIWIDGET ui_splitpane(UiObject *obj, int max, UiOrientation orientation) {
405     GtkWidget *paned = create_paned(orientation);
406     UiContainer *ct = uic_get_current_container(obj);
407     ct->add(ct, paned, TRUE);
408     
409     if(max <= 0) max = INT_MAX;
410     
411     UiPanedContainer *pctn = ucx_mempool_calloc(
412             obj->ctx->mempool,
413             1,
414             sizeof(UiPanedContainer));
415     pctn->container.widget = paned;
416     pctn->container.add = ui_paned_container_add;
417     pctn->current_pane = paned;
418     pctn->orientation = orientation;
419     pctn->max = max;
420     pctn->cur = 0;
421     
422     UiObject *pobj = uic_object_new(obj, paned);
423     pobj->container = (UiContainer*)pctn;
424     
425     uic_obj_add(obj, pobj);
426     
427     return paned;
428 }
429
430 UIWIDGET ui_hsplitpane(UiObject *obj, int max) {
431     return ui_splitpane(obj, max, UI_HORIZONTAL);
432 }
433
434 UIWIDGET ui_vsplitpane(UiObject *obj, int max) {
435     return ui_splitpane(obj, max, UI_VERTICAL);
436 }
437
438 void ui_paned_container_add(UiContainer *ct, GtkWidget *widget, UiBool fill) {
439     UiPanedContainer *pctn = (UiPanedContainer*)ct;
440     
441     gboolean resize = (ct->layout.hexpand || ct->layout.vexpand) ? TRUE : FALSE;
442     int width = ct->layout.width;
443     ui_reset_layout(ct->layout);
444     
445     if(pctn->cur == 0) {
446         gtk_paned_pack1(GTK_PANED(pctn->current_pane), widget, resize, resize);
447     } else if(pctn->cur < pctn->max-1) {
448         GtkWidget *nextPane = create_paned(pctn->orientation);
449         gtk_paned_pack2(GTK_PANED(pctn->current_pane), nextPane, TRUE, TRUE);
450         gtk_paned_pack1(GTK_PANED(nextPane), widget, resize, resize);
451         pctn->current_pane = nextPane;
452     } else if(pctn->cur == pctn->max-1) {
453         gtk_paned_pack2(GTK_PANED(pctn->current_pane), widget, resize, resize);
454         width = 0; // disable potential call of gtk_paned_set_position
455     } else {
456         fprintf(stderr, "Splitpane max reached: %d\n", pctn->max);
457         return;
458     }
459     
460     if(width > 0) {
461         gtk_paned_set_position(GTK_PANED(pctn->current_pane), width);
462     }
463     
464     pctn->cur++;
465 }
466
467
468 /* -------------------- Sidebar (deprecated) -------------------- */
469 UIWIDGET ui_sidebar(UiObject *obj) {
470 #ifdef UI_GTK3
471     GtkWidget *paned = gtk_paned_new(GTK_ORIENTATION_HORIZONTAL);
472 #else
473     GtkWidget *paned = gtk_hpaned_new();
474 #endif
475     gtk_paned_set_position(GTK_PANED(paned), 200);
476     
477     GtkWidget *sidebar = ui_gtk_vbox_new(0);
478     gtk_paned_pack1(GTK_PANED(paned), sidebar, TRUE, FALSE);
479     
480     UiObject *left = uic_object_new(obj, sidebar);
481     UiContainer *ct1 = ui_box_container(obj, sidebar);
482     left->container = ct1;
483     
484     UiObject *right = uic_object_new(obj, sidebar);
485     UiContainer *ct2 = ucx_mempool_malloc(
486             obj->ctx->mempool,
487             sizeof(UiContainer));
488     ct2->widget = paned;
489     ct2->add = ui_split_container_add2;
490     right->container = ct2;
491     
492     UiContainer *ct = uic_get_current_container(obj);
493     ct->add(ct, paned, TRUE);
494     
495     uic_obj_add(obj, right);
496     uic_obj_add(obj, left);
497     
498     return sidebar;
499 }
500
501 void ui_split_container_add1(UiContainer *ct, GtkWidget *widget, UiBool fill) {
502     // TODO: remove
503     gtk_paned_pack1(GTK_PANED(ct->widget), widget, TRUE, FALSE);
504     
505     ui_reset_layout(ct->layout);
506     ct->current = widget;
507 }
508
509 void ui_split_container_add2(UiContainer *ct, GtkWidget *widget, UiBool fill) {
510     gtk_paned_pack2(GTK_PANED(ct->widget), widget, TRUE, FALSE);
511     
512     ui_reset_layout(ct->layout);
513     ct->current = widget;
514 }
515
516
517 /* -------------------- Document Tabview -------------------- */
518 static void page_change(GtkNotebook *notebook, GtkWidget *page, guint page_num, gpointer data) {
519     GQuark q = g_quark_from_static_string("ui.tab.object");
520     UiObject *tab = g_object_get_qdata(G_OBJECT(page), q);
521     if(!tab) {
522         return;
523     }
524     
525     //printf("page_change: %d\n", page_num);
526     UiContext *ctx = tab->ctx;
527     uic_context_detach_all(ctx->parent); // TODO: fix?
528     ctx->parent->attach_document(ctx->parent, ctx->document);
529 }
530
531 UiTabbedPane* ui_tabbed_document_view(UiObject *obj) {
532     GtkWidget *tabview = gtk_notebook_new();
533     gtk_notebook_set_show_border(GTK_NOTEBOOK(tabview), FALSE);
534     
535     g_signal_connect(
536                 tabview,
537                 "switch-page",
538                 G_CALLBACK(page_change),
539                 NULL);
540     
541     UiContainer *ct = uic_get_current_container(obj);
542     ct->add(ct, tabview, TRUE);
543     
544     UiTabbedPane *tabbedpane = ui_malloc(obj->ctx, sizeof(UiTabbedPane));
545     tabbedpane->ctx = uic_current_obj(obj)->ctx;
546     tabbedpane->widget = tabview;
547     tabbedpane->document = NULL;
548     
549     return tabbedpane;
550 }
551
552 UiObject* ui_document_tab(UiTabbedPane *view) {
553     GtkWidget *frame = gtk_frame_new(NULL);
554     gtk_frame_set_shadow_type(GTK_FRAME(frame), GTK_SHADOW_NONE);
555     // TODO: label
556     gtk_notebook_append_page(GTK_NOTEBOOK(view->widget), frame, NULL);
557     
558     UiObject *tab = ui_malloc(view->ctx, sizeof(UiObject));
559     tab->widget = NULL; // initialization for uic_context()
560     tab->ctx = uic_context(tab, view->ctx->mempool);
561     tab->ctx->parent = view->ctx;
562     tab->ctx->attach_document = uic_context_attach_document;
563     tab->ctx->detach_document2 = uic_context_detach_document2;
564     tab->widget = frame;
565     tab->window = view->ctx->obj->window;
566     tab->container = ui_frame_container(tab, frame);
567     tab->next = NULL;
568     
569     GQuark q = g_quark_from_static_string("ui.tab.object");
570     g_object_set_qdata(G_OBJECT(frame), q, tab);
571     
572     return tab;
573 }
574
575 void ui_tab_set_document(UiContext *ctx, void *document) {
576     // TODO: remove?
577     if(ctx->parent->document) {
578         //ctx->parent->detach_document(ctx->parent, ctx->parent->document);
579     }
580     //uic_context_set_document(ctx, document);
581     //uic_context_set_document(ctx->parent, document);
582     //ctx->parent->document = document;
583 }
584
585 void ui_tab_detach_document(UiContext *ctx) {
586     // TODO: remove?
587     //uic_context_detach_document(ctx->parent);
588 }
589
590
591 /*
592  * -------------------- Layout Functions --------------------
593  * 
594  * functions for setting layout attributes for the current container
595  *
596  */
597
598 void ui_layout_fill(UiObject *obj, UiBool fill) {
599     UiContainer *ct = uic_get_current_container(obj);
600     ct->layout.fill = ui_bool2lb(fill);
601 }
602
603 void ui_layout_hexpand(UiObject *obj, UiBool expand) {
604     UiContainer *ct = uic_get_current_container(obj);
605     ct->layout.hexpand = expand;
606 }
607
608 void ui_layout_vexpand(UiObject *obj, UiBool expand) {
609     UiContainer *ct = uic_get_current_container(obj);
610     ct->layout.vexpand = expand;
611 }
612
613 void ui_layout_width(UiObject *obj, int width) {
614     UiContainer *ct = uic_get_current_container(obj);
615     ct->layout.width = width;
616 }
617
618 void ui_layout_gridwidth(UiObject *obj, int width) {
619     UiContainer *ct = uic_get_current_container(obj);
620     ct->layout.gridwidth = width;
621 }
622
623 void ui_newline(UiObject *obj) {
624     UiContainer *ct = uic_get_current_container(obj);
625     ct->layout.newline = TRUE;
626 }
627