2006-12-17 03:23:19 -06:00
|
|
|
/*
|
|
|
|
* mooencodings.c
|
|
|
|
*
|
2007-04-07 03:21:52 -05:00
|
|
|
* Copyright (C) 2004-2007 by Yevgen Muntyan <muntyan@math.tamu.edu>
|
2006-12-17 03:23:19 -06:00
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
|
|
* (at your option) any later version.
|
|
|
|
*
|
|
|
|
* See COPYING file that comes with this distribution.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "mooutils/mooencodings.h"
|
|
|
|
#include "mooutils/mooi18n.h"
|
|
|
|
#include "mooutils/mooencodings-data.h"
|
2007-04-07 02:33:53 -05:00
|
|
|
#include "mooutils/mooprefs.h"
|
2006-12-17 03:23:19 -06:00
|
|
|
#include <gtk/gtk.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
|
|
|
#define MAX_RECENT_ENCODINGS 5
|
|
|
|
#define ROW_RECENT(save_mode) ((save_mode) ? 1 : 3)
|
|
|
|
#define ROW_LOCALE(save_mode) ((save_mode) ? 0 : 2)
|
|
|
|
#define ROW_AUTO 0
|
|
|
|
|
2007-04-07 02:33:53 -05:00
|
|
|
#define ELM_ENCODINGS "Editor/encodings"
|
|
|
|
#define ELM_RECENT "recent"
|
|
|
|
#define ELM_LAST_OPEN "last-open"
|
|
|
|
#define ELM_LAST_SAVE "last-save"
|
|
|
|
|
2006-12-17 03:23:19 -06:00
|
|
|
|
|
|
|
typedef struct {
|
|
|
|
const char *name;
|
|
|
|
const char *short_display_name;
|
|
|
|
char *display_name;
|
|
|
|
char *subgroup_name;
|
|
|
|
} Encoding;
|
|
|
|
|
|
|
|
typedef struct {
|
|
|
|
Encoding **encodings;
|
|
|
|
guint n_encodings;
|
|
|
|
char *name;
|
|
|
|
} EncodingGroup;
|
|
|
|
|
|
|
|
typedef struct {
|
|
|
|
GSList *recent;
|
|
|
|
EncodingGroup *groups;
|
|
|
|
guint n_groups;
|
|
|
|
Encoding *locale_encoding;
|
|
|
|
GSList *user_defined;
|
|
|
|
char *last_open;
|
|
|
|
char *last_save;
|
|
|
|
GHashTable *encodings;
|
|
|
|
} EncodingsManager;
|
|
|
|
|
|
|
|
enum {
|
|
|
|
COLUMN_DISPLAY,
|
|
|
|
COLUMN_ENCODING
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2007-04-07 02:33:53 -05:00
|
|
|
static void combo_changed (GtkComboBox *combo,
|
|
|
|
gpointer save_mode);
|
|
|
|
static void enc_mgr_load (EncodingsManager *enc_mgr);
|
2006-12-17 03:23:19 -06:00
|
|
|
|
|
|
|
|
|
|
|
static Encoding *
|
2006-12-17 10:58:17 -06:00
|
|
|
lookup_encoding (EncodingsManager *mgr,
|
|
|
|
const char *name)
|
2006-12-17 03:23:19 -06:00
|
|
|
{
|
|
|
|
char *upper;
|
2006-12-17 10:58:17 -06:00
|
|
|
Encoding *enc;
|
2006-12-17 03:23:19 -06:00
|
|
|
|
|
|
|
g_return_val_if_fail (name != NULL, NULL);
|
|
|
|
|
|
|
|
upper = g_ascii_strup (name, -1);
|
|
|
|
enc = g_hash_table_lookup (mgr->encodings, upper);
|
|
|
|
|
2006-12-17 10:58:17 -06:00
|
|
|
g_free (upper);
|
|
|
|
return enc;
|
|
|
|
}
|
|
|
|
|
|
|
|
static Encoding *
|
|
|
|
get_encoding (EncodingsManager *mgr,
|
|
|
|
const char *name)
|
|
|
|
{
|
|
|
|
Encoding *enc;
|
|
|
|
|
|
|
|
g_return_val_if_fail (name != NULL, NULL);
|
|
|
|
|
|
|
|
enc = lookup_encoding (mgr, name);
|
|
|
|
|
2006-12-17 03:23:19 -06:00
|
|
|
if (!enc)
|
|
|
|
{
|
|
|
|
enc = g_new0 (Encoding, 1);
|
2006-12-17 10:58:17 -06:00
|
|
|
enc->name = g_ascii_strup (name, -1);
|
2006-12-17 03:23:19 -06:00
|
|
|
enc->short_display_name = enc->name;
|
|
|
|
enc->display_name = g_strdup (name);
|
|
|
|
mgr->user_defined = g_slist_prepend (mgr->user_defined, enc);
|
|
|
|
g_hash_table_insert (mgr->encodings, (char*) enc->name, enc);
|
|
|
|
}
|
|
|
|
|
|
|
|
return enc;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
validate_encoding_name (const char *name)
|
|
|
|
{
|
|
|
|
return name && name[0];
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
compare_encodings (Encoding **enc1,
|
|
|
|
Encoding **enc2)
|
|
|
|
{
|
|
|
|
int result;
|
|
|
|
|
|
|
|
result = g_utf8_collate ((*enc1)->subgroup_name, (*enc2)->subgroup_name);
|
|
|
|
|
|
|
|
if (!result)
|
|
|
|
result = g_utf8_collate ((*enc1)->short_display_name, (*enc2)->short_display_name);
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
fill_encoding_group (EncodingGroup *group,
|
|
|
|
GSList *encodings)
|
|
|
|
{
|
|
|
|
guint i;
|
|
|
|
|
|
|
|
if (!encodings)
|
|
|
|
return;
|
|
|
|
|
|
|
|
group->n_encodings = g_slist_length (encodings);
|
|
|
|
group->encodings = g_new0 (Encoding*, group->n_encodings);
|
|
|
|
|
|
|
|
for (i = 0; encodings != NULL; encodings = encodings->next, ++i)
|
|
|
|
group->encodings[i] = encodings->data;
|
|
|
|
|
|
|
|
qsort (group->encodings, group->n_encodings, sizeof (Encoding*),
|
|
|
|
(int(*)(const void *, const void *)) compare_encodings);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
sort_encoding_groups (EncodingsManager *mgr)
|
|
|
|
{
|
2007-02-27 22:55:05 -06:00
|
|
|
/* Translators: 012345 denotes order in which character encoding
|
|
|
|
groups should appear, e.g. 543210 means inverse order.
|
|
|
|
0 - "West European",
|
|
|
|
1 - "East European",
|
|
|
|
2 - "East Asian",
|
|
|
|
3 - "SE & SW Asian",
|
|
|
|
4 - "Middle Eastern",
|
|
|
|
5 - "Unicode" */
|
2006-12-17 03:23:19 -06:00
|
|
|
const char *order = N_("012345");
|
|
|
|
const char *new_order_s;
|
|
|
|
int new_order[N_ENCODING_GROUPS] = {0, 0, 0, 0, 0, 0};
|
|
|
|
guint i;
|
|
|
|
EncodingGroup *new_array;
|
|
|
|
|
|
|
|
g_return_if_fail (strlen (order) == N_ENCODING_GROUPS);
|
|
|
|
g_return_if_fail (mgr->n_groups == N_ENCODING_GROUPS);
|
|
|
|
|
|
|
|
new_order_s = _(order);
|
|
|
|
|
|
|
|
if (!strcmp (order, new_order_s))
|
|
|
|
return;
|
|
|
|
|
|
|
|
for (i = 0; new_order_s[i]; ++i)
|
|
|
|
{
|
|
|
|
int n = new_order_s[i] - '0';
|
|
|
|
|
|
|
|
if (n < 0 || n >= (int) mgr->n_groups)
|
|
|
|
{
|
|
|
|
g_critical ("invalid order string %s", new_order_s);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
new_order[i] = n;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < mgr->n_groups; ++i)
|
|
|
|
{
|
|
|
|
if (!new_order[i])
|
|
|
|
{
|
|
|
|
g_critical ("invalid order string %s", new_order_s);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
new_array = g_new (EncodingGroup, mgr->n_groups);
|
|
|
|
|
|
|
|
for (i = 0; i < mgr->n_groups; ++i)
|
|
|
|
new_array[i] = mgr->groups[new_order[i]];
|
|
|
|
|
|
|
|
g_free (mgr->groups);
|
|
|
|
mgr->groups = new_array;
|
|
|
|
}
|
|
|
|
|
|
|
|
static EncodingsManager *
|
|
|
|
get_enc_mgr (void)
|
|
|
|
{
|
|
|
|
static EncodingsManager *mgr;
|
|
|
|
|
|
|
|
if (!mgr)
|
|
|
|
{
|
|
|
|
guint i;
|
|
|
|
const char *locale_charset;
|
|
|
|
GSList *enc_groups[N_ENCODING_GROUPS] = {NULL, NULL, NULL, NULL, NULL, NULL};
|
|
|
|
|
|
|
|
mgr = g_new0 (EncodingsManager, 1);
|
|
|
|
|
|
|
|
mgr->n_groups = N_ENCODING_GROUPS;
|
|
|
|
mgr->groups = g_new0 (EncodingGroup, N_ENCODING_GROUPS);
|
|
|
|
|
|
|
|
for (i = 0; i < mgr->n_groups; ++i)
|
|
|
|
mgr->groups[i].name = g_strdup (_(moo_encoding_groups_names[i]));
|
|
|
|
|
|
|
|
mgr->encodings = g_hash_table_new (g_str_hash, g_str_equal);
|
|
|
|
|
|
|
|
for (i = 0; i < G_N_ELEMENTS (moo_encodings_data); ++i)
|
|
|
|
{
|
|
|
|
Encoding *enc;
|
|
|
|
|
|
|
|
enc = g_new0 (Encoding, 1);
|
|
|
|
enc->name = moo_encodings_data[i].name;
|
|
|
|
enc->subgroup_name = g_strdup (_(moo_encodings_data[i].display_subgroup));
|
|
|
|
enc->short_display_name = moo_encodings_data[i].short_display_name;
|
|
|
|
enc->display_name = g_strdup_printf ("%s (%s)", enc->subgroup_name,
|
|
|
|
enc->short_display_name);
|
|
|
|
|
|
|
|
enc_groups[moo_encodings_data[i].group] =
|
|
|
|
g_slist_prepend (enc_groups[moo_encodings_data[i].group], enc);
|
|
|
|
g_hash_table_insert (mgr->encodings, (char*) moo_encodings_data[i].name, enc);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < G_N_ELEMENTS (moo_encoding_aliases); ++i)
|
|
|
|
{
|
|
|
|
Encoding *enc;
|
|
|
|
|
|
|
|
enc = g_hash_table_lookup (mgr->encodings, moo_encoding_aliases[i].name);
|
|
|
|
|
|
|
|
if (!enc)
|
2006-12-17 04:09:40 -06:00
|
|
|
g_critical ("%s: oops %s", G_STRLOC, moo_encoding_aliases[i].name);
|
2006-12-17 03:23:19 -06:00
|
|
|
else
|
|
|
|
g_hash_table_insert (mgr->encodings, (char*)
|
|
|
|
moo_encoding_aliases[i].alias,
|
|
|
|
enc);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < N_ENCODING_GROUPS; ++i)
|
|
|
|
{
|
|
|
|
fill_encoding_group (&mgr->groups[i], enc_groups[i]);
|
|
|
|
g_slist_free (enc_groups[i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
sort_encoding_groups (mgr);
|
|
|
|
|
|
|
|
if (!g_get_charset (&locale_charset))
|
|
|
|
locale_charset = MOO_ENCODING_UTF8;
|
|
|
|
|
2006-12-17 10:58:17 -06:00
|
|
|
mgr->locale_encoding = get_encoding (mgr, locale_charset);
|
2007-04-07 02:33:53 -05:00
|
|
|
|
|
|
|
enc_mgr_load (mgr);
|
2006-12-17 03:23:19 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
return mgr;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
enc_mgr_save (EncodingsManager *enc_mgr)
|
|
|
|
{
|
2007-04-07 02:33:53 -05:00
|
|
|
MooMarkupDoc *doc;
|
|
|
|
MooMarkupNode *root;
|
|
|
|
GString *string = NULL;
|
|
|
|
|
|
|
|
doc = moo_prefs_get_markup (MOO_PREFS_STATE);
|
|
|
|
g_return_if_fail (doc != NULL);
|
|
|
|
|
|
|
|
if (!(root = moo_markup_get_element (MOO_MARKUP_NODE (doc), ELM_ENCODINGS)))
|
|
|
|
{
|
|
|
|
root = moo_markup_create_element (MOO_MARKUP_NODE (doc), ELM_ENCODINGS);
|
|
|
|
moo_markup_create_element (root, ELM_RECENT);
|
|
|
|
moo_markup_create_element (root, ELM_LAST_OPEN);
|
|
|
|
moo_markup_create_element (root, ELM_LAST_SAVE);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (enc_mgr->recent)
|
|
|
|
{
|
|
|
|
GSList *l;
|
|
|
|
|
|
|
|
string = g_string_new (NULL);
|
|
|
|
|
|
|
|
for (l = enc_mgr->recent; l != NULL; l = l->next)
|
|
|
|
{
|
|
|
|
if (l != enc_mgr->recent)
|
|
|
|
g_string_append (string, ",");
|
|
|
|
g_string_append (string, ((Encoding*)l->data)->name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (string)
|
|
|
|
moo_markup_set_content (moo_markup_get_element (root, ELM_RECENT), string->str);
|
|
|
|
else
|
|
|
|
moo_markup_set_content (moo_markup_get_element (root, ELM_RECENT), NULL);
|
|
|
|
|
|
|
|
moo_markup_set_content (moo_markup_get_element (root, ELM_LAST_OPEN), enc_mgr->last_open);
|
|
|
|
moo_markup_set_content (moo_markup_get_element (root, ELM_LAST_SAVE), enc_mgr->last_save);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
enc_mgr_load (EncodingsManager *enc_mgr)
|
|
|
|
{
|
|
|
|
MooMarkupDoc *doc;
|
|
|
|
MooMarkupNode *root;
|
|
|
|
MooMarkupNode *node;
|
|
|
|
|
|
|
|
g_return_if_fail (enc_mgr->recent == NULL);
|
|
|
|
|
|
|
|
if (!(doc = moo_prefs_get_markup (MOO_PREFS_STATE)) ||
|
|
|
|
!(root = moo_markup_get_element (MOO_MARKUP_NODE (doc), ELM_ENCODINGS)))
|
|
|
|
return;
|
|
|
|
|
|
|
|
if ((node = moo_markup_get_element (root, ELM_RECENT)))
|
|
|
|
{
|
|
|
|
const char *string = moo_markup_get_content (node);
|
|
|
|
|
|
|
|
if (string && *string)
|
|
|
|
{
|
|
|
|
char **encs, **p;
|
|
|
|
encs = g_strsplit (string, ",", 0);
|
|
|
|
for (p = encs; p && *p; ++p)
|
|
|
|
{
|
|
|
|
Encoding *e = get_encoding (enc_mgr, *p);
|
|
|
|
if (e)
|
|
|
|
enc_mgr->recent = g_slist_prepend (enc_mgr->recent, e);
|
|
|
|
}
|
|
|
|
enc_mgr->recent = g_slist_reverse (enc_mgr->recent);
|
|
|
|
g_strfreev (encs);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((node = moo_markup_get_element (root, ELM_LAST_OPEN)))
|
|
|
|
{
|
|
|
|
g_free (enc_mgr->last_open);
|
|
|
|
enc_mgr->last_open = g_strdup (moo_markup_get_content (node));
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((node = moo_markup_get_element (root, ELM_LAST_SAVE)))
|
|
|
|
{
|
|
|
|
g_free (enc_mgr->last_save);
|
|
|
|
enc_mgr->last_save = g_strdup (moo_markup_get_content (node));
|
|
|
|
}
|
2006-12-17 03:23:19 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static const char *
|
|
|
|
current_locale_name (EncodingsManager *enc_mgr)
|
|
|
|
{
|
|
|
|
static char *display_name;
|
|
|
|
|
|
|
|
if (!display_name)
|
|
|
|
{
|
|
|
|
if (enc_mgr->locale_encoding)
|
|
|
|
display_name = g_strdup_printf (_("Current locale (%s)"),
|
|
|
|
enc_mgr->locale_encoding->short_display_name);
|
|
|
|
else
|
|
|
|
display_name = g_strdup (_("Current locale"));
|
|
|
|
}
|
|
|
|
|
|
|
|
return display_name;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
typedef struct {
|
|
|
|
const char *name;
|
|
|
|
GtkTreeIter iter;
|
|
|
|
gboolean found;
|
|
|
|
} FindEncodingData;
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
find_encoding_func (GtkTreeModel *model,
|
|
|
|
G_GNUC_UNUSED GtkTreePath *path,
|
|
|
|
GtkTreeIter *iter,
|
|
|
|
FindEncodingData *data)
|
|
|
|
{
|
|
|
|
char *name;
|
|
|
|
|
|
|
|
gtk_tree_model_get (model, iter, COLUMN_ENCODING, &name, -1);
|
|
|
|
|
|
|
|
if (name && !strcmp (name, data->name))
|
|
|
|
{
|
|
|
|
data->iter = *iter;
|
|
|
|
data->found = TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
g_free (name);
|
|
|
|
return data->found;
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
find_encoding_iter (GtkTreeModel *model,
|
|
|
|
GtkTreeIter *iter,
|
|
|
|
const char *name)
|
|
|
|
{
|
|
|
|
FindEncodingData data;
|
|
|
|
|
|
|
|
data.found = FALSE;
|
|
|
|
data.name = name;
|
|
|
|
|
|
|
|
gtk_tree_model_foreach (model, (GtkTreeModelForeachFunc) find_encoding_func, &data);
|
|
|
|
|
|
|
|
if (data.found && iter)
|
|
|
|
*iter = data.iter;
|
|
|
|
|
|
|
|
return data.found;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
sync_recent_list (GtkTreeStore *store,
|
|
|
|
guint n_old_items,
|
|
|
|
GSList *list,
|
|
|
|
gboolean save_mode)
|
|
|
|
{
|
|
|
|
GtkTreeIter iter;
|
|
|
|
guint i;
|
|
|
|
|
|
|
|
for (i = 0; i < n_old_items; ++i)
|
|
|
|
{
|
|
|
|
gtk_tree_model_iter_nth_child (GTK_TREE_MODEL (store), &iter,
|
|
|
|
NULL, ROW_RECENT (save_mode));
|
|
|
|
gtk_tree_store_remove (store, &iter);
|
|
|
|
}
|
|
|
|
|
|
|
|
while (list)
|
|
|
|
{
|
|
|
|
Encoding *enc = list->data;
|
|
|
|
gtk_tree_store_insert (store, &iter, NULL, ROW_RECENT (save_mode));
|
|
|
|
gtk_tree_store_set (store, &iter,
|
|
|
|
COLUMN_DISPLAY, enc->display_name,
|
|
|
|
COLUMN_ENCODING, enc->name, -1);
|
|
|
|
list = list->next;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
set_last (EncodingsManager *mgr,
|
|
|
|
const char *name,
|
|
|
|
gboolean save_mode)
|
|
|
|
{
|
|
|
|
char **ptr = save_mode ? &mgr->last_save : &mgr->last_open;
|
|
|
|
char *tmp = *ptr;
|
|
|
|
*ptr = g_strdup (name);
|
|
|
|
g_free (tmp);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
encoding_combo_set_active (GtkComboBox *combo,
|
|
|
|
const char *enc_name,
|
|
|
|
gboolean save_mode)
|
|
|
|
{
|
|
|
|
GtkTreeModel *model;
|
|
|
|
GtkTreeIter iter;
|
|
|
|
GSList *l, *recent_copy;
|
|
|
|
EncodingsManager *mgr;
|
|
|
|
gboolean found_recent;
|
|
|
|
guint n_recent;
|
|
|
|
Encoding *new_enc;
|
|
|
|
|
|
|
|
mgr = get_enc_mgr ();
|
|
|
|
|
2007-05-02 00:37:36 -05:00
|
|
|
if (!validate_encoding_name (enc_name))
|
|
|
|
enc_name = save_mode ? MOO_ENCODING_UTF8 : MOO_ENCODING_AUTO;
|
|
|
|
|
2006-12-17 03:23:19 -06:00
|
|
|
if (!strcmp (enc_name, mgr->locale_encoding->name))
|
|
|
|
enc_name = MOO_ENCODING_LOCALE;
|
|
|
|
|
|
|
|
set_last (mgr, enc_name, save_mode);
|
|
|
|
|
|
|
|
model = gtk_combo_box_get_model (combo);
|
|
|
|
g_signal_handlers_block_by_func (combo, (gpointer) combo_changed,
|
|
|
|
GINT_TO_POINTER (save_mode));
|
|
|
|
|
|
|
|
if (!strcmp (enc_name, MOO_ENCODING_AUTO))
|
|
|
|
{
|
|
|
|
gtk_tree_model_iter_nth_child (model, &iter, NULL, ROW_AUTO);
|
|
|
|
gtk_combo_box_set_active_iter (combo, &iter);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!strcmp (enc_name, MOO_ENCODING_LOCALE))
|
|
|
|
{
|
|
|
|
gtk_tree_model_iter_nth_child (model, &iter, NULL, ROW_LOCALE (save_mode));
|
|
|
|
gtk_combo_box_set_active_iter (combo, &iter);
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
2006-12-17 10:58:17 -06:00
|
|
|
new_enc = get_encoding (mgr, enc_name);
|
2006-12-17 03:23:19 -06:00
|
|
|
g_return_if_fail (new_enc != NULL);
|
|
|
|
|
|
|
|
set_last (mgr, new_enc->name, save_mode);
|
|
|
|
|
|
|
|
n_recent = g_slist_length (mgr->recent);
|
|
|
|
|
|
|
|
for (l = mgr->recent, found_recent = FALSE; l != NULL; l = l->next)
|
|
|
|
{
|
|
|
|
Encoding *enc = l->data;
|
|
|
|
|
|
|
|
if (!strcmp (new_enc->name, enc->name))
|
|
|
|
{
|
|
|
|
found_recent = TRUE;
|
|
|
|
|
|
|
|
if (l != mgr->recent)
|
|
|
|
{
|
|
|
|
mgr->recent = g_slist_remove_link (mgr->recent, l);
|
|
|
|
l->next = mgr->recent;
|
|
|
|
mgr->recent = l;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!found_recent)
|
|
|
|
{
|
|
|
|
mgr->recent = g_slist_prepend (mgr->recent, new_enc);
|
|
|
|
|
|
|
|
if (g_slist_length (mgr->recent) > MAX_RECENT_ENCODINGS)
|
|
|
|
mgr->recent = g_slist_delete_link (mgr->recent, g_slist_last (mgr->recent));
|
|
|
|
}
|
|
|
|
|
|
|
|
recent_copy = g_slist_reverse (g_slist_copy (mgr->recent));
|
|
|
|
sync_recent_list (GTK_TREE_STORE (model), n_recent, recent_copy, save_mode);
|
|
|
|
g_slist_free (recent_copy);
|
|
|
|
|
|
|
|
if (find_encoding_iter (model, &iter, new_enc->name))
|
|
|
|
gtk_combo_box_set_active_iter (combo, &iter);
|
|
|
|
|
|
|
|
out:
|
|
|
|
g_signal_handlers_unblock_by_func (combo, (gpointer) combo_changed,
|
|
|
|
GINT_TO_POINTER (save_mode));
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
combo_changed (GtkComboBox *combo,
|
|
|
|
gpointer save_mode)
|
|
|
|
{
|
|
|
|
GtkTreeModel *model;
|
|
|
|
GtkTreeIter iter;
|
|
|
|
char *enc_name;
|
|
|
|
|
|
|
|
if (!gtk_combo_box_get_active_iter (combo, &iter))
|
|
|
|
return;
|
|
|
|
|
|
|
|
model = gtk_combo_box_get_model (combo);
|
|
|
|
|
|
|
|
if (gtk_tree_model_iter_has_child (model, &iter))
|
|
|
|
return;
|
|
|
|
|
|
|
|
gtk_tree_model_get (model, &iter, COLUMN_ENCODING, &enc_name, -1);
|
|
|
|
|
|
|
|
if (!enc_name)
|
|
|
|
return;
|
|
|
|
|
|
|
|
encoding_combo_set_active (combo, enc_name, GPOINTER_TO_INT (save_mode));
|
|
|
|
g_free (enc_name);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-06-14 02:18:35 -05:00
|
|
|
#if 0
|
2006-12-17 03:23:19 -06:00
|
|
|
static gboolean
|
|
|
|
row_separator_func (GtkTreeModel *model,
|
|
|
|
GtkTreeIter *iter)
|
|
|
|
{
|
|
|
|
char *text;
|
|
|
|
gboolean separator;
|
|
|
|
|
|
|
|
gtk_tree_model_get (model, iter, COLUMN_DISPLAY, &text, -1);
|
|
|
|
separator = text == NULL;
|
|
|
|
|
|
|
|
g_free (text);
|
|
|
|
return separator;
|
|
|
|
}
|
2007-06-14 02:18:35 -05:00
|
|
|
#endif
|
|
|
|
|
2006-12-17 03:23:19 -06:00
|
|
|
|
|
|
|
static void
|
|
|
|
cell_data_func (G_GNUC_UNUSED GtkCellLayout *layout,
|
|
|
|
GtkCellRenderer *cell,
|
|
|
|
GtkTreeModel *model,
|
|
|
|
GtkTreeIter *iter)
|
|
|
|
{
|
|
|
|
gboolean sensitive;
|
|
|
|
sensitive = !gtk_tree_model_iter_has_child (model, iter);
|
|
|
|
g_object_set (cell, "sensitive", sensitive, NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
setup_combo (GtkComboBox *combo,
|
|
|
|
EncodingsManager *enc_mgr,
|
|
|
|
gboolean save_mode)
|
|
|
|
{
|
|
|
|
GtkCellRenderer *cell;
|
|
|
|
GtkTreeStore *store;
|
|
|
|
GtkTreeIter iter;
|
|
|
|
GSList *l;
|
|
|
|
guint i;
|
|
|
|
char *start;
|
|
|
|
|
|
|
|
store = gtk_tree_store_new (2, G_TYPE_STRING, G_TYPE_STRING);
|
|
|
|
|
|
|
|
if (!save_mode)
|
|
|
|
{
|
|
|
|
gtk_tree_store_append (store, &iter, NULL);
|
|
|
|
gtk_tree_store_set (store, &iter,
|
|
|
|
COLUMN_DISPLAY, _("Auto Detected"),
|
|
|
|
COLUMN_ENCODING, MOO_ENCODING_AUTO,
|
|
|
|
-1);
|
|
|
|
|
|
|
|
gtk_tree_store_append (store, &iter, NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
gtk_tree_store_append (store, &iter, NULL);
|
|
|
|
gtk_tree_store_set (store, &iter,
|
|
|
|
COLUMN_DISPLAY, current_locale_name (enc_mgr),
|
|
|
|
COLUMN_ENCODING, MOO_ENCODING_LOCALE,
|
|
|
|
-1);
|
|
|
|
|
|
|
|
for (l = enc_mgr->recent; l != NULL; l = l->next)
|
|
|
|
{
|
|
|
|
Encoding *enc = l->data;
|
|
|
|
gtk_tree_store_append (store, &iter, NULL);
|
|
|
|
gtk_tree_store_set (store, &iter,
|
|
|
|
COLUMN_DISPLAY, enc->display_name,
|
|
|
|
COLUMN_ENCODING, enc->name,
|
|
|
|
-1);
|
|
|
|
}
|
|
|
|
|
|
|
|
gtk_tree_store_append (store, &iter, NULL);
|
2007-06-14 02:18:35 -05:00
|
|
|
/* Translators: do not translate the part before | */
|
|
|
|
gtk_tree_store_set (store, &iter, COLUMN_DISPLAY, _("Encodings combo submenu|Other"), -1);
|
2006-12-17 03:23:19 -06:00
|
|
|
|
|
|
|
for (i = 0; i < enc_mgr->n_groups; ++i)
|
|
|
|
{
|
|
|
|
guint j;
|
|
|
|
GtkTreeIter child, group_iter;
|
|
|
|
EncodingGroup *group;
|
|
|
|
|
|
|
|
group = &enc_mgr->groups[i];
|
|
|
|
gtk_tree_store_append (store, &group_iter, &iter);
|
|
|
|
gtk_tree_store_set (store, &group_iter, COLUMN_DISPLAY, group->name, -1);
|
|
|
|
|
|
|
|
for (j = 0; j < group->n_encodings; ++j)
|
|
|
|
{
|
|
|
|
gtk_tree_store_append (store, &child, &group_iter);
|
|
|
|
gtk_tree_store_set (store, &child,
|
|
|
|
COLUMN_DISPLAY, group->encodings[j]->display_name,
|
|
|
|
COLUMN_ENCODING, group->encodings[j]->name,
|
|
|
|
-1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
gtk_combo_box_set_model (combo, GTK_TREE_MODEL (store));
|
|
|
|
gtk_combo_box_entry_set_text_column (GTK_COMBO_BOX_ENTRY (combo), COLUMN_DISPLAY);
|
2007-06-14 02:18:35 -05:00
|
|
|
|
2006-12-17 03:23:19 -06:00
|
|
|
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_DISPLAY, NULL);
|
|
|
|
gtk_cell_layout_set_cell_data_func (GTK_CELL_LAYOUT (combo), cell,
|
|
|
|
(GtkCellLayoutDataFunc) cell_data_func,
|
|
|
|
NULL, NULL);
|
|
|
|
|
|
|
|
if (save_mode)
|
|
|
|
{
|
|
|
|
if (enc_mgr->last_save)
|
|
|
|
start = g_strdup (enc_mgr->last_save);
|
|
|
|
else
|
|
|
|
start = g_strdup (MOO_ENCODING_UTF8);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (enc_mgr->last_open)
|
|
|
|
start = g_strdup (enc_mgr->last_open);
|
|
|
|
else
|
|
|
|
start = g_strdup (MOO_ENCODING_AUTO);
|
|
|
|
}
|
|
|
|
|
|
|
|
g_signal_connect (combo, "changed", G_CALLBACK (combo_changed),
|
|
|
|
GINT_TO_POINTER (save_mode));
|
|
|
|
encoding_combo_set_active (GTK_COMBO_BOX (combo), start, save_mode);
|
|
|
|
|
|
|
|
g_free (start);
|
|
|
|
g_object_unref (store);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-05-02 00:37:36 -05:00
|
|
|
void
|
|
|
|
_moo_encodings_combo_init (GtkComboBox *combo,
|
|
|
|
MooEncodingComboType type)
|
|
|
|
{
|
|
|
|
g_return_if_fail (GTK_IS_COMBO_BOX (combo));
|
|
|
|
setup_combo (combo, get_enc_mgr (), type == MOO_ENCODING_COMBO_SAVE);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
_moo_encodings_combo_set_enc (GtkComboBox *combo,
|
|
|
|
const char *enc,
|
|
|
|
MooEncodingComboType type)
|
|
|
|
{
|
|
|
|
g_return_if_fail (GTK_IS_COMBO_BOX (combo));
|
|
|
|
encoding_combo_set_active (combo, enc, type == MOO_ENCODING_COMBO_SAVE);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-12-17 03:23:19 -06:00
|
|
|
void
|
|
|
|
_moo_encodings_attach_combo (GtkWidget *dialog,
|
|
|
|
GtkWidget *parent,
|
|
|
|
gboolean save_mode,
|
|
|
|
const char *encoding)
|
|
|
|
{
|
|
|
|
GtkWidget *hbox;
|
|
|
|
GtkWidget *label;
|
|
|
|
GtkWidget *combo;
|
|
|
|
|
|
|
|
g_return_if_fail (GTK_IS_FILE_CHOOSER (dialog));
|
|
|
|
|
|
|
|
hbox = gtk_hbox_new (FALSE, 0);
|
|
|
|
gtk_widget_show (hbox);
|
|
|
|
gtk_box_pack_start (GTK_BOX (parent), hbox, FALSE, FALSE, 0);
|
|
|
|
|
|
|
|
label = gtk_label_new_with_mnemonic (_("Charact_er encoding:"));
|
|
|
|
gtk_widget_show (label);
|
|
|
|
gtk_box_pack_start (GTK_BOX (hbox), label, FALSE, FALSE, 0);
|
|
|
|
|
|
|
|
combo = gtk_combo_box_entry_new ();
|
|
|
|
gtk_widget_show (combo);
|
|
|
|
gtk_box_pack_start (GTK_BOX (hbox), combo, TRUE, TRUE, 0);
|
|
|
|
|
|
|
|
setup_combo (GTK_COMBO_BOX (combo), get_enc_mgr (), save_mode);
|
|
|
|
|
|
|
|
if (save_mode && encoding)
|
|
|
|
encoding_combo_set_active (GTK_COMBO_BOX (combo), encoding, save_mode);
|
|
|
|
|
|
|
|
g_object_set_data (G_OBJECT (dialog), "moo-encodings-combo", combo);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-12-31 04:53:45 -06:00
|
|
|
static void
|
2007-05-02 00:37:36 -05:00
|
|
|
sync_combo (GtkComboBox *combo,
|
|
|
|
gboolean save_mode)
|
2006-12-17 03:23:19 -06:00
|
|
|
{
|
|
|
|
const char *enc_name;
|
|
|
|
GtkTreeIter dummy;
|
|
|
|
|
|
|
|
if (gtk_combo_box_get_active_iter (combo, &dummy))
|
|
|
|
return;
|
|
|
|
|
|
|
|
enc_name = gtk_entry_get_text (GTK_ENTRY (GTK_BIN (combo)->child));
|
|
|
|
|
|
|
|
if (!validate_encoding_name (enc_name))
|
|
|
|
enc_name = save_mode ? MOO_ENCODING_UTF8 : MOO_ENCODING_AUTO;
|
|
|
|
|
|
|
|
encoding_combo_set_active (combo, enc_name, save_mode);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-05-02 00:37:36 -05:00
|
|
|
static const char *
|
|
|
|
combo_get (GtkComboBox *combo,
|
|
|
|
gboolean save_mode)
|
2006-12-17 03:23:19 -06:00
|
|
|
{
|
|
|
|
const char *enc_name;
|
|
|
|
EncodingsManager *mgr;
|
|
|
|
|
2007-05-02 00:37:36 -05:00
|
|
|
sync_combo (combo, save_mode);
|
2006-12-17 03:23:19 -06:00
|
|
|
|
|
|
|
mgr = get_enc_mgr ();
|
|
|
|
enc_mgr_save (mgr);
|
|
|
|
|
|
|
|
if (save_mode)
|
|
|
|
{
|
|
|
|
if (mgr->last_save)
|
|
|
|
enc_name = mgr->last_save;
|
|
|
|
else
|
|
|
|
enc_name = MOO_ENCODING_UTF8;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (mgr->last_open)
|
|
|
|
enc_name = mgr->last_open;
|
|
|
|
else
|
|
|
|
enc_name = MOO_ENCODING_AUTO;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!strcmp (enc_name, MOO_ENCODING_LOCALE))
|
|
|
|
enc_name = mgr->locale_encoding->name;
|
|
|
|
|
|
|
|
return enc_name;
|
|
|
|
}
|
2006-12-17 10:58:17 -06:00
|
|
|
|
2007-05-02 00:37:36 -05:00
|
|
|
const char *
|
|
|
|
_moo_encodings_combo_get (GtkWidget *dialog,
|
|
|
|
gboolean save_mode)
|
|
|
|
{
|
|
|
|
GtkComboBox *combo;
|
|
|
|
|
|
|
|
combo = g_object_get_data (G_OBJECT (dialog), "moo-encodings-combo");
|
|
|
|
g_return_val_if_fail (GTK_IS_COMBO_BOX (combo), MOO_ENCODING_UTF8);
|
|
|
|
|
|
|
|
return combo_get (combo, save_mode);
|
|
|
|
}
|
|
|
|
|
|
|
|
const char *
|
|
|
|
_moo_encodings_combo_get_enc (GtkComboBox *combo,
|
|
|
|
MooEncodingComboType type)
|
|
|
|
{
|
|
|
|
g_return_val_if_fail (GTK_IS_COMBO_BOX (combo), MOO_ENCODING_UTF8);
|
|
|
|
return combo_get (combo, type == MOO_ENCODING_COMBO_SAVE);
|
|
|
|
}
|
|
|
|
|
2006-12-17 10:58:17 -06:00
|
|
|
|
|
|
|
const char *
|
|
|
|
_moo_encoding_locale (void)
|
|
|
|
{
|
|
|
|
EncodingsManager *mgr;
|
|
|
|
|
|
|
|
mgr = get_enc_mgr ();
|
|
|
|
|
|
|
|
return mgr->locale_encoding->name;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
gboolean
|
|
|
|
_moo_encodings_equal (const char *enc1_name,
|
|
|
|
const char *enc2_name)
|
|
|
|
{
|
|
|
|
Encoding *enc1, *enc2;
|
|
|
|
EncodingsManager *mgr;
|
|
|
|
|
|
|
|
enc1_name = enc1_name && enc1_name[0] ? enc1_name : MOO_ENCODING_UTF8;
|
|
|
|
enc2_name = enc2_name && enc2_name[0] ? enc2_name : MOO_ENCODING_UTF8;
|
|
|
|
|
|
|
|
mgr = get_enc_mgr ();
|
|
|
|
|
|
|
|
enc1 = lookup_encoding (mgr, enc1_name);
|
|
|
|
enc2 = lookup_encoding (mgr, enc2_name);
|
|
|
|
|
|
|
|
if (!enc1 || !enc2)
|
|
|
|
return !strcmp (enc1_name, enc2_name);
|
|
|
|
|
|
|
|
return enc1 == enc2;
|
|
|
|
}
|