linux/scripts/kconfig/gconf.c

1336 lines
35 KiB
C
Raw Permalink Normal View History

// 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 !");
kconfig: allow all config targets to write auto.conf if missing Currently, only syncconfig creates or updates include/config/auto.conf and some other files. Other config targets create or update only the .config file. When you configure and build the kernel from a pristine source tree, any config target is followed by syncconfig in the build stage since include/config/auto.conf is missing. We are moving compiler tests from Makefile to Kconfig. It means that parsing Kconfig files will be more costly since Kconfig invokes the compiler commands internally. Thus, we want to avoid invoking Kconfig twice (one for *config to create the .config, and one for syncconfig to synchronize the auto.conf). If auto.conf does not exist, we can generate all configuration files in the first configuration stage, which will save the syncconfig in the build stage. Please note this should be done only when auto.conf is missing. If *config blindly did this, time stamp files under include/config/ would be unnecessarily touched, triggering unneeded rebuild of objects. I assume a scenario like this: 1. You have a source tree that has already been built with CONFIG_FOO disabled 2. Run "make menuconfig" to enable CONFIG_FOO 3. CONFIG_FOO turns out to be unnecessary. Run "make menuconfig" again to disable CONFIG_FOO 4. Run "make" In this case, include/config/foo.h should not be touched since there is no change in CONFIG_FOO. The sync process should be delayed until the user really attempts to build the kernel. This commit has another motivation; I want to suppress the 'No such file or directory' warning from the 'include' directive. The top-level Makefile includes auto.conf with '-include' directive, like this: ifeq ($(dot-config),1) -include include/config/auto.conf endif This looks strange because auto.conf is mandatory when dot-config is 1. I guess only the reason of using '-include' is to suppress the warning 'include/config/auto.conf: No such file or directory' when building from a clean tree. However, this has a side-effect; Make considers the files included by '-include' are optional. Hence, Make continues to build even if it fails to generate include/config/auto.conf. I will change this in the next commit, but the warning message is annoying. (At least, kbuild test robot reports it as a regression.) With this commit, Kconfig will generate all configuration files together with the .config and I guess it is a solution good enough to suppress the warning. Note: GNU Make 4.2 or later does not display the warning from the 'include' directive if include files are successfully generated. See GNU Make commit 87a5f98d248f ("[SV 102] Don't show unnecessary include file errors.") However, older GNU Make versions are still widely used. Signed-off-by: Masahiro Yamada <yamada.masahiro@socionext.com>
2018-07-20 16:46:31 +09:00
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"
[PATCH] Kconfig i18n support This patch adds i18n support for make *config, allowing users to have the config process in their own language. No printk was harmed in the process, don't worry, so all the bug reports, kernel messages, etc, remain in english, just the user tools to configure the kernel are internationalized. Users not interested in translations can just unset the related LANG, LC_ALL, etc env variables and have the config process in plain english, something like: LANG= make menuconfig is enough for having the whole config process in english. Or just don't install any translation file. Translations for brazilian portuguese are being done by a team of volunteers at: http://www.visionflex.inf.br/kernel_ptbr/pmwiki.php/Principal/Traducoes To start the translation process: make update-po-config This will generate the pot template named scripts/kconfig/linux.pot, copy it to, say, ~/es.po, to start the translation for spanish. To test your translation, as root issue this command: msgfmt -o /usr/share/locale/es/LC_MESSAGES/linux.mo ~/es.po Replace "es" with your language code. Then execute, for instance: make menuconfig The current patch doesn't use any optimization to reduce the size of the generated .mo file, it is possible to use the config option as a key, but this doesn't prevent the current patch from being used or the translations done under the current scheme to be in any way lost if we chose to do any kind of keying. Thanks to Fabricio Vaccari for starting the pt_BR (brazilian portuguese) translation effort, Thiago Maciera for helping me with the gconf.cc (QT frontent) i18n coding and to all the volunteers that are already working on the first translation, to pt_BR. I left the question on whether to ship the translations with the stock kernel sources to be discussed here, please share your suggestions. Signed-off-by: Arnaldo Carvalho de Melo <acme@conectiva.com.br> Signed-off-by: Andrew Morton <akpm@osdl.org> Signed-off-by: Linus Torvalds <torvalds@osdl.org Signed-off-by: Andrew Morton <akpm@osdl.org>
2005-05-05 15:09:46 -07:00
"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;
}