| // 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; |
| } |