Add xpmimg widget to makegdesc.
[doldaconnect.git] / common / makegdesc
1 #!/usr/bin/perl
2
3 use Getopt::Long;
4 use IO::File;
5
6 GetOptions(\%args, "h=s") || exit 1;
7
8 $tempvar = 0;
9
10 sub printwidgets
11 {
12     my($widget, $sl, $p, $sig, $cb, $data, $pf, $cpf, $mod, $key, @delayedlines);
13     $sl = $_[1];
14     $p = "    " . (" " x $sl);
15     $cpf = $_[2];
16     @delayedlines = ();
17     foreach $widget (@{$_[0]})
18     {
19         if($widget->{"type"} eq "wnd")
20         {
21             print "${p}stack[$sl] = gtk_window_new(GTK_WINDOW_TOPLEVEL);\n";
22             if($options{"hasaccels"}) {
23                 print "${p}gtk_window_add_accel_group(GTK_WINDOW(stack[$sl]), accel_group);\n";
24             }
25             if($widget->{"title"}) {
26                 print "${p}gtk_window_set_title(GTK_WINDOW(stack[$sl]), \"" . $widget->{"title"} . "\");\n";
27             }
28             $pf = sub
29             {
30                 my($widget, $p, $sl) = @_;
31                 print "${p}gtk_container_add(GTK_CONTAINER(stack[" . ($sl - 1) . "]), stack[$sl]);\n";
32             }
33         } elsif($widget->{"type"} eq "assistant") {
34             print "${p}stack[$sl] = gtk_assistant_new();\n";
35             if(defined($widget->{"title"})) {
36                 print "${p}gtk_window_set_title(GTK_WINDOW(stack[$sl]), \"" . $widget->{"title"} . "\");\n";
37             }
38             $pf = sub
39             {
40                 my($widget, $p, $sl) = @_;
41                 print "${p}gtk_assistant_append_page(GTK_ASSISTANT(stack[" . ($sl - 1) . "]), stack[$sl]);\n";
42                 if(defined($widget->{"title"})) {
43                     print "${p}gtk_assistant_set_page_title(GTK_ASSISTANT(stack[" . ($sl - 1) . "]), stack[" . $sl . "], _(\"" . $widget->{"title"} . "\"));\n";
44                 }
45                 if(defined($widget->{"ptype"})) {
46                     print "${p}gtk_assistant_set_page_type(GTK_ASSISTANT(stack[" . ($sl - 1) . "]), stack[" . $sl . "], GTK_ASSISTANT_PAGE_" . $widget->{"ptype"} . ");\n";
47                 }
48                 if(defined($widget->{"cmpl"})) {
49                     print "${p}gtk_assistant_set_page_complete(GTK_ASSISTANT(stack[" . ($sl - 1) . "]), stack[" . $sl . "], " . $widget->{"cmpl"} . ");\n";
50                 }
51             }
52         } elsif($widget->{"type"} =~ /[hv]box/) {
53             print "${p}stack[$sl] = gtk_" . $widget->{"type"} . "_new(";
54             print $widget->{"homo"}?"TRUE, ":"FALSE, ";
55             print $widget->{"spacing"} || "0";
56             print ");\n";
57             $pf = sub
58             {
59                 my($widget, $p, $sl) = @_;
60                 print "${p}gtk_box_pack_start(GTK_BOX(stack[" . ($sl - 1) . "]), stack[$sl], ";
61                 print (($widget->{"expand"} || $widget->{"parent"}->{"dexpand"})?"TRUE, ":"FALSE, ");
62                 print (($widget->{"fill"} || $widget->{"parent"}->{"dfill"})?"TRUE, ":"FALSE, ");
63                 print $widget->{"pad"} || "0";
64                 print ");\n";
65             }
66         } elsif($widget->{"type"} eq "table") {
67             print "${p}stack[$sl] = gtk_table_new(" . $widget->{"rows"} . ", " . $widget->{"cols"};
68             print ", " . (($widget->{"homo"} eq "TRUE")?"TRUE":"FALSE");
69             print ");\n";
70             $pf = sub
71             {
72                 my($widget, $p, $sl) = @_;
73                 print "${p}gtk_table_attach(GTK_TABLE(stack[" . ($sl - 1) . "]), stack[$sl]";
74                 print ", " . $widget->{"tx"};
75                 print ", " . ($widget->{"tx"} + (defined($widget->{"tw"})?$widget->{"tw"}:1));
76                 print ", " . $widget->{"ty"};
77                 print ", " . ($widget->{"ty"} + (defined($widget->{"th"})?$widget->{"th"}:1));
78                 if($widget->{"fill"} eq "y") {
79                     $widget->{"fillx"} = "y";
80                     $widget->{"filly"} = "y";
81                 }
82                 if($widget->{"shrink"} eq "y") {
83                     $widget->{"shrinkx"} = "y";
84                     $widget->{"shrinky"} = "y";
85                 }
86                 if($widget->{"expand"} eq "y") {
87                     $widget->{"expandx"} = "y";
88                     $widget->{"expandy"} = "y";
89                 }
90                 print ", 0";
91                 print " | GTK_FILL" if $widget->{"fillx"} eq "y";
92                 print " | GTK_SHRINK" if $widget->{"shrinkx"} eq "y";
93                 print " | GTK_EXPAND" if $widget->{"expandx"} eq "y";
94                 print ", 0";
95                 print " | GTK_FILL" if $widget->{"filly"} eq "y";
96                 print " | GTK_SHRINK" if $widget->{"shrinky"} eq "y";
97                 print " | GTK_EXPAND" if $widget->{"expandy"} eq "y";
98                 print ", " . (defined($widget->{"padx"})?$widget->{"padx"}:"0");
99                 print ", " . (defined($widget->{"pady"})?$widget->{"pady"}:"0");
100                 print ");\n";
101             }
102         } elsif($widget->{"type"} eq "btn") {
103             $widget->{"label"} || die("Can't have button without label\n");
104             print "${p}stack[$sl] = gtk_button_new_with_mnemonic(_(\"" . $widget->{"label"} . "\"));\n";
105         } elsif($widget->{"type"} eq "chk") {
106             $widget->{"label"} || die("Can't have check button without label\n");
107             print "${p}stack[$sl] = gtk_check_button_new_with_mnemonic(_(\"" . $widget->{"label"} . "\"));\n";
108         } elsif($widget->{"type"} eq "radio") {
109             $widget->{"label"} || die("Can't have check button without label\n");
110             if(defined($widget->{"group"})) {
111                 print "${p}stack[$sl] = gtk_radio_button_new_with_mnemonic_from_widget(GTK_RADIO_BUTTON(" . $options{"prefix"} . $widget->{"group"} . "), _(\"" . $widget->{"label"} . "\"));\n";
112             } else {
113                 print "${p}stack[$sl] = gtk_radio_button_new_with_mnemonic(NULL, _(\"" . $widget->{"label"} . "\"));\n";
114             }
115         } elsif($widget->{"type"} eq "sbtn") {
116             $widget->{"stock"} || die("Can't have button without stock\n");
117             print "${p}stack[$sl] = gtk_button_new_from_stock(GTK_STOCK_" . $widget->{"stock"} . ");\n";
118         } elsif($widget->{"type"} eq "simg") {
119             $widget->{"stock"} || die("Can't have image without stock\n");
120             $widget->{"size"} || die("Can't have image without size\n");
121             print "${p}stack[$sl] = gtk_image_new_from_stock(GTK_STOCK_" . $widget->{"stock"} . ", GTK_ICON_SIZE_" . $widget->{"size"} . ");\n";
122         } elsif($widget->{"type"} eq "xpmimg") {
123             $widget->{"data"} || die("Can't have XPM image without data\n");
124             print "${p}stack[$sl] = gtk_image_new_from_pixbuf(gdk_pixbuf_new_from_xpm_data((const char **)" . $widget->{"data"} . "));\n";
125         } elsif($widget->{"type"} eq "lbl") {
126             $widget->{"label"} || $widget->{"markup"} || die("Can't have label without label\n");
127             if($widget->{"label"}) {
128                 print "${p}stack[$sl] = gtk_label_new(_(\"" . $widget->{"label"} . "\"));\n";
129             } else {
130                 print "${p}stack[$sl] = gtk_label_new(NULL);\n";
131                 print "${p}gtk_label_set_markup(GTK_LABEL(stack[$sl]), _(\"" . $widget->{"markup"} . "\"));\n";
132             }
133             if($widget->{"wrap"} eq "y") {
134                 print "${p}gtk_label_set_line_wrap(GTK_LABEL(stack[$sl]), TRUE);\n";
135             }
136         } elsif($widget->{"type"} eq "mlbl") {
137             $widget->{"label"} || die("Can't have label without label\n");
138             print "${p}stack[$sl] = gtk_label_new_with_mnemonic(_(\"" . $widget->{"label"} . "\"));\n";
139             if(defined($widget->{"mwidget"}))
140             {
141                 if($widget->{"var"} ne "y") {
142                     $widget->{"var"} = "l";
143                 }
144                 if(!defined($widget->{"name"})) {
145                     $widget->{"name"} = "temp" . $tempvar++;
146                 }
147                 $str = "gtk_label_set_mnemonic_widget(GTK_LABEL(";
148                 if($widget->{"var"} eq "y") {
149                     $str .= $options{"prefix"};
150                 }
151                 $str .= $widget->{"name"};
152                 $str .= "), " . $options{"prefix"} . $widget->{"mwidget"} . ");";
153                 push @delayedlines, ($str);
154             }
155         } elsif($widget->{"type"} eq "text") {
156             print "${p}stack[$sl] = gtk_entry_new();\n";
157             if($widget->{"default"}) {
158                 print "${p}gtk_entry_set_text(GTK_ENTRY(stack[$sl]), \"" . $widget->{"default"} . "\");\n";
159             }
160         } elsif($widget->{"type"} eq "menubar") {
161             print "${p}stack[$sl] = gtk_menu_bar_new();\n";
162             $pf = sub
163             {
164                 my($widget, $p, $sl) = @_;
165                 print "${p}gtk_menu_shell_append(GTK_MENU_SHELL(stack[" . ($sl - 1) . "]), stack[$sl]);\n";
166             }
167         } elsif($widget->{"type"} eq "menuitem") {
168             print "${p}stack[$sl] = gtk_menu_item_new_with_mnemonic(_(\"" . $widget->{"label"} . "\"));\n";
169             $pf = sub
170             {
171                 my($widget, $p, $sl) = @_;
172                 print "${p}gtk_menu_item_set_submenu(GTK_MENU_ITEM(stack[" . ($sl - 1) . "]), stack[$sl]);\n";
173             }
174         } elsif($widget->{"type"} eq "smenuitem") {
175             print "${p}stack[$sl] = gtk_image_menu_item_new_from_stock(GTK_STOCK_" . $widget->{"stock"} . ", accel_group);\n";
176             $pf = sub
177             {
178                 my($widget, $p, $sl) = @_;
179                 print "${p}gtk_menu_item_set_submenu(GTK_MENU_ITEM(stack[" . ($sl - 1) . "]), stack[$sl]);\n";
180             }
181         } elsif($widget->{"type"} eq "menusep") {
182             print "${p}stack[$sl] = gtk_separator_menu_item_new();\n";
183         } elsif($widget->{"type"} eq "menu") {
184             print "${p}stack[$sl] = gtk_menu_new();\n";
185             if($options{"hasaccels"}) {
186                 print "${p}gtk_menu_set_accel_group(GTK_MENU(stack[$sl]), accel_group);\n";
187             }
188             $pf = sub
189             {
190                 my($widget, $p, $sl) = @_;
191                 print "${p}gtk_menu_shell_append(GTK_MENU_SHELL(stack[" . ($sl - 1) . "]), stack[$sl]);\n";
192             };
193             $widget->{"noshow"} = 1;
194         } elsif($widget->{"type"} =~ /^[hv]paned$/) {
195             print "${p}stack[$sl] = gtk_" . $widget->{"type"} . "_new();\n";
196             $widget->{"cur"} = 1;
197             $pf = sub
198             {
199                 my($widget, $p, $sl) = @_;
200                 print "${p}gtk_paned_pack" . ($widget->{"parent"}->{"cur"}) . "(GTK_PANED(stack[" . ($sl - 1) . "]), stack[$sl]";
201                 print ", " . ((index($widget->{"parent"}->{"resize"}, $widget->{"parent"}->{"cur"}) < 0)?"FALSE":"TRUE");
202                 print ", " . ((index($widget->{"parent"}->{"shrink"}, $widget->{"parent"}->{"cur"}) < 0)?"FALSE":"TRUE");
203                 print ");\n";
204                 $widget->{"parent"}->{"cur"}++;
205             }
206         } elsif($widget->{"type"} eq "notebook") {
207             print "${p}stack[$sl] = gtk_notebook_new();\n";
208             if(defined($widget->{"pos"})) {
209                 print "${p}gtk_notebook_set_tab_pos(GTK_NOTEBOOK(stack[$sl]), GTK_POS_" . $widget->{"pos"} . ");\n";
210             }
211             $pf = sub
212             {
213                 my($widget, $p, $sl) = @_;
214                 print "${p}gtk_notebook_append_page(GTK_NOTEBOOK(stack[" . ($sl - 1) . "]), stack[$sl]";
215                 print ", gtk_label_new_with_mnemonic(_(\"" . $widget->{"nblabel"} . "\"))";
216                 print ");\n";
217             }
218         } elsif($widget->{"type"} eq "sw") {
219             print "${p}stack[$sl] = gtk_scrolled_window_new(NULL, NULL);\n";
220             $pf = sub
221             {
222                 my($widget, $p, $sl) = @_;
223                 print "${p}gtk_container_add(GTK_CONTAINER(stack[" . ($sl - 1) . "]), stack[$sl]);\n";
224             }
225         } elsif($widget->{"type"} eq "frame") {
226             print "${p}stack[$sl] = gtk_frame_new(_(\"" . $widget->{"label"} . "\"));\n";
227             $pf = sub
228             {
229                 my($widget, $p, $sl) = @_;
230                 print "${p}gtk_container_add(GTK_CONTAINER(stack[" . ($sl - 1) . "]), stack[$sl]);\n";
231             }
232         } elsif($widget->{"type"} eq "exp") {
233             print "${p}stack[$sl] = gtk_expander_new_with_mnemonic(_(\"" . $widget->{"label"} . "\"));\n";
234             $pf = sub
235             {
236                 my($widget, $p, $sl) = @_;
237                 print "${p}gtk_container_add(GTK_CONTAINER(stack[" . ($sl - 1) . "]), stack[$sl]);\n";
238             }
239         } elsif($widget->{"type"} eq "align") {
240             print "${p}stack[$sl] = gtk_alignment_new(";
241             if(defined($widget->{"xa"})) {
242                 print $widget->{"xa"};
243             } else {
244                 print "0.5";
245             }
246             print ", ";
247             if(defined($widget->{"ya"})) {
248                 print $widget->{"ya"};
249             } else {
250                 print "0.5";
251             }
252             print ", ";
253             if(defined($widget->{"xs"})) {
254                 print $widget->{"xs"};
255             } else {
256                 print "1.0";
257             }
258             print ", ";
259             if(defined($widget->{"ys"})) {
260                 print $widget->{"ys"};
261             } else {
262                 print "1.0";
263             }
264             print ");\n";
265             $pf = sub
266             {
267                 my($widget, $p, $sl) = @_;
268                 print "${p}gtk_container_add(GTK_CONTAINER(stack[" . ($sl - 1) . "]), stack[$sl]);\n";
269             }
270         } elsif($widget->{"type"} eq "treeview") {
271             print "${p}stack[$sl] = gtk_tree_view_new();\n";
272             if(defined($widget->{"hvis"})) {
273                 print "${p}gtk_tree_view_set_headers_visible(GTK_TREE_VIEW(stack[$sl]), " . $widget->{"hvis"} . ");\n";
274             }
275             if(defined($widget->{"rules"})) {
276                 print "${p}gtk_tree_view_set_rules_hint(GTK_TREE_VIEW(stack[$sl]), " . $widget->{"rules"} . ");\n";
277             }
278             if(defined($widget->{"searchcol"})) {
279                 print "${p}gtk_tree_view_set_search_column(GTK_TREE_VIEW(stack[$sl]), " . $widget->{"searchcol"} . ");\n";
280                 print "${p}gtk_tree_view_set_enable_search(GTK_TREE_VIEW(stack[$sl]), TRUE);\n";
281             }
282             $pf = sub
283             {
284                 my($widget, $p, $sl) = @_;
285                 print "${p}gtk_tree_view_append_column(GTK_TREE_VIEW(stack[" . ($sl - 1) . "]), column);\n";
286                 if($widget->{"expander"} eq "y") {
287                     print "${p}gtk_tree_view_set_expander_column(GTK_TREE_VIEW(stack[" . ($sl - 1) . "]), column);\n";
288                 }
289             }
290         } elsif($widget->{"type"} eq "tvcol") {
291             if(!defined($widget->{"subwidgets"}))
292             {
293                 print "${p}column = gtk_tree_view_column_new_with_attributes(";
294                 print "_(\"" . $widget->{"title"} . "\")";
295                 print ", gtk_cell_renderer_text_new()";
296                 if(defined($widget->{"text"})) {
297                     print ", \"text\", " . $widget->{"text"};
298                 }
299                 print ", NULL);\n";
300             } else {
301                 print "${p}column = gtk_tree_view_column_new();\n";
302                 print "${p}gtk_tree_view_column_set_title(column, _(\"" . $widget->{"title"} . "\"));\n";
303             }
304             if(defined($widget->{"sortcol"})) {
305                 print "${p}gtk_tree_view_column_set_sort_column_id(column, " . $widget->{"sortcol"} . ");\n";
306             }
307             if(defined($widget->{"resizable"})) {
308                 print "${p}gtk_tree_view_column_set_resizable(column, " . $widget->{"resizable"} . ");\n";
309             }
310             $widget->{"noshow"} = 1;
311             $pf = sub
312             {
313             }
314         } elsif($widget->{"type"} eq "textrend") {
315             print "${p}renderer = gtk_cell_renderer_text_new();\n";
316             print "${p}gtk_tree_view_column_pack_start(column, renderer, " . (defined($widget->{"expand"})?$widget->{"expand"}:"TRUE") . ");\n";
317             if(defined($widget->{"text"})) {
318                 print "${p}gtk_tree_view_column_add_attribute(column, renderer, \"text\", " . $widget->{"text"} . ");\n";
319             }
320             if(defined($widget->{"func"})) {
321                 print "${p}gtk_tree_view_column_set_cell_data_func(column, renderer, " . $widget->{"func"} . ", " . ($widget->{"funcdata"} || "NULL") . ", NULL);\n";
322             }
323             $widget->{"noshow"} = 1;
324         } elsif($widget->{"type"} eq "custrend") {
325             print "${p}renderer = GTK_CELL_RENDERER(" . $widget->{"newfunc"} . "());\n";
326             print "${p}gtk_tree_view_column_pack_start(column, renderer, " . (defined($widget->{"expand"})?$widget->{"expand"}:"FALSE") . ");\n";
327             foreach $attr (keys %{$widget})
328             {
329                 if($attr =~ /attr\((\S+)\)/)
330                 {
331                     print "${p}gtk_tree_view_column_add_attribute(column, renderer, \"" . $1 . "\", " . $widget->{$attr} . ");\n";
332                 }
333             }
334             $widget->{"noshow"} = 1;
335         } elsif($widget->{"type"} eq "pixbufrend") {
336             print "${p}renderer = gtk_cell_renderer_pixbuf_new();\n";
337             print "${p}gtk_tree_view_column_pack_start(column, renderer, FALSE);\n";
338             if(defined($widget->{"stock_id"})) {
339                 print "${p}gtk_tree_view_column_add_attribute(column, renderer, \"stock_id\", " . $widget->{"stock_id"} . ");\n";
340             }
341             $widget->{"noshow"} = 1;
342         } elsif($widget->{"type"} eq "textview") {
343             print "${p}stack[$sl] = gtk_text_view_new();\n";
344             if(defined($widget->{"editable"})) {
345                 print "${p}gtk_text_view_set_editable(GTK_TEXT_VIEW(stack[$sl]), " . $widget->{"editable"} . ");\n";
346             }
347         } elsif($widget->{"type"} eq "pbar") {
348             print "${p}stack[$sl] = gtk_progress_bar_new();\n";
349         } elsif($widget->{"type"} eq "hr") {
350             print "${p}stack[$sl] = gtk_hseparator_new();\n";
351         } elsif($widget->{"type"} eq "sbar") {
352             print "${p}stack[$sl] = gtk_statusbar_new();\n";
353             if($widget->{"grip"} eq "n") {
354                 print "${p}gtk_statusbar_set_has_resize_grip(GTK_STATUSBAR(stack[$sl]), FALSE);\n";
355             }
356         } elsif($widget->{"type"} eq "filechooser") {
357             print "${p}stack[$sl] = gtk_file_chooser_widget_new(GTK_FILE_CHOOSER_ACTION_" . (defined($widget->{"action"})?$widget->{"action"}:"OPEN") . ");\n";
358         } else {
359             print STDERR "Unknown widget: " . $widget->{"type"} ."\n";
360         }
361         if($widget->{"sensitive"}) {
362             print "${p}gtk_widget_set_sensitive(stack[$sl], " . $widget->{"sensitive"} . ");\n";
363         }
364         if($widget->{"rqsz"} =~ /(\d+)x(\d+)/) {
365             print "${p}gtk_widget_set_size_request(stack[$sl], $1, $2);\n";
366         }
367         if($widget->{"align"} =~ /([\d\.]+):([\d\.]+)/) {
368             print "${p}gtk_misc_set_alignment(GTK_MISC(stack[$sl]), $1, $2);\n";
369         }
370         if($widget->{"wpad"} =~ /([\d\.]+):([\d\.]+)/) {
371             print "${p}gtk_misc_set_padding(GTK_MISC(stack[$sl]), $1, $2);\n";
372         }
373         if($widget->{"var"} eq "y") {
374             print $p . $options{"prefix"} . $widget->{"name"} . " = stack[$sl];\n";
375         }
376         if($widget->{"var"} eq "l") {
377             print $p . "GtkWidget *" . $widget->{"name"} . " = stack[$sl];\n";
378         }
379         if($widget->{"sig"})
380         {
381             while($widget->{"sig"} =~ /\G([\w_]+),?/g) {
382                 print "${p}g_signal_connect(G_OBJECT(stack[$sl]), \"$1\", G_CALLBACK(cb_" . $options{"prefix"} . $widget->{"name"} . "_" . $1 . "), (gpointer)NULL);\n";
383             }
384         }
385         if($widget->{"accel"})
386         {
387             $mod = "";
388             while($widget->{"accel"} =~ /\G(\w+)\+/gc)
389             {
390                 $mod .= " | " if($mod);
391                 $mod = $mod . "GDK_" . $1 . "_MASK";
392             }
393             $mod || ($mod = "0");
394             $widget->{"accel"} =~ /\G(\w+)/g;
395             $key = $1;
396             print "${p}gtk_widget_add_accelerator(stack[$sl], \"activate\", accel_group, GDK_$key, $mod, GTK_ACCEL_VISIBLE);\n";
397         }
398         foreach $attr (keys %{$widget})
399         {
400             if($attr =~ /^sig\((\S+)\)/)
401             {
402                 $sig = $1;
403                 if($widget->{$attr} =~ /([^,]*),(.*)/)
404                 {
405                     $cb = $1;
406                     $data = $2;
407                 } else {
408                     $cb = $widget->{$attr};
409                     $data = "NULL";
410                 }
411                 print "${p}g_signal_connect(G_OBJECT(stack[$sl]), \"$sig\", G_CALLBACK($cb), (gpointer)$data);\n";
412             }
413         }
414         if($widget->{"subwidgets"})
415         {
416             print "$p\n";
417             printwidgets($widget->{"subwidgets"}, $sl + 1, $pf);
418         }
419         if($sl > 0)
420         {
421             &$cpf($widget, $p, $sl);
422             if(!$widget->{"noshow"}) {
423                 print "${p}gtk_widget_show(stack[$sl]);\n";
424             }
425         }
426         print "$p\n";
427     }
428     foreach $line (@delayedlines)
429     {
430         print $p . $line . "\n";
431     }
432 }
433
434 sub printheader
435 {
436     my($widget, $handle);
437     $handle = $_[1];
438     foreach $widget (@{$_[0]})
439     {
440         if($widget->{"var"})
441         {
442             $handle->print("extern GtkWidget *" . $options{"prefix"} . $widget->{"name"} .";\n");
443         }
444         printheader($widget->{"subwidgets"}, $handle) if($widget->{"subwidgets"});
445     }
446 }
447
448 sub printvars
449 {
450     my($widget);
451     foreach $widget (@{$_[0]})
452     {
453         if($widget->{"var"})
454         {
455             print "GtkWidget *" . $options{"prefix"} . $widget->{"name"} .";\n";
456         }
457         printvars($widget->{"subwidgets"}) if($widget->{"subwidgets"});
458     }
459 }
460
461 sub dequote
462 {
463     my($text);
464     ($text) = @_;
465     $text =~ s/([^\\]|^)\"/$1/g;
466     $text =~ s/\\(.)/$1/g;
467     return $text;
468 }
469
470 $rootwidgets = [];
471 @estack = ($rootwidgets);
472 @wstack = ();
473 $curwidget = 0;
474 $maxstack = 1;
475
476 while(<>)
477 {
478     chomp;
479     s/(^|\s+)\#.*$//;
480     s/^\s*//;
481     if(/^;\s*(\w+)\s*:\s*(\w.*)/)
482     {
483         $options{$1} = $2;
484     } elsif(/^([:\$])\s*(\w+)/g) {
485         $curwidget = {"type" => $2};
486         push @{$estack[0]}, $curwidget;
487         if((scalar @wstack) > 0) {
488             $curwidget->{"parent"} = $wstack[0];
489         }
490         if($1 eq ":")
491         {
492             unshift @estack, ($curwidget->{"subwidgets"} = []);
493             unshift @wstack, $curwidget;
494         }
495         $maxstack = (scalar @estack) if((scalar @estack) > $maxstack);
496         while(/\G\s*(\S+)\s*:\s*((\w+|\"([^\\\"]+|\\.)*([^\\]|)\"|\\.)+)/g)
497         {
498             $curwidget->{$1} = dequote($2);
499         }
500     } elsif(/^%\s*(\S+)\s*:\s*((\w+|\"([^\\\"]+|\\.)*([^\\]|)\"|\\.)+)/) {
501         $curwidget || die("No current widget\n");
502         $curwidget->{$1} = dequote($2);
503     } elsif(/^end/) {
504         shift @estack;
505         shift @wstack;
506         $curwidget = $wstack[0] if((scalar @wstack) > 0);
507     } elsif(!$_) {
508     } else {
509         print STDERR "Invalid construct: $_\n";
510     }
511 }
512
513 if($args{"h"})
514 {
515     $hfile = IO::File->new($args{"h"}, "w") || die;
516     $hfile->print("GtkWidget *create_" . $options{"prefix"} . "wnd(void);\n");
517     printheader $rootwidgets, $hfile;
518     $hfile->close();
519 }
520
521 printvars $rootwidgets;
522 print "\n";
523 print "GtkWidget *create_" . $options{"prefix"} . "wnd(void)\n";
524 print "{\n";
525 print "    GtkWidget *stack[$maxstack];\n";
526 print "    GtkAccelGroup *accel_group;\n" if $options{"hasaccels"};
527 print "    GtkTreeViewColumn *column;\n" if $options{"hascolumns"};
528 print "    GtkCellRenderer *renderer;\n" if $options{"hasrenderers"};
529 print "    \n";
530 print "    accel_group = gtk_accel_group_new();\n" if $options{"hasaccels"};
531 printwidgets $rootwidgets, 0;
532 print "    return(stack[0]);\n";
533 print "}\n";