medit/moo/mooedit/mooeditprefspage.c

1200 lines
34 KiB
C

/*
* mooeditprefspage.c
*
* Copyright (C) 2004-2007 by Yevgen Muntyan <muntyan@math.tamu.edu>
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License version 2.1 as published by the Free Software Foundation.
*
* See COPYING file that comes with this distribution.
*/
#define MOOEDIT_COMPILATION
#include "mooedit/mooedit-private.h"
#include "mooedit/mooeditor-private.h"
#include "mooedit/mooeditprefs.h"
#include "mooedit/mooeditprefs-general-glade.h"
#include "mooedit/mooeditprefs-view-glade.h"
#include "mooedit/mooeditprefs-file-glade.h"
#include "mooedit/mooeditprefs-langs-glade.h"
#include "mooedit/moolangmgr.h"
#include "mooedit/mooeditfiltersettings.h"
#include "mooutils/mooprefsdialog.h"
#include "mooutils/moostock.h"
#include "mooutils/mooglade.h"
#include "mooutils/moofontsel.h"
#include "mooutils/mooutils-treeview.h"
#include "mooutils/mooutils-misc.h"
#include "mooutils/mooencodings.h"
#include "mooutils/mooi18n.h"
#include "mooutils/moohelp.h"
#include "help-sections.h"
#include <string.h>
typedef struct PrefsPage PrefsPage;
static void moo_edit_prefs_page_init (MooPrefsDialogPage *page);
static void moo_edit_prefs_page_apply (MooPrefsDialogPage *page);
static gboolean moo_edit_prefs_page_help (GtkWidget *widget);
static void prefs_page_apply_lang_prefs (PrefsPage *page);
static void apply_filter_settings (PrefsPage *page);
static void scheme_combo_init (GtkComboBox *combo,
MooEditor *editor);
static void scheme_combo_data_func (GtkCellLayout *layout,
GtkCellRenderer *cell,
GtkTreeModel *model,
GtkTreeIter *iter);
static void scheme_combo_set_scheme (GtkComboBox *combo,
MooTextStyleScheme *scheme);
static void default_lang_combo_init (GtkComboBox *combo,
PrefsPage *page);
static void default_lang_combo_set_lang (GtkComboBox *combo,
const char *id);
static void lang_combo_init (GtkComboBox *combo,
PrefsPage *page);
static void filter_treeview_init (MooGladeXML *xml);
static GtkTreeModel *create_lang_model (MooEditor *editor);
static void save_encoding_combo_init (MooGladeXML *xml);
static void save_encoding_combo_apply (MooGladeXML *xml);
static GtkTreeModel *page_get_lang_model (PrefsPage *page);
static MooTextStyleScheme *page_get_scheme (PrefsPage *page);
static char *page_get_default_lang (PrefsPage *page);
static void page_view_init_xml (MooGladeXML *xml);
static void page_general_init (PrefsPage *page);
static void page_general_apply (PrefsPage *page);
static void page_view_init (PrefsPage *page);
static void page_view_apply (PrefsPage *page);
static void page_file_init (PrefsPage *page);
static void page_file_apply (PrefsPage *page);
static void page_langs_init (PrefsPage *page);
static void page_langs_apply (PrefsPage *page);
typedef struct {
const char *label;
const char *ui;
void (*init_xml) (MooGladeXML *xml);
void (*init) (PrefsPage *page);
void (*apply) (PrefsPage *page);
} PrefsPageInfo;
struct PrefsPage {
MooEditor *editor;
MooLangMgr *lang_mgr;
MooPrefsDialogPage *master;
MooPrefsDialogPage *page;
gboolean initialized;
void (*init) (PrefsPage *page);
void (*apply) (PrefsPage *page);
};
typedef struct {
MooEditor *editor;
MooLangMgr *lang_mgr;
PrefsPage **pages;
guint n_pages;
GtkNotebook *notebook;
} PrefsPageData;
static void
prefs_page_data_free (PrefsPageData *data)
{
guint i;
for (i = 0; i < data->n_pages; ++i)
{
PrefsPage *page = data->pages[i];
g_object_unref (page->editor);
g_object_unref (page->lang_mgr);
g_free (page);
}
g_object_unref (data->editor);
g_object_unref (data->lang_mgr);
g_free (data->pages);
g_free (data);
}
static void
init_page (PrefsPageData *data,
guint page_num)
{
PrefsPage *page;
g_return_if_fail (page_num < data->n_pages);
page = data->pages[page_num];
if (!page->initialized)
{
page->initialized = TRUE;
g_signal_emit_by_name (page->page, "init");
if (page->init)
page->init (page);
}
}
static void
notebook_switch_page (G_GNUC_UNUSED GtkNotebook *notebook,
G_GNUC_UNUSED GtkNotebookPage *nb_page,
guint page_num,
MooPrefsDialogPage *prefs_page)
{
PrefsPageData *data;
data = g_object_get_data (G_OBJECT (prefs_page), "prefs-page-data");
g_return_if_fail (data != NULL);
init_page (data, page_num);
}
static void
moo_edit_prefs_page_destroy (GtkWidget *page)
{
PrefsPageData *data;
data = g_object_get_data (G_OBJECT (page), "prefs-page-data");
if (data)
{
g_signal_handlers_disconnect_by_func (data->notebook, (gpointer) notebook_switch_page, page);
g_object_set_data (G_OBJECT (page), "prefs-page-data", NULL);
}
}
GtkWidget *
moo_edit_prefs_page_new (MooEditor *editor)
{
GtkWidget *prefs_page;
PrefsPageData *data;
guint i;
const PrefsPageInfo prefs_pages[] = {
{ N_("General"), mooeditprefs_general_glade_xml, NULL, page_general_init, page_general_apply },
{ N_("View"), mooeditprefs_view_glade_xml, page_view_init_xml, page_view_init, page_view_apply },
{ N_("Loading and saving"), mooeditprefs_file_glade_xml, NULL, page_file_init, page_file_apply },
{ N_("Languages and files"), mooeditprefs_langs_glade_xml, NULL, page_langs_init, page_langs_apply }
};
g_return_val_if_fail (MOO_IS_EDITOR (editor), NULL);
_moo_edit_init_config ();
prefs_page = moo_prefs_dialog_page_new ("Editor", GTK_STOCK_EDIT);
moo_help_set_func (prefs_page, moo_edit_prefs_page_help);
data = g_new0 (PrefsPageData, 1);
data->pages = g_new0 (PrefsPage*, G_N_ELEMENTS (prefs_pages));
g_object_set_data_full (G_OBJECT (prefs_page), "prefs-page-data", data,
(GDestroyNotify) prefs_page_data_free);
data->notebook = GTK_NOTEBOOK (gtk_notebook_new ());
gtk_widget_show (GTK_WIDGET (data->notebook));
gtk_container_add (GTK_CONTAINER (prefs_page), GTK_WIDGET (data->notebook));
data->editor = g_object_ref (editor);
data->lang_mgr = g_object_ref (moo_editor_get_lang_mgr (editor));
for (i = 0; i < G_N_ELEMENTS (prefs_pages); i++)
{
PrefsPage *page;
const PrefsPageInfo *info;
MooGladeXML *xml;
info = &prefs_pages[i];
page = g_new0 (PrefsPage, 1);
page->init = info->init;
page->apply = info->apply;
page->master = MOO_PREFS_DIALOG_PAGE (prefs_page);
data->pages[data->n_pages] = page;
xml = moo_glade_xml_new_empty (GETTEXT_PACKAGE);
if (info->init_xml)
info->init_xml (xml);
page->page = moo_prefs_dialog_page_new_from_xml (NULL, NULL,
xml,
info->ui,
"page",
MOO_EDIT_PREFS_PREFIX);
g_object_unref (xml);
gtk_notebook_append_page (data->notebook,
GTK_WIDGET (page->page),
gtk_label_new (_(info->label)));
page->editor = g_object_ref (editor);
page->lang_mgr = g_object_ref (moo_editor_get_lang_mgr (editor));
data->n_pages++;
}
g_signal_connect (data->notebook, "switch-page", G_CALLBACK (notebook_switch_page), prefs_page);
g_signal_connect (prefs_page, "init", G_CALLBACK (moo_edit_prefs_page_init), NULL);
g_signal_connect (prefs_page, "apply", G_CALLBACK (moo_edit_prefs_page_apply), NULL);
g_signal_connect (prefs_page, "destroy", G_CALLBACK (moo_edit_prefs_page_destroy), NULL);
return prefs_page;
}
static void
page_view_init_xml (MooGladeXML *xml)
{
moo_glade_xml_map_id (xml, "fontbutton", MOO_TYPE_FONT_BUTTON);
moo_glade_xml_set_property (xml, "fontbutton", "monospace", "True");
}
static void
page_general_init (PrefsPage *page)
{
GtkComboBox *default_lang_combo;
const char *lang;
default_lang_combo = moo_glade_xml_get_widget (page->page->xml, "default_lang_combo");
default_lang_combo_init (default_lang_combo, page);
lang = moo_prefs_get_string (moo_edit_setting (MOO_EDIT_PREFS_DEFAULT_LANG));
default_lang_combo_set_lang (default_lang_combo, lang);
}
static void
page_general_apply (PrefsPage *page)
{
char *lang;
lang = page_get_default_lang (page);
moo_prefs_set_string (moo_edit_setting (MOO_EDIT_PREFS_DEFAULT_LANG), lang);
g_free (lang);
}
static void
page_view_init (PrefsPage *page)
{
MooTextStyleScheme *scheme;
GtkComboBox *scheme_combo;
scheme = moo_lang_mgr_get_active_scheme (page->lang_mgr);
g_return_if_fail (scheme != NULL);
scheme_combo = moo_glade_xml_get_widget (page->page->xml, "color_scheme_combo");
scheme_combo_init (scheme_combo, page->editor);
scheme_combo_set_scheme (scheme_combo, scheme);
}
static void
page_view_apply (PrefsPage *page)
{
MooTextStyleScheme *scheme;
scheme = page_get_scheme (page);
g_return_if_fail (scheme != NULL);
moo_prefs_set_string (moo_edit_setting (MOO_EDIT_PREFS_COLOR_SCHEME),
moo_text_style_scheme_get_id (scheme));
g_object_unref (scheme);
}
static void
page_file_init (PrefsPage *page)
{
save_encoding_combo_init (page->page->xml);
}
static void
page_file_apply (PrefsPage *page)
{
save_encoding_combo_apply (page->page->xml);
}
static void
page_langs_init (PrefsPage *page)
{
GtkComboBox *lang_combo;
MooTreeHelper *helper;
moo_help_set_id (GTK_WIDGET (page->page), HELP_SECTION_PREFS_LANGS_AND_FILTERS);
lang_combo = moo_glade_xml_get_widget (page->page->xml, "lang_combo");
lang_combo_init (lang_combo, page);
filter_treeview_init (page->page->xml);
helper = g_object_get_data (G_OBJECT (page->page), "moo-tree-helper");
_moo_tree_helper_update_widgets (helper);
}
static void
page_langs_apply (PrefsPage *page)
{
prefs_page_apply_lang_prefs (page);
apply_filter_settings (page);
}
static void
scheme_combo_init (GtkComboBox *combo,
MooEditor *editor)
{
GtkListStore *store;
MooLangMgr *mgr;
GSList *list, *l;
GtkCellRenderer *cell;
mgr = moo_editor_get_lang_mgr (editor);
list = moo_lang_mgr_list_schemes (mgr);
g_return_if_fail (list != NULL);
store = gtk_list_store_new (1, MOO_TYPE_TEXT_STYLE_SCHEME);
for (l = list; l != NULL; l = l->next)
{
GtkTreeIter iter;
gtk_list_store_append (store, &iter);
gtk_list_store_set (store, &iter, 0, l->data, -1);
}
gtk_combo_box_set_model (combo, GTK_TREE_MODEL (store));
cell = gtk_cell_renderer_text_new ();
gtk_cell_layout_clear (GTK_CELL_LAYOUT (combo));
gtk_cell_layout_pack_start (GTK_CELL_LAYOUT (combo), cell, TRUE);
gtk_cell_layout_set_cell_data_func (GTK_CELL_LAYOUT (combo), cell,
(GtkCellLayoutDataFunc) scheme_combo_data_func,
NULL, NULL);
g_object_unref (store);
g_slist_foreach (list, (GFunc) g_object_unref, NULL);
g_slist_free (list);
}
static void
scheme_combo_data_func (G_GNUC_UNUSED GtkCellLayout *layout,
GtkCellRenderer *cell,
GtkTreeModel *model,
GtkTreeIter *iter)
{
MooTextStyleScheme *scheme = NULL;
gtk_tree_model_get (model, iter, 0, &scheme, -1);
g_return_if_fail (scheme != NULL);
g_object_set (cell, "text", moo_text_style_scheme_get_name (scheme), NULL);
g_object_unref (scheme);
}
static void
moo_edit_prefs_page_init (MooPrefsDialogPage *page)
{
PrefsPageData *data;
int current_page;
data = g_object_get_data (G_OBJECT (page), "prefs-page-data");
g_return_if_fail (data != NULL);
current_page = gtk_notebook_get_current_page (data->notebook);
init_page (data, current_page);
}
static GtkTreeModel *
page_get_lang_model (PrefsPage *page)
{
GtkTreeModel *model;
model = g_object_get_data (G_OBJECT (page->master), "moo-lang-model");
if (!model)
{
model = create_lang_model (page->editor);
g_object_set_data_full (G_OBJECT (page->master), "moo-lang-model",
model, g_object_unref);
}
return model;
}
static void
scheme_combo_set_scheme (GtkComboBox *combo,
MooTextStyleScheme *scheme)
{
GtkTreeModel *model = gtk_combo_box_get_model (combo);
gboolean found = FALSE;
GtkTreeIter iter;
gtk_tree_model_get_iter_first (model, &iter);
do {
MooTextStyleScheme *s;
gtk_tree_model_get (model, &iter, 0, &s, -1);
g_object_unref (s);
if (scheme == s)
{
found = TRUE;
break;
}
}
while (gtk_tree_model_iter_next (model, &iter));
g_return_if_fail (found);
gtk_combo_box_set_active_iter (combo, &iter);
}
static MooTextStyleScheme *
page_get_scheme (PrefsPage *page)
{
GtkTreeModel *model;
GtkTreeIter iter;
MooTextStyleScheme *scheme = NULL;
GtkComboBox *combo;
combo = moo_glade_xml_get_widget (page->page->xml, "color_scheme_combo");
g_return_val_if_fail (combo != NULL, NULL);
if (!gtk_combo_box_get_active_iter (combo, &iter))
{
g_critical ("%s: oops", G_STRLOC);
return NULL;
}
model = gtk_combo_box_get_model (combo);
gtk_tree_model_get (model, &iter, 0, &scheme, -1);
return scheme;
}
static void
moo_edit_prefs_page_apply (MooPrefsDialogPage *page)
{
PrefsPageData *data;
guint i;
data = g_object_get_data (G_OBJECT (page), "prefs-page-data");
g_return_if_fail (data != NULL);
for (i = 0; i < data->n_pages; ++i)
{
PrefsPage *prefs_page = data->pages[i];
if (prefs_page->initialized)
{
g_signal_emit_by_name (prefs_page->page, "apply");
if (prefs_page->apply)
prefs_page->apply (prefs_page);
}
}
moo_editor_apply_prefs (data->editor);
}
/*********************************************************************/
/* Language combo
*/
static void
fix_style (G_GNUC_UNUSED GtkWidget *combo)
{
#ifdef __WIN32__
static gboolean been_here = FALSE;
gtk_widget_set_name (combo, "moo-lang-combo");
if (!been_here)
{
been_here = TRUE;
gtk_rc_parse_string ("style \"moo-lang-combo\"\n"
"{\n"
" GtkComboBox::appears-as-list = 0\n"
"}\n"
"widget \"*.moo-lang-combo\" style \"moo-lang-combo\"\n");
}
#endif
}
enum {
COLUMN_ID,
COLUMN_NAME,
COLUMN_LANG,
COLUMN_EXTENSIONS,
COLUMN_MIMETYPES,
COLUMN_CONFIG
};
static char *
list_to_string (GSList *list,
gboolean free_list)
{
GSList *l;
GString *string = g_string_new (NULL);
for (l = list; l != NULL; l = l->next)
{
g_string_append (string, l->data);
if (l->next)
g_string_append_c (string, ';');
}
if (free_list)
{
g_slist_foreach (list, (GFunc) g_free, NULL);
g_slist_free (list);
}
return g_string_free (string, FALSE);
}
static int
lang_cmp (MooLang *lang1,
MooLang *lang2)
{
const char *name1, *name2;
name1 = _moo_lang_display_name (lang1);
name2 = _moo_lang_display_name (lang2);
return g_utf8_collate (name1, name2);
}
static GtkTreeModel *
create_lang_model (MooEditor *editor)
{
GtkTreeStore *store;
MooLangMgr *mgr;
GSList *langs, *sections, *l;
GtkTreeIter iter;
const char *config;
char *ext, *mime;
mgr = moo_editor_get_lang_mgr (editor);
langs = g_slist_sort (moo_lang_mgr_get_available_langs (mgr), (GCompareFunc) lang_cmp);
sections = moo_lang_mgr_get_sections (mgr);
store = gtk_tree_store_new (6, G_TYPE_STRING, G_TYPE_STRING, MOO_TYPE_LANG,
G_TYPE_STRING, G_TYPE_STRING, G_TYPE_STRING);
gtk_tree_store_append (store, &iter, NULL);
config = _moo_lang_mgr_get_config (mgr, MOO_LANG_NONE);
ext = list_to_string (_moo_lang_mgr_get_globs (mgr, NULL), TRUE);
mime = list_to_string (_moo_lang_mgr_get_mime_types (mgr, NULL), TRUE);
gtk_tree_store_set (store, &iter, COLUMN_ID, MOO_LANG_NONE,
COLUMN_NAME, "None",
COLUMN_CONFIG, config,
COLUMN_MIMETYPES, mime,
COLUMN_EXTENSIONS, ext,
-1);
g_free (ext);
g_free (mime);
while (sections)
{
char *section = sections->data;
gtk_tree_store_append (store, &iter, NULL);
gtk_tree_store_set (store, &iter, COLUMN_NAME, section, -1);
for (l = langs; l != NULL; l = l->next)
{
MooLang *lang = l->data;
if (!_moo_lang_get_hidden (lang) &&
!strcmp (_moo_lang_get_section (lang), section))
{
GtkTreeIter child;
ext = list_to_string (_moo_lang_mgr_get_globs (mgr, _moo_lang_id (lang)), TRUE);
mime = list_to_string (_moo_lang_mgr_get_mime_types (mgr, _moo_lang_id (lang)), TRUE);
config = _moo_lang_mgr_get_config (mgr, _moo_lang_id (lang));
gtk_tree_store_append (store, &child, &iter);
gtk_tree_store_set (store, &child,
COLUMN_ID, _moo_lang_id (lang),
COLUMN_NAME, _moo_lang_display_name (lang),
COLUMN_LANG, lang,
COLUMN_CONFIG, config,
COLUMN_MIMETYPES, mime,
COLUMN_EXTENSIONS, ext,
-1);
g_free (ext);
g_free (mime);
}
}
g_free (section);
sections = g_slist_delete_link (sections, sections);
}
g_slist_foreach (langs, (GFunc) g_object_unref, NULL);
g_slist_free (langs);
return GTK_TREE_MODEL (store);
}
#if 0
static gboolean
separator_func (GtkTreeModel *model,
GtkTreeIter *iter,
G_GNUC_UNUSED gpointer data)
{
char *name = NULL;
gtk_tree_model_get (model, iter, COLUMN_NAME, &name, -1);
if (!name)
return TRUE;
g_free (name);
return FALSE;
}
#endif
static void
set_sensitive (G_GNUC_UNUSED GtkCellLayout *cell_layout,
GtkCellRenderer *cell,
GtkTreeModel *model,
GtkTreeIter *iter,
G_GNUC_UNUSED gpointer data)
{
g_object_set (cell, "sensitive",
!gtk_tree_model_iter_has_child (model, iter),
NULL);
}
static void
default_lang_combo_init (GtkComboBox *combo,
PrefsPage *page)
{
GtkTreeModel *model;
GtkCellRenderer *cell;
fix_style (GTK_WIDGET (combo));
model = page_get_lang_model (page);
g_return_if_fail (model != NULL);
cell = gtk_cell_renderer_text_new ();
gtk_cell_layout_clear (GTK_CELL_LAYOUT (combo));
gtk_cell_layout_pack_start (GTK_CELL_LAYOUT (combo), cell, TRUE);
gtk_cell_layout_set_attributes (GTK_CELL_LAYOUT (combo), cell,
"text", COLUMN_NAME, NULL);
gtk_cell_layout_set_cell_data_func (GTK_CELL_LAYOUT (combo), cell,
set_sensitive, NULL, NULL);
gtk_combo_box_set_model (combo, model);
}
static gboolean
find_lang_by_id (GtkTreeModel *model,
G_GNUC_UNUSED GtkTreePath *path,
GtkTreeIter *iter,
gpointer user_data)
{
struct {
gboolean found;
GtkTreeIter iter;
const char *id;
} *data = user_data;
char *lang_id = NULL;
gtk_tree_model_get (model, iter, COLUMN_ID, &lang_id, -1);
if (_moo_str_equal(data->id, lang_id))
{
data->found = TRUE;
data->iter = *iter;
g_free (lang_id);
return TRUE;
}
g_free (lang_id);
return FALSE;
}
static void
default_lang_combo_set_lang (GtkComboBox *combo,
const char *id)
{
GtkTreeModel *model;
struct {
gboolean found;
GtkTreeIter iter;
const char *id;
} data;
g_return_if_fail (GTK_IS_COMBO_BOX (combo));
model = gtk_combo_box_get_model (combo);
data.found = FALSE;
data.id = id ? id : MOO_LANG_NONE;
gtk_tree_model_foreach (model,
(GtkTreeModelForeachFunc) find_lang_by_id,
&data);
g_return_if_fail (data.found);
gtk_combo_box_set_active_iter (combo, &data.iter);
}
static char *
page_get_default_lang (PrefsPage *page)
{
GtkTreeModel *model;
GtkTreeIter iter;
char *lang = NULL;
GtkComboBox *combo;
combo = moo_glade_xml_get_widget (page->page->xml, "default_lang_combo");
g_return_val_if_fail (combo != NULL, NULL);
if (!gtk_combo_box_get_active_iter (combo, &iter))
{
g_critical ("%s: oops", G_STRLOC);
return NULL;
}
model = gtk_combo_box_get_model (combo);
gtk_tree_model_get (model, &iter, COLUMN_ID, &lang, -1);
return lang;
}
/*********************************************************************/
/* Per-lang settings
*/
static void
helper_update_widgets (PrefsPage *page,
GtkTreeModel *model,
G_GNUC_UNUSED GtkTreePath *path,
GtkTreeIter *iter)
{
GtkEntry *extensions, *mimetypes, *config;
GtkWidget *label_mimetypes;
MooLang *lang = NULL;
char *ext = NULL, *mime = NULL, *id = NULL, *conf = NULL;
g_return_if_fail (iter != NULL);
extensions = moo_glade_xml_get_widget (page->page->xml, "extensions");
mimetypes = moo_glade_xml_get_widget (page->page->xml, "mimetypes");
label_mimetypes = moo_glade_xml_get_widget (page->page->xml, "label_mimetypes");
config = moo_glade_xml_get_widget (page->page->xml, "config");
gtk_tree_model_get (model, iter,
COLUMN_LANG, &lang,
COLUMN_MIMETYPES, &mime,
COLUMN_EXTENSIONS, &ext,
COLUMN_CONFIG, &conf,
COLUMN_ID, &id,
-1);
g_return_if_fail (id != NULL);
gtk_entry_set_text (extensions, ext ? ext : "");
gtk_entry_set_text (mimetypes, mime ? mime : "");
gtk_entry_set_text (config, conf ? conf : "");
gtk_widget_set_sensitive (GTK_WIDGET (mimetypes), lang != NULL);
gtk_widget_set_sensitive (label_mimetypes, lang != NULL);
if (lang)
g_object_unref (lang);
g_free (conf);
g_free (ext);
g_free (mime);
g_free (id);
}
static void
helper_update_model (PrefsPage *page,
GtkTreeModel *model,
G_GNUC_UNUSED GtkTreePath *path,
GtkTreeIter *iter)
{
GtkEntry *extensions, *mimetypes, *config;
const char *ext, *mime, *conf;
extensions = moo_glade_xml_get_widget (page->page->xml, "extensions");
ext = gtk_entry_get_text (extensions);
mimetypes = moo_glade_xml_get_widget (page->page->xml, "mimetypes");
mime = gtk_entry_get_text (mimetypes);
config = moo_glade_xml_get_widget (page->page->xml, "config");
conf = gtk_entry_get_text (config);
gtk_tree_store_set (GTK_TREE_STORE (model), iter,
COLUMN_MIMETYPES, mime,
COLUMN_EXTENSIONS, ext,
COLUMN_CONFIG, conf, -1);
}
static void
lang_combo_init (GtkComboBox *combo,
PrefsPage *page)
{
GtkTreeModel *model;
GtkCellRenderer *cell;
MooTreeHelper *helper;
fix_style (GTK_WIDGET (combo));
model = page_get_lang_model (page);
g_return_if_fail (model != NULL);
cell = gtk_cell_renderer_text_new ();
gtk_cell_layout_clear (GTK_CELL_LAYOUT (combo));
gtk_cell_layout_pack_start (GTK_CELL_LAYOUT (combo), cell, TRUE);
gtk_cell_layout_set_attributes (GTK_CELL_LAYOUT (combo), cell,
"text", COLUMN_NAME, NULL);
gtk_cell_layout_set_cell_data_func (GTK_CELL_LAYOUT (combo), cell,
set_sensitive, NULL, NULL);
gtk_combo_box_set_model (combo, model);
_moo_combo_box_select_first (combo);
helper = _moo_tree_helper_new (GTK_WIDGET (combo), NULL, NULL, NULL, NULL);
g_return_if_fail (helper != NULL);
g_object_set_data_full (G_OBJECT (page->page), "moo-tree-helper",
helper, g_object_unref);
g_signal_connect_swapped (helper, "update-widgets",
G_CALLBACK (helper_update_widgets), page);
g_signal_connect_swapped (helper, "update-model",
G_CALLBACK (helper_update_model), page);
_moo_tree_helper_update_widgets (helper);
}
static gboolean
apply_one_lang (GtkTreeModel *model,
G_GNUC_UNUSED GtkTreePath *path,
GtkTreeIter *iter,
MooLangMgr *mgr)
{
MooLang *lang = NULL;
char *config = NULL, *id = NULL;
char *mime, *ext;
gtk_tree_model_get (model, iter,
COLUMN_LANG, &lang,
COLUMN_ID, &id,
COLUMN_CONFIG, &config, -1);
if (!id)
return FALSE;
gtk_tree_model_get (model, iter,
COLUMN_MIMETYPES, &mime,
COLUMN_EXTENSIONS, &ext, -1);
_moo_lang_mgr_set_mime_types (mgr, id, mime);
_moo_lang_mgr_set_globs (mgr, id, ext);
_moo_lang_mgr_set_config (mgr, id, config);
if (lang)
g_object_unref (lang);
g_free (mime);
g_free (ext);
g_free (config);
g_free (id);
return FALSE;
}
static void
prefs_page_apply_lang_prefs (PrefsPage *page)
{
GtkTreeModel *model;
MooTreeHelper *helper;
MooLangMgr *mgr;
helper = g_object_get_data (G_OBJECT (page->page), "moo-tree-helper");
_moo_tree_helper_update_model (helper, NULL, NULL);
model = page_get_lang_model (page);
g_return_if_fail (model != NULL);
mgr = moo_editor_get_lang_mgr (page->editor);
gtk_tree_model_foreach (model, (GtkTreeModelForeachFunc) apply_one_lang, mgr);
_moo_lang_mgr_save_config (mgr);
_moo_edit_update_lang_config ();
}
/*********************************************************************/
/* Filters
*/
enum {
FILTER_COLUMN_FILTER,
FILTER_COLUMN_CONFIG,
FILTER_NUM_COLUMNS
};
static void
filter_store_set_modified (gpointer store,
gboolean modified)
{
g_return_if_fail (GTK_IS_LIST_STORE (store));
g_object_set_data (store, "filter-store-modified",
GINT_TO_POINTER (modified));
_moo_tree_helper_set_modified (g_object_get_data (store, "tree-helper"), FALSE);
}
static gboolean
filter_store_get_modified (gpointer store)
{
g_return_val_if_fail (GTK_IS_LIST_STORE (store), FALSE);
return g_object_get_data (store, "filter-store-modified") != NULL ||
_moo_tree_helper_get_modified (g_object_get_data (store, "tree-helper"));
}
static void
populate_filter_settings_store (GtkListStore *store)
{
GSList *strings, *l;
_moo_edit_filter_settings_load ();
l = strings = _moo_edit_filter_settings_get_strings ();
while (l)
{
GtkTreeIter iter;
g_return_if_fail (l->next != NULL);
gtk_list_store_append (store, &iter);
gtk_list_store_set (store, &iter,
FILTER_COLUMN_FILTER, l->data,
FILTER_COLUMN_CONFIG, l->next->data,
-1);
l = l->next->next;
}
g_slist_foreach (strings, (GFunc) g_free, NULL);
g_slist_free (strings);
}
static void
filter_cell_edited (GtkCellRendererText *cell,
const char *path_string,
const char *text,
GtkListStore *store)
{
GtkTreeIter iter;
GtkTreePath *path;
int column;
char *old_text;
g_return_if_fail (GTK_IS_LIST_STORE (store));
column = GPOINTER_TO_INT (g_object_get_data (G_OBJECT (cell), "filter-store-column-id"));
g_return_if_fail (column >= 0 && column < FILTER_NUM_COLUMNS);
path = gtk_tree_path_new_from_string (path_string);
if (!gtk_tree_model_get_iter (GTK_TREE_MODEL (store), &iter, path))
{
gtk_tree_path_free (path);
return;
}
gtk_tree_model_get (GTK_TREE_MODEL (store), &iter, column, &old_text, -1);
if (!_moo_str_equal (old_text, text))
{
gtk_list_store_set (store, &iter, column, text, -1);
filter_store_set_modified (store, TRUE);
}
g_free (old_text);
gtk_tree_path_free (path);
}
static void
create_filter_cell (GtkTreeView *treeview,
GtkListStore *store,
const char *title,
int column_id)
{
GtkTreeViewColumn *column;
GtkCellRenderer *cell;
cell = gtk_cell_renderer_text_new ();
column = gtk_tree_view_column_new_with_attributes (title, cell, "text", column_id, NULL);
gtk_tree_view_append_column (treeview, column);
g_object_set (cell, "editable", TRUE, NULL);
g_object_set_data (G_OBJECT (cell), "filter-store-column-id", GINT_TO_POINTER (column_id));
g_signal_connect (cell, "edited", G_CALLBACK (filter_cell_edited), store);
}
static void
filter_treeview_init (MooGladeXML *xml)
{
GtkTreeView *filter_treeview;
GtkListStore *store;
MooTreeHelper *helper;
filter_treeview = moo_glade_xml_get_widget (xml, "filter_treeview");
g_return_if_fail (filter_treeview != NULL);
store = gtk_list_store_new (FILTER_NUM_COLUMNS, G_TYPE_STRING, G_TYPE_STRING);
populate_filter_settings_store (store);
gtk_tree_view_set_model (filter_treeview, GTK_TREE_MODEL (store));
create_filter_cell (filter_treeview, store, "Filter", FILTER_COLUMN_FILTER);
create_filter_cell (filter_treeview, store, "Options", FILTER_COLUMN_CONFIG);
helper = _moo_tree_helper_new (GTK_WIDGET (filter_treeview),
moo_glade_xml_get_widget (xml, "new_filter_setting"),
moo_glade_xml_get_widget (xml, "delete_filter_setting"),
moo_glade_xml_get_widget (xml, "filter_setting_up"),
moo_glade_xml_get_widget (xml, "filter_setting_down"));
MOO_OBJECT_REF_SINK (helper);
g_object_set_data_full (G_OBJECT (filter_treeview), "tree-helper", helper, g_object_unref);
g_object_set_data (G_OBJECT (store), "tree-helper", helper);
g_object_unref (store);
}
static gboolean
prepend_filter_and_config (GtkTreeModel *model,
G_GNUC_UNUSED GtkTreePath *path,
GtkTreeIter *iter,
GSList **list)
{
char *filter = NULL, *config = NULL;
gtk_tree_model_get (model, iter,
FILTER_COLUMN_FILTER, &filter,
FILTER_COLUMN_CONFIG, &config,
-1);
*list = g_slist_prepend (*list, filter ? filter : g_strdup (""));
*list = g_slist_prepend (*list, config ? config : g_strdup (""));
return FALSE;
}
static void
apply_filter_settings (PrefsPage *page)
{
GtkTreeView *filter_treeview;
GSList *strings = NULL;
GtkTreeModel *model;
filter_treeview = moo_glade_xml_get_widget (page->page->xml, "filter_treeview");
g_return_if_fail (filter_treeview != NULL);
model = gtk_tree_view_get_model (filter_treeview);
if (!filter_store_get_modified (model))
return;
gtk_tree_model_foreach (model,
(GtkTreeModelForeachFunc) prepend_filter_and_config,
&strings);
strings = g_slist_reverse (strings);
_moo_edit_filter_settings_set_strings (strings);
filter_store_set_modified (model, FALSE);
g_slist_foreach (strings, (GFunc) g_free, NULL);
g_slist_free (strings);
}
/**************************************************************************/
/* Encoding combos
*/
static void
save_encoding_combo_init (MooGladeXML *xml)
{
GtkComboBox *combo;
const char *enc;
combo = moo_glade_xml_get_widget (xml, "encoding_save");
g_return_if_fail (combo != NULL);
_moo_encodings_combo_init (combo, MOO_ENCODING_COMBO_SAVE, TRUE);
enc = moo_prefs_get_string (moo_edit_setting (MOO_EDIT_PREFS_ENCODING_SAVE));
_moo_encodings_combo_set_enc (combo, enc, MOO_ENCODING_COMBO_SAVE);
}
static void
save_encoding_combo_apply (MooGladeXML *xml)
{
GtkComboBox *combo;
const char *enc;
combo = moo_glade_xml_get_widget (xml, "encoding_save");
g_return_if_fail (combo != NULL);
enc = _moo_encodings_combo_get_enc (combo, MOO_ENCODING_COMBO_SAVE);
moo_prefs_set_string (moo_edit_setting (MOO_EDIT_PREFS_ENCODING_SAVE), enc);
}
/**************************************************************************/
/* Help
*/
static gboolean
moo_edit_prefs_page_help (GtkWidget *widget)
{
PrefsPageData *data;
int index;
GtkWidget *current_page = NULL;
data = g_object_get_data (G_OBJECT (widget), "prefs-page-data");
g_return_val_if_fail (data != NULL, FALSE);
if ((index = gtk_notebook_get_current_page (data->notebook)) >= 0)
current_page = gtk_notebook_get_nth_page (data->notebook, index);
return current_page && moo_help_open (current_page);
}