| // SPDX-License-Identifier: GPL-2.0 | 
 | /* | 
 |  * Copyright (C) 2002-2003 Romain Lievin <roms@tilp.info> | 
 |  */ | 
 |  | 
 | #include <stdlib.h> | 
 | #include "lkc.h" | 
 | #include "images.h" | 
 |  | 
 | #include <gtk/gtk.h> | 
 |  | 
 | #include <stdio.h> | 
 | #include <string.h> | 
 | #include <strings.h> | 
 | #include <unistd.h> | 
 | #include <time.h> | 
 |  | 
 | enum view_mode { | 
 | 	SINGLE_VIEW, SPLIT_VIEW, FULL_VIEW | 
 | }; | 
 |  | 
 | enum { | 
 | 	OPT_NORMAL, OPT_ALL, OPT_PROMPT | 
 | }; | 
 |  | 
 | static gint view_mode = FULL_VIEW; | 
 | static gboolean show_name = TRUE; | 
 | static gboolean show_range = TRUE; | 
 | static gboolean show_value = TRUE; | 
 | static int opt_mode = OPT_NORMAL; | 
 |  | 
 | static GtkWidget *main_wnd; | 
 | static GtkWidget *tree1_w;	// left  frame | 
 | static GtkWidget *tree2_w;	// right frame | 
 | static GtkWidget *text_w; | 
 | static GtkWidget *hpaned; | 
 | static GtkWidget *vpaned; | 
 | static GtkWidget *back_btn, *save_btn, *single_btn, *split_btn, *full_btn; | 
 | static GtkWidget *save_menu_item; | 
 |  | 
 | static GtkTextTag *tag1, *tag2; | 
 |  | 
 | static GtkTreeStore *tree1, *tree2; | 
 | static GdkPixbuf *pix_menu; | 
 |  | 
 | static struct menu *browsed; // browsed menu for SINGLE/SPLIT view | 
 | static struct menu *selected; // selected entry | 
 |  | 
 | enum { | 
 | 	COL_OPTION, COL_NAME, COL_NO, COL_MOD, COL_YES, COL_VALUE, | 
 | 	COL_MENU, COL_COLOR, COL_EDIT, COL_PIXBUF, | 
 | 	COL_PIXVIS, COL_BTNVIS, COL_BTNACT, COL_BTNINC, COL_BTNRAD, | 
 | 	COL_NUMBER | 
 | }; | 
 |  | 
 | static void display_tree(GtkTreeStore *store, struct menu *menu); | 
 | static void recreate_tree(void); | 
 |  | 
 | static void conf_changed(bool dirty) | 
 | { | 
 | 	gtk_widget_set_sensitive(save_btn, dirty); | 
 | 	gtk_widget_set_sensitive(save_menu_item, dirty); | 
 | } | 
 |  | 
 | /* Utility Functions */ | 
 |  | 
 | static void text_insert_msg(const char *title, const char *msg) | 
 | { | 
 | 	GtkTextBuffer *buffer; | 
 | 	GtkTextIter start, end; | 
 |  | 
 | 	buffer = gtk_text_view_get_buffer(GTK_TEXT_VIEW(text_w)); | 
 | 	gtk_text_buffer_get_bounds(buffer, &start, &end); | 
 | 	gtk_text_buffer_delete(buffer, &start, &end); | 
 | 	gtk_text_view_set_left_margin(GTK_TEXT_VIEW(text_w), 15); | 
 |  | 
 | 	gtk_text_buffer_get_end_iter(buffer, &end); | 
 | 	gtk_text_buffer_insert_with_tags(buffer, &end, title, -1, tag1, | 
 | 					 NULL); | 
 | 	gtk_text_buffer_insert_at_cursor(buffer, "\n\n", 2); | 
 | 	gtk_text_buffer_get_end_iter(buffer, &end); | 
 | 	gtk_text_buffer_insert_with_tags(buffer, &end, msg, -1, tag2, | 
 | 					 NULL); | 
 | } | 
 |  | 
 | static void text_insert_help(struct menu *menu) | 
 | { | 
 | 	struct gstr help = str_new(); | 
 |  | 
 | 	menu_get_ext_help(menu, &help); | 
 | 	text_insert_msg(menu_get_prompt(menu), str_get(&help)); | 
 | 	str_free(&help); | 
 | } | 
 |  | 
 | static void _select_menu(GtkTreeView *view, GtkTreeModel *model, | 
 | 			 GtkTreeIter *parent, struct menu *match) | 
 | { | 
 | 	GtkTreeIter iter; | 
 | 	gboolean valid; | 
 |  | 
 | 	valid = gtk_tree_model_iter_children(model, &iter, parent); | 
 | 	while (valid) { | 
 | 		struct menu *menu; | 
 |  | 
 | 		gtk_tree_model_get(model, &iter, COL_MENU, &menu, -1); | 
 |  | 
 | 		if (menu == match) { | 
 | 			GtkTreeSelection *selection; | 
 | 			GtkTreePath *path; | 
 |  | 
 | 			/* | 
 | 			 * Expand parents to reflect the selection, and | 
 | 			 * scroll down to it. | 
 | 			 */ | 
 | 			path = gtk_tree_model_get_path(model, &iter); | 
 | 			gtk_tree_view_expand_to_path(view, path); | 
 | 			gtk_tree_view_scroll_to_cell(view, path, NULL, TRUE, | 
 | 						     0.5, 0.0); | 
 | 			gtk_tree_path_free(path); | 
 |  | 
 | 			selection = gtk_tree_view_get_selection(view); | 
 | 			gtk_tree_selection_select_iter(selection, &iter); | 
 |  | 
 | 			text_insert_help(menu); | 
 | 		} | 
 |  | 
 | 		_select_menu(view, model, &iter, match); | 
 |  | 
 | 		valid = gtk_tree_model_iter_next(model, &iter); | 
 | 	} | 
 | } | 
 |  | 
 | static void select_menu(GtkTreeView *view, struct menu *match) | 
 | { | 
 | 	_select_menu(view, gtk_tree_view_get_model(view), NULL, match); | 
 | } | 
 |  | 
 | static void _update_row_visibility(GtkTreeView *view) | 
 | { | 
 | 	GtkTreeModelFilter *filter = GTK_TREE_MODEL_FILTER(gtk_tree_view_get_model(view)); | 
 |  | 
 | 	gtk_tree_model_filter_refilter(filter); | 
 | } | 
 |  | 
 | static void update_row_visibility(void) | 
 | { | 
 | 	if (view_mode == SPLIT_VIEW) | 
 | 		_update_row_visibility(GTK_TREE_VIEW(tree1_w)); | 
 | 	_update_row_visibility(GTK_TREE_VIEW(tree2_w)); | 
 | } | 
 |  | 
 | static void set_node(GtkTreeStore *tree, GtkTreeIter *node, struct menu *menu) | 
 | { | 
 | 	struct symbol *sym = menu->sym; | 
 | 	tristate val; | 
 | 	gchar *option; | 
 | 	const gchar *_no = ""; | 
 | 	const gchar *_mod = ""; | 
 | 	const gchar *_yes = ""; | 
 | 	const gchar *value = ""; | 
 | 	GdkRGBA color; | 
 | 	gboolean editable = FALSE; | 
 | 	gboolean btnvis = FALSE; | 
 |  | 
 | 	option = g_strdup_printf("%s %s %s %s", | 
 | 				 menu->type == M_COMMENT ? "***" : "", | 
 | 				 menu_get_prompt(menu), | 
 | 				 menu->type == M_COMMENT ? "***" : "", | 
 | 				 sym && !sym_has_value(sym) ? "(NEW)" : ""); | 
 |  | 
 | 	gdk_rgba_parse(&color, menu_is_visible(menu) ? "Black" : "DarkGray"); | 
 |  | 
 | 	if (!sym) | 
 | 		goto set; | 
 |  | 
 | 	sym_calc_value(sym); | 
 |  | 
 | 	if (menu->type == M_CHOICE) {	// parse children to get a final value | 
 | 		struct symbol *def_sym = sym_calc_choice(menu); | 
 | 		struct menu *def_menu = NULL; | 
 |  | 
 | 		for (struct menu *child = menu->list; child; child = child->next) { | 
 | 			if (menu_is_visible(child) && child->sym == def_sym) | 
 | 				def_menu = child; | 
 | 		} | 
 |  | 
 | 		if (def_menu) | 
 | 			value = menu_get_prompt(def_menu); | 
 |  | 
 | 		goto set; | 
 | 	} | 
 |  | 
 | 	switch (sym_get_type(sym)) { | 
 | 	case S_BOOLEAN: | 
 | 	case S_TRISTATE: | 
 |  | 
 | 		btnvis = TRUE; | 
 |  | 
 | 		val = sym_get_tristate_value(sym); | 
 | 		switch (val) { | 
 | 		case no: | 
 | 			_no = "N"; | 
 | 			value = "N"; | 
 | 			break; | 
 | 		case mod: | 
 | 			_mod = "M"; | 
 | 			value = "M"; | 
 | 			break; | 
 | 		case yes: | 
 | 			_yes = "Y"; | 
 | 			value = "Y"; | 
 | 			break; | 
 | 		} | 
 |  | 
 | 		if (val != no && sym_tristate_within_range(sym, no)) | 
 | 			_no = "_"; | 
 | 		if (val != mod && sym_tristate_within_range(sym, mod)) | 
 | 			_mod = "_"; | 
 | 		if (val != yes && sym_tristate_within_range(sym, yes)) | 
 | 			_yes = "_"; | 
 | 		break; | 
 | 	default: | 
 | 		value = sym_get_string_value(sym); | 
 | 		editable = TRUE; | 
 | 		break; | 
 | 	} | 
 |  | 
 | set: | 
 | 	gtk_tree_store_set(tree, node, | 
 | 			   COL_OPTION, option, | 
 | 			   COL_NAME, sym ? sym->name : "", | 
 | 			   COL_NO, _no, | 
 | 			   COL_MOD, _mod, | 
 | 			   COL_YES, _yes, | 
 | 			   COL_VALUE, value, | 
 | 			   COL_MENU, (gpointer) menu, | 
 | 			   COL_COLOR, &color, | 
 | 			   COL_EDIT, editable, | 
 | 			   COL_PIXBUF, pix_menu, | 
 | 			   COL_PIXVIS, view_mode == SINGLE_VIEW && menu->type == M_MENU, | 
 | 			   COL_BTNVIS, btnvis, | 
 | 			   COL_BTNACT, _yes[0] == 'Y', | 
 | 			   COL_BTNINC, _mod[0] == 'M', | 
 | 			   COL_BTNRAD, sym && sym_is_choice_value(sym), | 
 | 			   -1); | 
 |  | 
 | 	g_free(option); | 
 | } | 
 |  | 
 | static void _update_tree(GtkTreeStore *store, GtkTreeIter *parent) | 
 | { | 
 | 	GtkTreeModel *model = GTK_TREE_MODEL(store); | 
 | 	GtkTreeIter iter; | 
 | 	gboolean valid; | 
 |  | 
 | 	valid = gtk_tree_model_iter_children(model, &iter, parent); | 
 | 	while (valid) { | 
 | 		struct menu *menu; | 
 |  | 
 | 		gtk_tree_model_get(model, &iter, COL_MENU, &menu, -1); | 
 |  | 
 | 		if (menu) | 
 | 			set_node(store, &iter, menu); | 
 |  | 
 | 		_update_tree(store, &iter); | 
 |  | 
 | 		valid = gtk_tree_model_iter_next(model, &iter); | 
 | 	} | 
 | } | 
 |  | 
 | static void update_tree(GtkTreeStore *store) | 
 | { | 
 | 	_update_tree(store, NULL); | 
 | 	update_row_visibility(); | 
 | } | 
 |  | 
 | static void update_trees(void) | 
 | { | 
 | 	if (view_mode == SPLIT_VIEW) | 
 | 		update_tree(tree1); | 
 | 	update_tree(tree2); | 
 | } | 
 |  | 
 | static void set_view_mode(enum view_mode mode) | 
 | { | 
 | 	view_mode = mode; | 
 |  | 
 | 	if (mode == SPLIT_VIEW) { // two panes | 
 | 		gint w; | 
 |  | 
 | 		gtk_widget_show(tree1_w); | 
 | 		gtk_window_get_default_size(GTK_WINDOW(main_wnd), &w, NULL); | 
 | 		gtk_paned_set_position(GTK_PANED(hpaned), w / 2); | 
 | 	} else { | 
 | 		gtk_widget_hide(tree1_w); | 
 | 		gtk_paned_set_position(GTK_PANED(hpaned), 0); | 
 | 	} | 
 |  | 
 | 	gtk_widget_set_sensitive(single_btn, TRUE); | 
 | 	gtk_widget_set_sensitive(split_btn, TRUE); | 
 | 	gtk_widget_set_sensitive(full_btn, TRUE); | 
 |  | 
 | 	switch (mode) { | 
 | 	case SINGLE_VIEW: | 
 | 		if (selected) | 
 | 			browsed = menu_get_parent_menu(selected) ?: &rootmenu; | 
 | 		else | 
 | 			browsed = &rootmenu; | 
 | 		recreate_tree(); | 
 | 		text_insert_msg("", ""); | 
 | 		select_menu(GTK_TREE_VIEW(tree2_w), selected); | 
 | 		gtk_widget_set_sensitive(single_btn, FALSE); | 
 | 		break; | 
 | 	case SPLIT_VIEW: | 
 | 		browsed = selected; | 
 | 		while (browsed && !(browsed->flags & MENU_ROOT)) | 
 | 			browsed = browsed->parent; | 
 | 		gtk_tree_store_clear(tree1); | 
 | 		display_tree(tree1, &rootmenu); | 
 | 		gtk_tree_view_expand_all(GTK_TREE_VIEW(tree1_w)); | 
 | 		gtk_tree_store_clear(tree2); | 
 | 		if (browsed) | 
 | 			display_tree(tree2, browsed); | 
 | 		text_insert_msg("", ""); | 
 | 		select_menu(GTK_TREE_VIEW(tree1_w), browsed); | 
 | 		select_menu(GTK_TREE_VIEW(tree2_w), selected); | 
 | 		gtk_widget_set_sensitive(split_btn, FALSE); | 
 | 		break; | 
 | 	case FULL_VIEW: | 
 | 		gtk_tree_store_clear(tree2); | 
 | 		display_tree(tree2, &rootmenu); | 
 | 		text_insert_msg("", ""); | 
 | 		select_menu(GTK_TREE_VIEW(tree2_w), selected); | 
 | 		gtk_widget_set_sensitive(full_btn, FALSE); | 
 | 		break; | 
 | 	} | 
 |  | 
 | 	gtk_widget_set_sensitive(back_btn, | 
 | 				 mode == SINGLE_VIEW && browsed != &rootmenu); | 
 | } | 
 |  | 
 | /* Menu & Toolbar Callbacks */ | 
 |  | 
 | static void on_load1_activate(GtkMenuItem *menuitem, gpointer user_data) | 
 | { | 
 | 	GtkWidget *dialog; | 
 | 	GtkFileChooser *chooser; | 
 | 	gint res; | 
 |  | 
 | 	dialog = gtk_file_chooser_dialog_new("Load file...", | 
 | 					     GTK_WINDOW(user_data), | 
 | 					     GTK_FILE_CHOOSER_ACTION_OPEN, | 
 | 					     "_Cancel", GTK_RESPONSE_CANCEL, | 
 | 					     "_Open", GTK_RESPONSE_ACCEPT, | 
 | 					     NULL); | 
 |  | 
 | 	chooser = GTK_FILE_CHOOSER(dialog); | 
 | 	gtk_file_chooser_set_filename(chooser, conf_get_configname()); | 
 |  | 
 | 	res = gtk_dialog_run(GTK_DIALOG(dialog)); | 
 | 	if (res == GTK_RESPONSE_ACCEPT) { | 
 | 		char *filename; | 
 |  | 
 | 		filename = gtk_file_chooser_get_filename(chooser); | 
 |  | 
 | 		if (conf_read(filename)) | 
 | 			text_insert_msg("Error", | 
 | 					"Unable to load configuration!"); | 
 | 		else | 
 | 			update_trees(); | 
 |  | 
 | 		g_free(filename); | 
 | 	} | 
 |  | 
 | 	gtk_widget_destroy(GTK_WIDGET(dialog)); | 
 | } | 
 |  | 
 | static void on_save_activate(GtkMenuItem *menuitem, gpointer user_data) | 
 | { | 
 | 	if (conf_write(NULL)) | 
 | 		text_insert_msg("Error", "Unable to save configuration !"); | 
 | 	conf_write_autoconf(0); | 
 | } | 
 |  | 
 | static void on_save_as1_activate(GtkMenuItem *menuitem, gpointer user_data) | 
 | { | 
 | 	GtkWidget *dialog; | 
 | 	GtkFileChooser *chooser; | 
 | 	gint res; | 
 |  | 
 | 	dialog = gtk_file_chooser_dialog_new("Save file as...", | 
 | 					     GTK_WINDOW(user_data), | 
 | 					     GTK_FILE_CHOOSER_ACTION_SAVE, | 
 | 					     "_Cancel", GTK_RESPONSE_CANCEL, | 
 | 					     "_Save", GTK_RESPONSE_ACCEPT, | 
 | 					     NULL); | 
 |  | 
 | 	chooser = GTK_FILE_CHOOSER(dialog); | 
 | 	gtk_file_chooser_set_filename(chooser, conf_get_configname()); | 
 |  | 
 | 	res = gtk_dialog_run(GTK_DIALOG(dialog)); | 
 | 	if (res == GTK_RESPONSE_ACCEPT) { | 
 | 		char *filename; | 
 |  | 
 | 		filename = gtk_file_chooser_get_filename(chooser); | 
 |  | 
 | 		if (conf_write(filename)) | 
 | 			text_insert_msg("Error", | 
 | 					"Unable to save configuration !"); | 
 |  | 
 | 		g_free(filename); | 
 | 	} | 
 |  | 
 | 	gtk_widget_destroy(dialog); | 
 | } | 
 |  | 
 | static void on_show_name1_activate(GtkMenuItem *menuitem, gpointer user_data) | 
 | { | 
 | 	GtkTreeViewColumn *col; | 
 |  | 
 | 	show_name = gtk_check_menu_item_get_active(GTK_CHECK_MENU_ITEM(menuitem)); | 
 | 	col = gtk_tree_view_get_column(GTK_TREE_VIEW(tree2_w), COL_NAME); | 
 | 	if (col) | 
 | 		gtk_tree_view_column_set_visible(col, show_name); | 
 | } | 
 |  | 
 | static void on_show_range1_activate(GtkMenuItem *menuitem, gpointer user_data) | 
 | { | 
 | 	GtkTreeViewColumn *col; | 
 |  | 
 | 	show_range = gtk_check_menu_item_get_active(GTK_CHECK_MENU_ITEM(menuitem)); | 
 | 	col = gtk_tree_view_get_column(GTK_TREE_VIEW(tree2_w), COL_NO); | 
 | 	if (col) | 
 | 		gtk_tree_view_column_set_visible(col, show_range); | 
 | 	col = gtk_tree_view_get_column(GTK_TREE_VIEW(tree2_w), COL_MOD); | 
 | 	if (col) | 
 | 		gtk_tree_view_column_set_visible(col, show_range); | 
 | 	col = gtk_tree_view_get_column(GTK_TREE_VIEW(tree2_w), COL_YES); | 
 | 	if (col) | 
 | 		gtk_tree_view_column_set_visible(col, show_range); | 
 |  | 
 | } | 
 |  | 
 | static void on_show_data1_activate(GtkMenuItem *menuitem, gpointer user_data) | 
 | { | 
 | 	GtkTreeViewColumn *col; | 
 |  | 
 | 	show_value = gtk_check_menu_item_get_active(GTK_CHECK_MENU_ITEM(menuitem)); | 
 | 	col = gtk_tree_view_get_column(GTK_TREE_VIEW(tree2_w), COL_VALUE); | 
 | 	if (col) | 
 | 		gtk_tree_view_column_set_visible(col, show_value); | 
 | } | 
 |  | 
 | static void on_set_option_mode1_activate(GtkMenuItem *menuitem, | 
 | 					 gpointer user_data) | 
 | { | 
 | 	opt_mode = OPT_NORMAL; | 
 | 	update_row_visibility(); | 
 | } | 
 |  | 
 | static void on_set_option_mode2_activate(GtkMenuItem *menuitem, | 
 | 					 gpointer user_data) | 
 | { | 
 | 	opt_mode = OPT_ALL; | 
 | 	update_row_visibility(); | 
 | } | 
 |  | 
 | static void on_set_option_mode3_activate(GtkMenuItem *menuitem, | 
 | 					 gpointer user_data) | 
 | { | 
 | 	opt_mode = OPT_PROMPT; | 
 | 	update_row_visibility(); | 
 | } | 
 |  | 
 | static void on_introduction1_activate(GtkMenuItem *menuitem, gpointer user_data) | 
 | { | 
 | 	GtkWidget *dialog; | 
 | 	const gchar *intro_text = | 
 | 	    "Welcome to gconfig, the GTK+ graphical configuration tool.\n" | 
 | 	    "For each option, a blank box indicates the feature is disabled, a\n" | 
 | 	    "check indicates it is enabled, and a dot indicates that it is to\n" | 
 | 	    "be compiled as a module.  Clicking on the box will cycle through the three states.\n" | 
 | 	    "\n" | 
 | 	    "If you do not see an option (e.g., a device driver) that you\n" | 
 | 	    "believe should be present, try turning on Show All Options\n" | 
 | 	    "under the Options menu.\n" | 
 | 	    "Although there is no cross reference yet to help you figure out\n" | 
 | 	    "what other options must be enabled to support the option you\n" | 
 | 	    "are interested in, you can still view the help of a grayed-out\n" | 
 | 	    "option."; | 
 |  | 
 | 	dialog = gtk_message_dialog_new(GTK_WINDOW(main_wnd), | 
 | 					GTK_DIALOG_DESTROY_WITH_PARENT, | 
 | 					GTK_MESSAGE_INFO, | 
 | 					GTK_BUTTONS_CLOSE, "%s", intro_text); | 
 | 	gtk_dialog_run(GTK_DIALOG(dialog)); | 
 | 	gtk_widget_destroy(dialog); | 
 | } | 
 |  | 
 | static void on_about1_activate(GtkMenuItem *menuitem, gpointer user_data) | 
 | { | 
 | 	GtkWidget *dialog; | 
 | 	const gchar *about_text = | 
 | 	    "gconfig is copyright (c) 2002 Romain Lievin <roms@lpg.ticalc.org>.\n" | 
 | 	      "Based on the source code from Roman Zippel.\n"; | 
 |  | 
 | 	dialog = gtk_message_dialog_new(GTK_WINDOW(main_wnd), | 
 | 					GTK_DIALOG_DESTROY_WITH_PARENT, | 
 | 					GTK_MESSAGE_INFO, | 
 | 					GTK_BUTTONS_CLOSE, "%s\nGTK version: %d.%d.%d", | 
 | 					about_text, | 
 | 					gtk_get_major_version(), | 
 | 					gtk_get_minor_version(), | 
 | 					gtk_get_micro_version()); | 
 | 	gtk_dialog_run(GTK_DIALOG(dialog)); | 
 | 	gtk_widget_destroy(dialog); | 
 | } | 
 |  | 
 | static void on_license1_activate(GtkMenuItem *menuitem, gpointer user_data) | 
 | { | 
 | 	GtkWidget *dialog; | 
 | 	const gchar *license_text = | 
 | 	    "gconfig is released under the terms of the GNU GPL v2.\n" | 
 | 	      "For more information, please see the source code or\n" | 
 | 	      "visit http://www.fsf.org/licenses/licenses.html\n"; | 
 |  | 
 | 	dialog = gtk_message_dialog_new(GTK_WINDOW(main_wnd), | 
 | 					GTK_DIALOG_DESTROY_WITH_PARENT, | 
 | 					GTK_MESSAGE_INFO, | 
 | 					GTK_BUTTONS_CLOSE, "%s", license_text); | 
 | 	gtk_dialog_run(GTK_DIALOG(dialog)); | 
 | 	gtk_widget_destroy(dialog); | 
 | } | 
 |  | 
 | /* toolbar handlers */ | 
 | static void on_back_clicked(GtkButton *button, gpointer user_data) | 
 | { | 
 | 	browsed = menu_get_parent_menu(browsed) ?: &rootmenu; | 
 |  | 
 | 	recreate_tree(); | 
 |  | 
 | 	if (browsed == &rootmenu) | 
 | 		gtk_widget_set_sensitive(back_btn, FALSE); | 
 | } | 
 |  | 
 | static void on_load_clicked(GtkButton *button, gpointer user_data) | 
 | { | 
 | 	on_load1_activate(NULL, user_data); | 
 | } | 
 |  | 
 | static void on_save_clicked(GtkButton *button, gpointer user_data) | 
 | { | 
 | 	on_save_activate(NULL, user_data); | 
 | } | 
 |  | 
 | static void on_single_clicked(GtkButton *button, gpointer user_data) | 
 | { | 
 | 	set_view_mode(SINGLE_VIEW); | 
 | } | 
 |  | 
 | static void on_split_clicked(GtkButton *button, gpointer user_data) | 
 | { | 
 | 	set_view_mode(SPLIT_VIEW); | 
 | } | 
 |  | 
 | static void on_full_clicked(GtkButton *button, gpointer user_data) | 
 | { | 
 | 	set_view_mode(FULL_VIEW); | 
 | } | 
 |  | 
 | static void on_collapse_clicked(GtkButton *button, gpointer user_data) | 
 | { | 
 | 	gtk_tree_view_collapse_all(GTK_TREE_VIEW(tree2_w)); | 
 | } | 
 |  | 
 | static void on_expand_clicked(GtkButton *button, gpointer user_data) | 
 | { | 
 | 	gtk_tree_view_expand_all(GTK_TREE_VIEW(tree2_w)); | 
 | } | 
 |  | 
 | /* Main Windows Callbacks */ | 
 |  | 
 | static void on_window1_destroy(GtkWidget *widget, gpointer user_data) | 
 | { | 
 | 	gtk_main_quit(); | 
 | } | 
 |  | 
 | static gboolean on_window1_configure(GtkWidget *self, | 
 | 				     GdkEventConfigure *event, | 
 | 				     gpointer user_data) | 
 | { | 
 | 	gtk_paned_set_position(GTK_PANED(vpaned), 2 * event->height / 3); | 
 | 	return FALSE; | 
 | } | 
 |  | 
 | static gboolean on_window1_delete_event(GtkWidget *widget, GdkEvent *event, | 
 | 					gpointer user_data) | 
 | { | 
 | 	GtkWidget *dialog, *label, *content_area; | 
 | 	gint result; | 
 | 	gint ret = FALSE; | 
 |  | 
 | 	if (!conf_get_changed()) | 
 | 		return FALSE; | 
 |  | 
 | 	dialog = gtk_dialog_new_with_buttons("Warning !", | 
 | 					     GTK_WINDOW(main_wnd), | 
 | 					     (GtkDialogFlags) | 
 | 					     (GTK_DIALOG_MODAL | | 
 | 					      GTK_DIALOG_DESTROY_WITH_PARENT), | 
 | 					     "_OK", | 
 | 					     GTK_RESPONSE_YES, | 
 | 					     "_No", | 
 | 					     GTK_RESPONSE_NO, | 
 | 					     "_Cancel", | 
 | 					     GTK_RESPONSE_CANCEL, NULL); | 
 | 	gtk_dialog_set_default_response(GTK_DIALOG(dialog), | 
 | 					GTK_RESPONSE_CANCEL); | 
 |  | 
 | 	label = gtk_label_new("\nSave configuration ?\n"); | 
 | 	content_area = gtk_dialog_get_content_area(GTK_DIALOG(dialog)); | 
 | 	gtk_container_add(GTK_CONTAINER(content_area), label); | 
 | 	gtk_widget_show(label); | 
 |  | 
 | 	result = gtk_dialog_run(GTK_DIALOG(dialog)); | 
 | 	switch (result) { | 
 | 	case GTK_RESPONSE_YES: | 
 | 		on_save_activate(NULL, NULL); | 
 | 		break; | 
 | 	case GTK_RESPONSE_NO: | 
 | 		break; | 
 | 	case GTK_RESPONSE_CANCEL: | 
 | 	case GTK_RESPONSE_DELETE_EVENT: | 
 | 	default: | 
 | 		ret = TRUE; | 
 | 		break; | 
 | 	} | 
 |  | 
 | 	gtk_widget_destroy(dialog); | 
 |  | 
 | 	if (!ret) | 
 | 		g_object_unref(pix_menu); | 
 |  | 
 | 	return ret; | 
 | } | 
 |  | 
 | static void on_quit1_activate(GtkMenuItem *menuitem, gpointer user_data) | 
 | { | 
 | 	if (!on_window1_delete_event(NULL, NULL, NULL)) | 
 | 		gtk_widget_destroy(GTK_WIDGET(main_wnd)); | 
 | } | 
 |  | 
 | /* CTree Callbacks */ | 
 |  | 
 | /* Change hex/int/string value in the cell */ | 
 | static void renderer_edited(GtkCellRendererText * cell, | 
 | 			    const gchar * path_string, | 
 | 			    const gchar * new_text, gpointer user_data) | 
 | { | 
 | 	GtkTreeView *view = GTK_TREE_VIEW(user_data); | 
 | 	GtkTreeModel *model = gtk_tree_view_get_model(view); | 
 | 	GtkTreePath *path = gtk_tree_path_new_from_string(path_string); | 
 | 	GtkTreeIter iter; | 
 | 	const char *old_def, *new_def; | 
 | 	struct menu *menu; | 
 | 	struct symbol *sym; | 
 |  | 
 | 	if (!gtk_tree_model_get_iter(model, &iter, path)) | 
 | 		goto free; | 
 |  | 
 | 	gtk_tree_model_get(model, &iter, COL_MENU, &menu, -1); | 
 | 	sym = menu->sym; | 
 |  | 
 | 	gtk_tree_model_get(model, &iter, COL_VALUE, &old_def, -1); | 
 | 	new_def = new_text; | 
 |  | 
 | 	sym_set_string_value(sym, new_def); | 
 |  | 
 | 	update_trees(); | 
 |  | 
 | free: | 
 | 	gtk_tree_path_free(path); | 
 | } | 
 |  | 
 | /* Change the value of a symbol and update the tree */ | 
 | static void change_sym_value(struct menu *menu, gint col) | 
 | { | 
 | 	struct symbol *sym = menu->sym; | 
 | 	tristate newval; | 
 |  | 
 | 	if (!sym) | 
 | 		return; | 
 |  | 
 | 	if (col == COL_NO) | 
 | 		newval = no; | 
 | 	else if (col == COL_MOD) | 
 | 		newval = mod; | 
 | 	else if (col == COL_YES) | 
 | 		newval = yes; | 
 | 	else | 
 | 		return; | 
 |  | 
 | 	switch (sym_get_type(sym)) { | 
 | 	case S_BOOLEAN: | 
 | 	case S_TRISTATE: | 
 | 		if (!sym_tristate_within_range(sym, newval)) | 
 | 			newval = yes; | 
 | 		sym_set_tristate_value(sym, newval); | 
 | 		update_trees(); | 
 | 		break; | 
 | 	case S_INT: | 
 | 	case S_HEX: | 
 | 	case S_STRING: | 
 | 	default: | 
 | 		break; | 
 | 	} | 
 | } | 
 |  | 
 | static void toggle_sym_value(struct menu *menu) | 
 | { | 
 | 	if (!menu->sym) | 
 | 		return; | 
 |  | 
 | 	sym_toggle_tristate_value(menu->sym); | 
 | 	update_trees(); | 
 | } | 
 |  | 
 | static gint column2index(GtkTreeViewColumn * column) | 
 | { | 
 | 	gint i; | 
 |  | 
 | 	for (i = 0; i < COL_NUMBER; i++) { | 
 | 		GtkTreeViewColumn *col; | 
 |  | 
 | 		col = gtk_tree_view_get_column(GTK_TREE_VIEW(tree2_w), i); | 
 | 		if (col == column) | 
 | 			return i; | 
 | 	} | 
 |  | 
 | 	return -1; | 
 | } | 
 |  | 
 |  | 
 | /* User click: update choice (full) or goes down (single) */ | 
 | static gboolean on_treeview2_button_press_event(GtkWidget *widget, | 
 | 						GdkEventButton *event, | 
 | 						gpointer user_data) | 
 | { | 
 | 	GtkTreeView *view = GTK_TREE_VIEW(widget); | 
 | 	GtkTreeModel *model = gtk_tree_view_get_model(view); | 
 | 	GtkTreePath *path; | 
 | 	GtkTreeViewColumn *column; | 
 | 	GtkTreeIter iter; | 
 | 	struct menu *menu; | 
 | 	gint col; | 
 | 	gint tx = (gint) event->x; | 
 | 	gint ty = (gint) event->y; | 
 |  | 
 | 	gtk_tree_view_get_path_at_pos(view, tx, ty, &path, &column, NULL, NULL); | 
 | 	if (path == NULL) | 
 | 		return FALSE; | 
 |  | 
 | 	if (!gtk_tree_model_get_iter(model, &iter, path)) | 
 | 		return FALSE; | 
 | 	gtk_tree_model_get(model, &iter, COL_MENU, &menu, -1); | 
 |  | 
 | 	selected = menu; | 
 |  | 
 | 	col = column2index(column); | 
 | 	if (event->type == GDK_2BUTTON_PRESS) { | 
 | 		enum prop_type ptype; | 
 | 		ptype = menu->prompt ? menu->prompt->type : P_UNKNOWN; | 
 |  | 
 | 		if (ptype == P_MENU && view_mode == SINGLE_VIEW && col == COL_OPTION) { | 
 | 			// goes down into menu | 
 | 			browsed = menu; | 
 | 			recreate_tree(); | 
 | 			gtk_widget_set_sensitive(back_btn, TRUE); | 
 | 		} else if (col == COL_OPTION) { | 
 | 			toggle_sym_value(menu); | 
 | 			gtk_tree_view_expand_row(view, path, TRUE); | 
 | 		} | 
 | 	} else { | 
 | 		if (col == COL_VALUE) { | 
 | 			toggle_sym_value(menu); | 
 | 			gtk_tree_view_expand_row(view, path, TRUE); | 
 | 		} else if (col == COL_NO || col == COL_MOD | 
 | 			   || col == COL_YES) { | 
 | 			change_sym_value(menu, col); | 
 | 			gtk_tree_view_expand_row(view, path, TRUE); | 
 | 		} | 
 | 	} | 
 |  | 
 | 	return FALSE; | 
 | } | 
 |  | 
 | /* Key pressed: update choice */ | 
 | static gboolean on_treeview2_key_press_event(GtkWidget *widget, | 
 | 					     GdkEventKey *event, | 
 | 					     gpointer user_data) | 
 | { | 
 | 	GtkTreeView *view = GTK_TREE_VIEW(widget); | 
 | 	GtkTreeModel *model = gtk_tree_view_get_model(view); | 
 | 	GtkTreePath *path; | 
 | 	GtkTreeIter iter; | 
 | 	struct menu *menu; | 
 | 	gint col; | 
 |  | 
 | 	gtk_tree_view_get_cursor(view, &path, NULL); | 
 | 	if (path == NULL) | 
 | 		return FALSE; | 
 |  | 
 | 	if (event->keyval == GDK_KEY_space) { | 
 | 		if (gtk_tree_view_row_expanded(view, path)) | 
 | 			gtk_tree_view_collapse_row(view, path); | 
 | 		else | 
 | 			gtk_tree_view_expand_row(view, path, FALSE); | 
 | 		return TRUE; | 
 | 	} | 
 |  | 
 | 	gtk_tree_model_get_iter(model, &iter, path); | 
 | 	gtk_tree_model_get(model, &iter, COL_MENU, &menu, -1); | 
 |  | 
 | 	if (!strcasecmp(event->string, "n")) | 
 | 		col = COL_NO; | 
 | 	else if (!strcasecmp(event->string, "m")) | 
 | 		col = COL_MOD; | 
 | 	else if (!strcasecmp(event->string, "y")) | 
 | 		col = COL_YES; | 
 | 	else | 
 | 		col = -1; | 
 | 	change_sym_value(menu, col); | 
 |  | 
 | 	return FALSE; | 
 | } | 
 |  | 
 |  | 
 | /* Row selection changed: update help */ | 
 | static void on_treeview2_cursor_changed(GtkTreeView *treeview, | 
 | 					gpointer user_data) | 
 | { | 
 | 	GtkTreeModel *model = gtk_tree_view_get_model(treeview); | 
 | 	GtkTreeSelection *selection; | 
 | 	GtkTreeIter iter; | 
 | 	struct menu *menu; | 
 |  | 
 | 	selection = gtk_tree_view_get_selection(treeview); | 
 | 	if (gtk_tree_selection_get_selected(selection, &model, &iter)) { | 
 | 		gtk_tree_model_get(model, &iter, COL_MENU, &menu, -1); | 
 | 		text_insert_help(menu); | 
 | 	} | 
 | } | 
 |  | 
 |  | 
 | /* User click: display sub-tree in the right frame. */ | 
 | static gboolean on_treeview1_button_press_event(GtkWidget *widget, | 
 | 						GdkEventButton *event, | 
 | 						gpointer user_data) | 
 | { | 
 | 	GtkTreeView *view = GTK_TREE_VIEW(widget); | 
 | 	GtkTreeModel *model = gtk_tree_view_get_model(view); | 
 | 	GtkTreePath *path; | 
 | 	GtkTreeIter iter; | 
 | 	struct menu *menu; | 
 | 	gint tx = (gint) event->x; | 
 | 	gint ty = (gint) event->y; | 
 |  | 
 | 	gtk_tree_view_get_path_at_pos(view, tx, ty, &path, NULL, NULL, NULL); | 
 | 	if (path == NULL) | 
 | 		return FALSE; | 
 |  | 
 | 	gtk_tree_model_get_iter(model, &iter, path); | 
 | 	gtk_tree_model_get(model, &iter, COL_MENU, &menu, -1); | 
 |  | 
 | 	if (event->type == GDK_2BUTTON_PRESS) | 
 | 		toggle_sym_value(menu); | 
 |  | 
 | 	selected = menu; | 
 |  | 
 | 	if (menu->type == M_MENU) { | 
 | 		browsed = menu; | 
 | 		recreate_tree(); | 
 | 	} | 
 |  | 
 | 	gtk_tree_view_set_cursor(view, path, NULL, FALSE); | 
 | 	gtk_widget_grab_focus(tree2_w); | 
 |  | 
 | 	return FALSE; | 
 | } | 
 |  | 
 | /* Display the whole tree (single/split/full view) */ | 
 | static void _display_tree(GtkTreeStore *tree, struct menu *menu, | 
 | 			  GtkTreeIter *parent) | 
 | { | 
 | 	struct menu *child; | 
 | 	GtkTreeIter iter; | 
 |  | 
 | 	for (child = menu->list; child; child = child->next) { | 
 | 		/* | 
 | 		 * REVISIT: | 
 | 		 * menu_finalize() creates empty "if" entries. | 
 | 		 * Do not confuse gtk_tree_model_get(), which would otherwise | 
 | 		 * return "if" menu entry. | 
 | 		 */ | 
 | 		if (child->type == M_IF) | 
 | 			continue; | 
 |  | 
 | 		if ((view_mode == SPLIT_VIEW) | 
 | 		    && !(child->flags & MENU_ROOT) && (tree == tree1)) | 
 | 			continue; | 
 |  | 
 | 		if ((view_mode == SPLIT_VIEW) && (child->flags & MENU_ROOT) | 
 | 		    && (tree == tree2)) | 
 | 			continue; | 
 |  | 
 | 		gtk_tree_store_append(tree, &iter, parent); | 
 | 		set_node(tree, &iter, child); | 
 |  | 
 | 		if (view_mode != SINGLE_VIEW || child->type != M_MENU) | 
 | 			_display_tree(tree, child, &iter); | 
 | 	} | 
 | } | 
 |  | 
 | static void display_tree(GtkTreeStore *store, struct menu *menu) | 
 | { | 
 | 	_display_tree(store, menu, NULL); | 
 | } | 
 |  | 
 | /* Recreate the tree store starting at 'browsed' node */ | 
 | static void recreate_tree(void) | 
 | { | 
 | 	gtk_tree_store_clear(tree2); | 
 | 	display_tree(tree2, browsed); | 
 | 	gtk_tree_view_expand_all(GTK_TREE_VIEW(tree2_w)); | 
 | } | 
 |  | 
 | static void fixup_rootmenu(struct menu *menu) | 
 | { | 
 | 	struct menu *child; | 
 | 	static int menu_cnt = 0; | 
 |  | 
 | 	menu->flags |= MENU_ROOT; | 
 | 	for (child = menu->list; child; child = child->next) { | 
 | 		if (child->prompt && child->prompt->type == P_MENU) { | 
 | 			menu_cnt++; | 
 | 			fixup_rootmenu(child); | 
 | 			menu_cnt--; | 
 | 		} else if (!menu_cnt) | 
 | 			fixup_rootmenu(child); | 
 | 	} | 
 | } | 
 |  | 
 | /* Main Window Initialization */ | 
 | static void replace_button_icon(GtkWidget *widget, const char * const xpm[]) | 
 | { | 
 | 	GdkPixbuf *pixbuf; | 
 | 	GtkWidget *image; | 
 |  | 
 | 	pixbuf = gdk_pixbuf_new_from_xpm_data((const char **)xpm); | 
 | 	image = gtk_image_new_from_pixbuf(pixbuf); | 
 | 	g_object_unref(pixbuf); | 
 |  | 
 | 	gtk_widget_show(image); | 
 | 	gtk_tool_button_set_icon_widget(GTK_TOOL_BUTTON(widget), image); | 
 | } | 
 |  | 
 | static void init_main_window(const gchar *glade_file) | 
 | { | 
 | 	GtkBuilder *builder; | 
 | 	GtkWidget *widget; | 
 | 	GtkTextBuffer *txtbuf; | 
 |  | 
 | 	builder = gtk_builder_new_from_file(glade_file); | 
 | 	if (!builder) | 
 | 		g_error("GUI loading failed !\n"); | 
 |  | 
 | 	main_wnd = GTK_WIDGET(gtk_builder_get_object(builder, "window1")); | 
 | 	g_signal_connect(main_wnd, "destroy", | 
 | 			 G_CALLBACK(on_window1_destroy), NULL); | 
 | 	g_signal_connect(main_wnd, "configure-event", | 
 | 			 G_CALLBACK(on_window1_configure), NULL); | 
 | 	g_signal_connect(main_wnd, "delete-event", | 
 | 			 G_CALLBACK(on_window1_delete_event), NULL); | 
 |  | 
 | 	hpaned = GTK_WIDGET(gtk_builder_get_object(builder, "hpaned1")); | 
 | 	vpaned = GTK_WIDGET(gtk_builder_get_object(builder, "vpaned1")); | 
 | 	tree1_w = GTK_WIDGET(gtk_builder_get_object(builder, "treeview1")); | 
 | 	g_signal_connect(tree1_w, "cursor-changed", | 
 | 			 G_CALLBACK(on_treeview2_cursor_changed), NULL); | 
 | 	g_signal_connect(tree1_w, "button-press-event", | 
 | 			 G_CALLBACK(on_treeview1_button_press_event), NULL); | 
 | 	g_signal_connect(tree1_w, "key-press-event", | 
 | 			 G_CALLBACK(on_treeview2_key_press_event), NULL); | 
 |  | 
 | 	tree2_w = GTK_WIDGET(gtk_builder_get_object(builder, "treeview2")); | 
 | 	g_signal_connect(tree2_w, "cursor-changed", | 
 | 			 G_CALLBACK(on_treeview2_cursor_changed), NULL); | 
 | 	g_signal_connect(tree2_w, "button-press-event", | 
 | 			 G_CALLBACK(on_treeview2_button_press_event), NULL); | 
 | 	g_signal_connect(tree2_w, "key-press-event", | 
 | 			 G_CALLBACK(on_treeview2_key_press_event), NULL); | 
 |  | 
 | 	text_w = GTK_WIDGET(gtk_builder_get_object(builder, "textview3")); | 
 |  | 
 | 	/* menubar */ | 
 | 	widget = GTK_WIDGET(gtk_builder_get_object(builder, "load1")); | 
 | 	g_signal_connect(widget, "activate", | 
 | 			 G_CALLBACK(on_load1_activate), NULL); | 
 |  | 
 | 	save_menu_item = GTK_WIDGET(gtk_builder_get_object(builder, "save1")); | 
 | 	g_signal_connect(save_menu_item, "activate", | 
 | 			 G_CALLBACK(on_save_activate), NULL); | 
 |  | 
 | 	widget = GTK_WIDGET(gtk_builder_get_object(builder, "save_as1")); | 
 | 	g_signal_connect(widget, "activate", | 
 | 			 G_CALLBACK(on_save_as1_activate), NULL); | 
 |  | 
 | 	widget = GTK_WIDGET(gtk_builder_get_object(builder, "quit1")); | 
 | 	g_signal_connect(widget, "activate", | 
 | 			 G_CALLBACK(on_quit1_activate), NULL); | 
 |  | 
 | 	widget = GTK_WIDGET(gtk_builder_get_object(builder, "show_name1")); | 
 | 	g_signal_connect(widget, "activate", | 
 | 			 G_CALLBACK(on_show_name1_activate), NULL); | 
 | 	gtk_check_menu_item_set_active((GtkCheckMenuItem *) widget, | 
 | 				       show_name); | 
 |  | 
 | 	widget = GTK_WIDGET(gtk_builder_get_object(builder, "show_range1")); | 
 | 	g_signal_connect(widget, "activate", | 
 | 			 G_CALLBACK(on_show_range1_activate), NULL); | 
 | 	gtk_check_menu_item_set_active((GtkCheckMenuItem *) widget, | 
 | 				       show_range); | 
 |  | 
 | 	widget = GTK_WIDGET(gtk_builder_get_object(builder, "show_data1")); | 
 | 	g_signal_connect(widget, "activate", | 
 | 			 G_CALLBACK(on_show_data1_activate), NULL); | 
 | 	gtk_check_menu_item_set_active((GtkCheckMenuItem *) widget, | 
 | 				       show_value); | 
 |  | 
 | 	widget = GTK_WIDGET(gtk_builder_get_object(builder, "set_option_mode1")); | 
 | 	g_signal_connect(widget, "activate", | 
 | 			 G_CALLBACK(on_set_option_mode1_activate), NULL); | 
 |  | 
 | 	widget = GTK_WIDGET(gtk_builder_get_object(builder, "set_option_mode2")); | 
 | 	g_signal_connect(widget, "activate", | 
 | 			 G_CALLBACK(on_set_option_mode2_activate), NULL); | 
 |  | 
 | 	widget = GTK_WIDGET(gtk_builder_get_object(builder, "set_option_mode3")); | 
 | 	g_signal_connect(widget, "activate", | 
 | 			 G_CALLBACK(on_set_option_mode3_activate), NULL); | 
 |  | 
 | 	widget = GTK_WIDGET(gtk_builder_get_object(builder, "introduction1")); | 
 | 	g_signal_connect(widget, "activate", | 
 | 			 G_CALLBACK(on_introduction1_activate), NULL); | 
 |  | 
 | 	widget = GTK_WIDGET(gtk_builder_get_object(builder, "about1")); | 
 | 	g_signal_connect(widget, "activate", | 
 | 			 G_CALLBACK(on_about1_activate), NULL); | 
 |  | 
 | 	widget = GTK_WIDGET(gtk_builder_get_object(builder, "license1")); | 
 | 	g_signal_connect(widget, "activate", | 
 | 			 G_CALLBACK(on_license1_activate), NULL); | 
 |  | 
 | 	/* toolbar */ | 
 | 	back_btn = GTK_WIDGET(gtk_builder_get_object(builder, "button1")); | 
 | 	g_signal_connect(back_btn, "clicked", | 
 | 			 G_CALLBACK(on_back_clicked), NULL); | 
 | 	gtk_widget_set_sensitive(back_btn, FALSE); | 
 |  | 
 | 	widget = GTK_WIDGET(gtk_builder_get_object(builder, "button2")); | 
 | 	g_signal_connect(widget, "clicked", | 
 | 			 G_CALLBACK(on_load_clicked), NULL); | 
 |  | 
 | 	save_btn = GTK_WIDGET(gtk_builder_get_object(builder, "button3")); | 
 | 	g_signal_connect(save_btn, "clicked", | 
 | 			 G_CALLBACK(on_save_clicked), NULL); | 
 |  | 
 | 	single_btn = GTK_WIDGET(gtk_builder_get_object(builder, "button4")); | 
 | 	g_signal_connect(single_btn, "clicked", | 
 | 			 G_CALLBACK(on_single_clicked), NULL); | 
 | 	replace_button_icon(single_btn, xpm_single_view); | 
 |  | 
 | 	split_btn = GTK_WIDGET(gtk_builder_get_object(builder, "button5")); | 
 | 	g_signal_connect(split_btn, "clicked", | 
 | 			 G_CALLBACK(on_split_clicked), NULL); | 
 | 	replace_button_icon(split_btn, xpm_split_view); | 
 |  | 
 | 	full_btn = GTK_WIDGET(gtk_builder_get_object(builder, "button6")); | 
 | 	g_signal_connect(full_btn, "clicked", | 
 | 			 G_CALLBACK(on_full_clicked), NULL); | 
 | 	replace_button_icon(full_btn, xpm_tree_view); | 
 |  | 
 | 	widget = GTK_WIDGET(gtk_builder_get_object(builder, "button7")); | 
 | 	g_signal_connect(widget, "clicked", | 
 | 			 G_CALLBACK(on_collapse_clicked), NULL); | 
 |  | 
 | 	widget = GTK_WIDGET(gtk_builder_get_object(builder, "button8")); | 
 | 	g_signal_connect(widget, "clicked", | 
 | 			 G_CALLBACK(on_expand_clicked), NULL); | 
 |  | 
 | 	txtbuf = gtk_text_view_get_buffer(GTK_TEXT_VIEW(text_w)); | 
 | 	tag1 = gtk_text_buffer_create_tag(txtbuf, "mytag1", | 
 | 					  "foreground", "red", | 
 | 					  "weight", PANGO_WEIGHT_BOLD, | 
 | 					  NULL); | 
 | 	tag2 = gtk_text_buffer_create_tag(txtbuf, "mytag2", | 
 | 					  /*"style", PANGO_STYLE_OBLIQUE, */ | 
 | 					  NULL); | 
 |  | 
 | 	gtk_window_set_title(GTK_WINDOW(main_wnd), rootmenu.prompt->text); | 
 |  | 
 | 	gtk_widget_show_all(main_wnd); | 
 |  | 
 | 	g_object_unref(builder); | 
 |  | 
 | 	conf_set_changed_callback(conf_changed); | 
 | } | 
 |  | 
 | static gboolean visible_func(GtkTreeModel *model, GtkTreeIter  *iter, | 
 | 			     gpointer data) | 
 | { | 
 | 	struct menu *menu; | 
 |  | 
 | 	gtk_tree_model_get(model, iter, COL_MENU, &menu, -1); | 
 |  | 
 | 	if (!menu) | 
 | 		return FALSE; | 
 |  | 
 | 	return menu_is_visible(menu) || opt_mode == OPT_ALL || | 
 | 		(opt_mode == OPT_PROMPT && menu_has_prompt(menu)); | 
 | } | 
 |  | 
 | static void init_left_tree(void) | 
 | { | 
 | 	GtkTreeView *view = GTK_TREE_VIEW(tree1_w); | 
 | 	GtkCellRenderer *renderer; | 
 | 	GtkTreeSelection *sel; | 
 | 	GtkTreeViewColumn *column; | 
 | 	GtkTreeModel *filter; | 
 |  | 
 | 	tree1 = gtk_tree_store_new(COL_NUMBER, | 
 | 				   G_TYPE_STRING, G_TYPE_STRING, | 
 | 				   G_TYPE_STRING, G_TYPE_STRING, | 
 | 				   G_TYPE_STRING, G_TYPE_STRING, | 
 | 				   G_TYPE_POINTER, GDK_TYPE_RGBA, | 
 | 				   G_TYPE_BOOLEAN, GDK_TYPE_PIXBUF, | 
 | 				   G_TYPE_BOOLEAN, G_TYPE_BOOLEAN, | 
 | 				   G_TYPE_BOOLEAN, G_TYPE_BOOLEAN, | 
 | 				   G_TYPE_BOOLEAN); | 
 |  | 
 | 	filter = gtk_tree_model_filter_new(GTK_TREE_MODEL(tree1), NULL); | 
 |  | 
 | 	gtk_tree_model_filter_set_visible_func(GTK_TREE_MODEL_FILTER(filter), | 
 | 					       visible_func, NULL, NULL); | 
 | 	gtk_tree_view_set_model(view, filter); | 
 |  | 
 | 	column = gtk_tree_view_column_new(); | 
 | 	gtk_tree_view_append_column(view, column); | 
 | 	gtk_tree_view_column_set_title(column, "Options"); | 
 |  | 
 | 	renderer = gtk_cell_renderer_toggle_new(); | 
 | 	gtk_tree_view_column_pack_start(GTK_TREE_VIEW_COLUMN(column), | 
 | 					renderer, FALSE); | 
 | 	gtk_tree_view_column_set_attributes(GTK_TREE_VIEW_COLUMN(column), | 
 | 					    renderer, | 
 | 					    "active", COL_BTNACT, | 
 | 					    "inconsistent", COL_BTNINC, | 
 | 					    "visible", COL_BTNVIS, | 
 | 					    "radio", COL_BTNRAD, NULL); | 
 | 	renderer = gtk_cell_renderer_text_new(); | 
 | 	gtk_tree_view_column_pack_start(GTK_TREE_VIEW_COLUMN(column), | 
 | 					renderer, FALSE); | 
 | 	gtk_tree_view_column_set_attributes(GTK_TREE_VIEW_COLUMN(column), | 
 | 					    renderer, | 
 | 					    "text", COL_OPTION, | 
 | 					    "foreground-rgba", | 
 | 					    COL_COLOR, NULL); | 
 |  | 
 | 	sel = gtk_tree_view_get_selection(view); | 
 | 	gtk_tree_selection_set_mode(sel, GTK_SELECTION_SINGLE); | 
 | } | 
 |  | 
 | static void init_right_tree(void) | 
 | { | 
 | 	GtkTreeView *view = GTK_TREE_VIEW(tree2_w); | 
 | 	GtkCellRenderer *renderer; | 
 | 	GtkTreeSelection *sel; | 
 | 	GtkTreeViewColumn *column; | 
 | 	GtkTreeModel *filter; | 
 | 	gint i; | 
 |  | 
 | 	tree2 = gtk_tree_store_new(COL_NUMBER, | 
 | 				   G_TYPE_STRING, G_TYPE_STRING, | 
 | 				   G_TYPE_STRING, G_TYPE_STRING, | 
 | 				   G_TYPE_STRING, G_TYPE_STRING, | 
 | 				   G_TYPE_POINTER, GDK_TYPE_RGBA, | 
 | 				   G_TYPE_BOOLEAN, GDK_TYPE_PIXBUF, | 
 | 				   G_TYPE_BOOLEAN, G_TYPE_BOOLEAN, | 
 | 				   G_TYPE_BOOLEAN, G_TYPE_BOOLEAN, | 
 | 				   G_TYPE_BOOLEAN); | 
 |  | 
 | 	filter = gtk_tree_model_filter_new(GTK_TREE_MODEL(tree2), NULL); | 
 |  | 
 | 	gtk_tree_model_filter_set_visible_func(GTK_TREE_MODEL_FILTER(filter), | 
 | 					       visible_func, NULL, NULL); | 
 | 	gtk_tree_view_set_model(view, filter); | 
 |  | 
 | 	column = gtk_tree_view_column_new(); | 
 | 	gtk_tree_view_append_column(view, column); | 
 | 	gtk_tree_view_column_set_title(column, "Options"); | 
 |  | 
 | 	renderer = gtk_cell_renderer_pixbuf_new(); | 
 | 	gtk_tree_view_column_pack_start(GTK_TREE_VIEW_COLUMN(column), | 
 | 					renderer, FALSE); | 
 | 	gtk_tree_view_column_set_attributes(GTK_TREE_VIEW_COLUMN(column), | 
 | 					    renderer, | 
 | 					    "pixbuf", COL_PIXBUF, | 
 | 					    "visible", COL_PIXVIS, NULL); | 
 | 	renderer = gtk_cell_renderer_toggle_new(); | 
 | 	gtk_tree_view_column_pack_start(GTK_TREE_VIEW_COLUMN(column), | 
 | 					renderer, FALSE); | 
 | 	gtk_tree_view_column_set_attributes(GTK_TREE_VIEW_COLUMN(column), | 
 | 					    renderer, | 
 | 					    "active", COL_BTNACT, | 
 | 					    "inconsistent", COL_BTNINC, | 
 | 					    "visible", COL_BTNVIS, | 
 | 					    "radio", COL_BTNRAD, NULL); | 
 | 	renderer = gtk_cell_renderer_text_new(); | 
 | 	gtk_tree_view_column_pack_start(GTK_TREE_VIEW_COLUMN(column), | 
 | 					renderer, FALSE); | 
 | 	gtk_tree_view_column_set_attributes(GTK_TREE_VIEW_COLUMN(column), | 
 | 					    renderer, | 
 | 					    "text", COL_OPTION, | 
 | 					    "foreground-rgba", | 
 | 					    COL_COLOR, NULL); | 
 |  | 
 | 	renderer = gtk_cell_renderer_text_new(); | 
 | 	gtk_tree_view_insert_column_with_attributes(view, -1, | 
 | 						    "Name", renderer, | 
 | 						    "text", COL_NAME, | 
 | 						    "foreground-rgba", | 
 | 						    COL_COLOR, NULL); | 
 | 	renderer = gtk_cell_renderer_text_new(); | 
 | 	gtk_tree_view_insert_column_with_attributes(view, -1, | 
 | 						    "N", renderer, | 
 | 						    "text", COL_NO, | 
 | 						    "foreground-rgba", | 
 | 						    COL_COLOR, NULL); | 
 | 	renderer = gtk_cell_renderer_text_new(); | 
 | 	gtk_tree_view_insert_column_with_attributes(view, -1, | 
 | 						    "M", renderer, | 
 | 						    "text", COL_MOD, | 
 | 						    "foreground-rgba", | 
 | 						    COL_COLOR, NULL); | 
 | 	renderer = gtk_cell_renderer_text_new(); | 
 | 	gtk_tree_view_insert_column_with_attributes(view, -1, | 
 | 						    "Y", renderer, | 
 | 						    "text", COL_YES, | 
 | 						    "foreground-rgba", | 
 | 						    COL_COLOR, NULL); | 
 | 	renderer = gtk_cell_renderer_text_new(); | 
 | 	gtk_tree_view_insert_column_with_attributes(view, -1, | 
 | 						    "Value", renderer, | 
 | 						    "text", COL_VALUE, | 
 | 						    "editable", | 
 | 						    COL_EDIT, | 
 | 						    "foreground-rgba", | 
 | 						    COL_COLOR, NULL); | 
 | 	g_signal_connect(G_OBJECT(renderer), "edited", | 
 | 			 G_CALLBACK(renderer_edited), tree2_w); | 
 |  | 
 | 	pix_menu = gdk_pixbuf_new_from_xpm_data((const char **)xpm_menu); | 
 |  | 
 | 	for (i = 0; i < COL_VALUE; i++) { | 
 | 		column = gtk_tree_view_get_column(view, i); | 
 | 		gtk_tree_view_column_set_resizable(column, TRUE); | 
 | 	} | 
 |  | 
 | 	sel = gtk_tree_view_get_selection(view); | 
 | 	gtk_tree_selection_set_mode(sel, GTK_SELECTION_SINGLE); | 
 | } | 
 |  | 
 | /* Main */ | 
 | int main(int ac, char *av[]) | 
 | { | 
 | 	const char *name; | 
 | 	char *env; | 
 | 	gchar *glade_file; | 
 |  | 
 | 	/* GTK stuffs */ | 
 | 	gtk_init(&ac, &av); | 
 |  | 
 | 	/* Determine GUI path */ | 
 | 	env = getenv(SRCTREE); | 
 | 	if (env) | 
 | 		glade_file = g_strconcat(env, "/scripts/kconfig/gconf.ui", NULL); | 
 | 	else if (av[0][0] == '/') | 
 | 		glade_file = g_strconcat(av[0], ".ui", NULL); | 
 | 	else | 
 | 		glade_file = g_strconcat(g_get_current_dir(), "/", av[0], ".ui", NULL); | 
 |  | 
 | 	/* Conf stuffs */ | 
 | 	if (ac > 1 && av[1][0] == '-') { | 
 | 		switch (av[1][1]) { | 
 | 		case 'a': | 
 | 			//showAll = 1; | 
 | 			break; | 
 | 		case 's': | 
 | 			conf_set_message_callback(NULL); | 
 | 			break; | 
 | 		case 'h': | 
 | 		case '?': | 
 | 			printf("%s [-s] <config>\n", av[0]); | 
 | 			exit(0); | 
 | 		} | 
 | 		name = av[2]; | 
 | 	} else | 
 | 		name = av[1]; | 
 |  | 
 | 	conf_parse(name); | 
 | 	fixup_rootmenu(&rootmenu); | 
 |  | 
 | 	/* Load the interface and connect signals */ | 
 | 	init_main_window(glade_file); | 
 | 	init_left_tree(); | 
 | 	init_right_tree(); | 
 |  | 
 | 	conf_read(NULL); | 
 |  | 
 | 	set_view_mode(view_mode); | 
 |  | 
 | 	gtk_main(); | 
 |  | 
 | 	return 0; | 
 | } |