user-accounts: Move more of the treeview setup to common/

Make more of the code common one, so we can re-use it for
the region panel. Show the user's current locale first, as well
as the list of languages used by other users, followed by a separator.
This commit is contained in:
Bastien Nocera 2011-01-25 17:07:08 +00:00
parent 107e4db2f8
commit f7f1b4c073
3 changed files with 188 additions and 230 deletions

View file

@ -34,51 +34,7 @@
#include "gdm-languages.h" #include "gdm-languages.h"
#if 0 static gint
struct _UmLanguageDialog {
GtkWidget *dialog;
GtkWidget *user_icon;
GtkWidget *user_name;
GtkWidget *dialog_combo;
GtkListStore *dialog_store;
GtkWidget *chooser;
GtkWidget *chooser_list;
GtkListStore *chooser_store;
char *language;
// UmUser *user;
gboolean force_setting;
};
#endif
enum {
LOCALE_COL,
DISPLAY_LOCALE_COL,
NUM_COLS
};
#if 0
gchar *
um_language_chooser_get_language (GtkWidget *chooser)
{
GtkTreeView *tv;
GtkTreeSelection *selection;
GtkTreeModel *model;
GtkTreeIter iter;
gchar *lang;
tv = (GtkTreeView *) g_object_get_data (G_OBJECT (chooser), "list");
selection = gtk_tree_view_get_selection (tv);
if (gtk_tree_selection_get_selected (selection, &model, &iter))
gtk_tree_model_get (model, &iter, LOCALE_COL, &lang, -1);
else
lang = NULL;
return lang;
}
#endif
gint
cc_common_language_sort_languages (GtkTreeModel *model, cc_common_language_sort_languages (GtkTreeModel *model,
GtkTreeIter *a, GtkTreeIter *a,
GtkTreeIter *b, GtkTreeIter *b,
@ -86,10 +42,38 @@ cc_common_language_sort_languages (GtkTreeModel *model,
{ {
char *ca, *cb; char *ca, *cb;
char *la, *lb; char *la, *lb;
gboolean sa, ula;
gboolean sb, ulb;
gint result; gint result;
gtk_tree_model_get (model, a, LOCALE_COL, &ca, DISPLAY_LOCALE_COL, &la, -1); gtk_tree_model_get (model, a,
gtk_tree_model_get (model, b, LOCALE_COL, &cb, DISPLAY_LOCALE_COL, &lb, -1); LOCALE_COL, &ca,
DISPLAY_LOCALE_COL, &la,
SEPARATOR_COL, &sa,
USER_LANGUAGE, &ula,
-1);
gtk_tree_model_get (model, b,
LOCALE_COL, &cb,
DISPLAY_LOCALE_COL, &lb,
SEPARATOR_COL, &sb,
USER_LANGUAGE, &ulb,
-1);
/* Sort before and after separator first */
if (sa && sb)
return 0;
if (sa)
return ulb ? 1 : -1;
if (sb)
return ula ? -1 : 1;
/* Sort user-languages first */
if (ula != ulb) {
if (ula)
return -1;
else
return 1;
}
if (!ca) if (!ca)
result = 1; result = 1;
@ -139,17 +123,6 @@ cc_common_language_get_iter_for_language (GtkTreeModel *model,
return FALSE; return FALSE;
} }
#if 0
static void
row_activated (GtkTreeView *tree_view,
GtkTreePath *path,
GtkTreeViewColumn *column,
GtkWidget *chooser)
{
gtk_dialog_response (GTK_DIALOG (chooser), GTK_RESPONSE_OK);
}
#endif
gboolean gboolean
cc_common_language_has_font (const gchar *locale) cc_common_language_has_font (const gchar *locale)
{ {
@ -209,7 +182,8 @@ cc_common_language_has_font (const gchar *locale)
} }
void void
cc_common_language_add_available_languages (GtkListStore *store) cc_common_language_add_available_languages (GtkListStore *store,
GHashTable *user_langs)
{ {
char **languages; char **languages;
int i; int i;
@ -217,15 +191,20 @@ cc_common_language_add_available_languages (GtkListStore *store)
char *language; char *language;
GtkTreeIter iter; GtkTreeIter iter;
gtk_list_store_clear (store);
languages = gdm_get_all_language_names (); languages = gdm_get_all_language_names ();
for (i = 0; languages[i] != NULL; i++) { for (i = 0; languages[i] != NULL; i++) {
if (!cc_common_language_has_font (languages[i]))
continue;
name = gdm_normalize_language_name (languages[i]); name = gdm_normalize_language_name (languages[i]);
if (g_hash_table_lookup (user_langs, name) != NULL) {
g_free (name);
continue;
}
if (!cc_common_language_has_font (languages[i])) {
g_free (name);
continue;
}
language = gdm_get_language_from_name (name, NULL); language = gdm_get_language_from_name (name, NULL);
gtk_list_store_append (store, &iter); gtk_list_store_append (store, &iter);
@ -238,74 +217,6 @@ cc_common_language_add_available_languages (GtkListStore *store)
g_strfreev (languages); g_strfreev (languages);
} }
#if 0
void
um_add_user_languages (GtkTreeModel *model)
{
#if 0
GHashTable *seen;
GSList *users, *l;
UmUser *user;
const char *lang;
char *name;
char *language;
GtkTreeIter iter;
UmUserManager *manager;
GtkListStore *store = GTK_LIST_STORE (model);
gtk_list_store_clear (store);
seen = g_hash_table_new_full (g_str_hash, g_str_equal, g_free, NULL);
manager = um_user_manager_ref_default ();
users = um_user_manager_list_users (manager);
g_object_unref (manager);
for (l = users; l; l = l->next) {
user = l->data;
lang = um_user_get_language (user);
if (!lang || !cc_common_language_has_font (lang)) {
continue;
}
name = gdm_normalize_language_name (lang);
if (g_hash_table_lookup (seen, name)) {
g_free (name);
continue;
}
g_hash_table_insert (seen, name, GINT_TO_POINTER (TRUE));
language = gdm_get_language_from_name (name, NULL);
gtk_list_store_append (store, &iter);
gtk_list_store_set (store, &iter, LOCALE_COL, name, DISPLAY_LOCALE_COL, language, -1);
g_free (language);
}
g_slist_free (users);
/* Make sure the current locale is present */
name = um_get_current_language ();
if (!g_hash_table_lookup (seen, name)) {
language = gdm_get_language_from_name (name, NULL);
gtk_list_store_append (store, &iter);
gtk_list_store_set (store, &iter, LOCALE_COL, name, DISPLAY_LOCALE_COL, language, -1);
g_free (language);
}
g_free (name);
g_hash_table_destroy (seen);
gtk_list_store_append (store, &iter);
gtk_list_store_set (store, &iter, LOCALE_COL, NULL, DISPLAY_LOCALE_COL, _("Other..."), -1);
#endif
}
#endif
gchar * gchar *
cc_common_language_get_current_language (void) cc_common_language_get_current_language (void)
{ {
@ -321,58 +232,80 @@ cc_common_language_get_current_language (void)
return language; return language;
} }
#if 0 static void
GtkWidget * languages_foreach_cb (gpointer key,
um_language_chooser_new (void) gpointer value,
gpointer user_data)
{ {
GtkBuilder *builder; GtkListStore *store = (GtkListStore *) user_data;
const char *filename; const char *locale = (const char *) key;
GError *error = NULL; const char *display_locale = (const char *) value;
GtkWidget *chooser; GtkTreeIter iter;
GtkWidget *list;
GtkWidget *button;
GtkTreeViewColumn *column;
GtkCellRenderer *cell;
GtkListStore *store;
builder = gtk_builder_new (); gtk_list_store_append (store, &iter);
filename = UIDIR "/language-chooser.ui"; gtk_list_store_set (store, &iter,
if (!g_file_test (filename, G_FILE_TEST_EXISTS)) LOCALE_COL, locale,
filename = "data/language-chooser.ui"; DISPLAY_LOCALE_COL, display_locale,
if (!gtk_builder_add_from_file (builder, filename, &error)) { SEPARATOR_COL, FALSE,
g_warning ("failed to load language chooser: %s", error->message); USER_LANGUAGE, TRUE,
g_error_free (error); -1);
return NULL;
g_message ("adding '%s' (%s) to the store", display_locale, locale);
} }
chooser = (GtkWidget *) gtk_builder_get_object (builder, "dialog"); static gboolean
separator_func (GtkTreeModel *model,
GtkTreeIter *iter,
gpointer data)
{
gboolean is_sep;
list = (GtkWidget *) gtk_builder_get_object (builder, "language-list"); gtk_tree_model_get (model, iter,
g_object_set_data (G_OBJECT (chooser), "list", list); SEPARATOR_COL, &is_sep,
g_signal_connect (list, "row-activated", -1);
G_CALLBACK (row_activated), chooser);
button = (GtkWidget *) gtk_builder_get_object (builder, "cancel-button"); return is_sep;
button = (GtkWidget *) gtk_builder_get_object (builder, "ok-button"); }
gtk_widget_grab_default (button);
void
cc_common_language_setup_list (GtkWidget *treeview,
GHashTable *initial)
{
GtkCellRenderer *cell;
GtkTreeViewColumn *column;
GtkListStore *store;
cell = gtk_cell_renderer_text_new (); cell = gtk_cell_renderer_text_new ();
column = gtk_tree_view_column_new_with_attributes (NULL, cell, "text", DISPLAY_LOCALE_COL, NULL); column = gtk_tree_view_column_new_with_attributes (NULL, cell, "text", DISPLAY_LOCALE_COL, NULL);
gtk_tree_view_append_column (GTK_TREE_VIEW (list), column); gtk_tree_view_append_column (GTK_TREE_VIEW (treeview), column);
store = gtk_list_store_new (NUM_COLS, G_TYPE_STRING, G_TYPE_STRING); store = gtk_list_store_new (NUM_COLS, G_TYPE_STRING, G_TYPE_STRING, G_TYPE_BOOLEAN, G_TYPE_BOOLEAN);
gtk_tree_sortable_set_default_sort_func (GTK_TREE_SORTABLE (store), gtk_tree_sortable_set_default_sort_func (GTK_TREE_SORTABLE (store),
sort_languages, NULL, NULL); cc_common_language_sort_languages, NULL, NULL);
gtk_tree_sortable_set_sort_column_id (GTK_TREE_SORTABLE (store), gtk_tree_sortable_set_sort_column_id (GTK_TREE_SORTABLE (store),
GTK_TREE_SORTABLE_DEFAULT_SORT_COLUMN_ID, GTK_TREE_SORTABLE_DEFAULT_SORT_COLUMN_ID,
GTK_SORT_ASCENDING); GTK_SORT_ASCENDING);
gtk_tree_view_set_row_separator_func (GTK_TREE_VIEW (treeview),
separator_func,
NULL, NULL);
gtk_tree_view_set_model (GTK_TREE_VIEW (list), GTK_TREE_MODEL (store)); gtk_tree_view_set_model (GTK_TREE_VIEW (treeview), GTK_TREE_MODEL (store));
add_available_languages (store);
g_object_unref (builder); /* Add languages from the initial hashtable */
g_hash_table_foreach (initial, (GHFunc) languages_foreach_cb, store);
return chooser; /* Add separator if we had any languages added */
if (initial != NULL &&
g_hash_table_size (initial) > 0) {
GtkTreeIter iter;
gtk_list_store_append (GTK_LIST_STORE (store), &iter);
gtk_list_store_set (GTK_LIST_STORE (store), &iter,
LOCALE_COL, NULL,
DISPLAY_LOCALE_COL, "Don't show",
SEPARATOR_COL, TRUE,
USER_LANGUAGE, FALSE,
-1);
}
} }
#endif

View file

@ -26,25 +26,25 @@
G_BEGIN_DECLS G_BEGIN_DECLS
#if 0 enum {
void um_add_user_languages (GtkTreeModel *model); LOCALE_COL,
gchar *um_get_current_language (void); DISPLAY_LOCALE_COL,
SEPARATOR_COL,
USER_LANGUAGE,
NUM_COLS
};
GtkWidget *um_language_chooser_new (void);
gchar *um_language_chooser_get_language (GtkWidget *chooser);
#endif
gint cc_common_language_sort_languages (GtkTreeModel *model,
GtkTreeIter *a,
GtkTreeIter *b,
gpointer data);
gboolean cc_common_language_get_iter_for_language (GtkTreeModel *model, gboolean cc_common_language_get_iter_for_language (GtkTreeModel *model,
const gchar *lang, const gchar *lang,
GtkTreeIter *iter); GtkTreeIter *iter);
void cc_common_language_add_available_languages (GtkListStore *store); void cc_common_language_add_available_languages (GtkListStore *store,
GHashTable *user_langs);
gboolean cc_common_language_has_font (const gchar *locale); gboolean cc_common_language_has_font (const gchar *locale);
gchar *cc_common_language_get_current_language (void); gchar *cc_common_language_get_current_language (void);
void cc_common_language_setup_list (GtkWidget *treeview,
GHashTable *initial);
G_END_DECLS G_END_DECLS
#endif #endif

View file

@ -53,12 +53,6 @@ struct _UmLanguageDialog {
gboolean force_setting; gboolean force_setting;
}; };
enum {
LOCALE_COL,
DISPLAY_LOCALE_COL,
NUM_COLS
};
gchar * gchar *
um_language_chooser_get_language (GtkWidget *chooser) um_language_chooser_get_language (GtkWidget *chooser)
{ {
@ -87,28 +81,25 @@ row_activated (GtkTreeView *tree_view,
gtk_dialog_response (GTK_DIALOG (chooser), GTK_RESPONSE_OK); gtk_dialog_response (GTK_DIALOG (chooser), GTK_RESPONSE_OK);
} }
void static GHashTable *
um_add_user_languages (GtkTreeModel *model) new_ht_for_user_languages (void)
{ {
GHashTable *seen; GHashTable *ht;
UmUserManager *manager;
GSList *users, *l; GSList *users, *l;
UmUser *user; UmUser *user;
const char *lang;
char *name; char *name;
char *language;
GtkTreeIter iter;
UmUserManager *manager;
GtkListStore *store = GTK_LIST_STORE (model);
gtk_list_store_clear (store); ht = g_hash_table_new_full (g_str_hash, g_str_equal, g_free, g_free);
seen = g_hash_table_new_full (g_str_hash, g_str_equal, g_free, NULL);
manager = um_user_manager_ref_default (); manager = um_user_manager_ref_default ();
users = um_user_manager_list_users (manager); users = um_user_manager_list_users (manager);
g_object_unref (manager); g_object_unref (manager);
for (l = users; l; l = l->next) { for (l = users; l; l = l->next) {
const char *lang;
char *language;
user = l->data; user = l->data;
lang = um_user_get_language (user); lang = um_user_get_language (user);
if (!lang || !cc_common_language_has_font (lang)) { if (!lang || !cc_common_language_has_font (lang)) {
@ -116,39 +107,78 @@ um_add_user_languages (GtkTreeModel *model)
} }
name = gdm_normalize_language_name (lang); name = gdm_normalize_language_name (lang);
if (g_hash_table_lookup (ht, name) != NULL) {
if (g_hash_table_lookup (seen, name)) {
g_free (name); g_free (name);
continue; continue;
} }
g_hash_table_insert (seen, name, GINT_TO_POINTER (TRUE));
language = gdm_get_language_from_name (name, NULL); language = gdm_get_language_from_name (name, NULL);
gtk_list_store_append (store, &iter);
gtk_list_store_set (store, &iter, LOCALE_COL, name, DISPLAY_LOCALE_COL, language, -1);
g_free (language); g_hash_table_insert (ht, name, language);
} }
g_slist_free (users); g_slist_free (users);
/* Make sure the current locale is present */ /* Make sure the current locale is present */
name = cc_common_language_get_current_language (); name = cc_common_language_get_current_language ();
if (g_hash_table_lookup (ht, name) == NULL) {
if (!g_hash_table_lookup (seen, name)) { char *language;
language = gdm_get_language_from_name (name, NULL); language = gdm_get_language_from_name (name, NULL);
gtk_list_store_append (store, &iter); g_hash_table_insert (ht, name, language);
gtk_list_store_set (store, &iter, LOCALE_COL, name, DISPLAY_LOCALE_COL, language, -1);
g_free (language);
} }
g_free (name); g_free (name);
g_hash_table_destroy (seen); return ht;
}
static void
languages_foreach_cb (gpointer key,
gpointer value,
gpointer user_data)
{
GtkListStore *store = (GtkListStore *) user_data;
const char *locale = (const char *) key;
const char *display_locale = (const char *) value;
GtkTreeIter iter;
gtk_list_store_append (store, &iter);
gtk_list_store_set (store, &iter,
LOCALE_COL, locale,
DISPLAY_LOCALE_COL, display_locale,
-1);
}
void
um_add_user_languages (GtkTreeModel *model)
{
char *name;
GtkTreeIter iter;
GtkListStore *store = GTK_LIST_STORE (model);
GHashTable *user_langs;
const char *display;
gtk_list_store_clear (store);
user_langs = new_ht_for_user_languages ();
/* Add the current locale first */
name = cc_common_language_get_current_language ();
display = g_hash_table_lookup (user_langs, name);
gtk_list_store_append (store, &iter);
gtk_list_store_set (store, &iter, LOCALE_COL, name, DISPLAY_LOCALE_COL, display, -1);
g_hash_table_remove (user_langs, name);
g_free (name);
/* The rest of the languages */
g_hash_table_foreach (user_langs, (GHFunc) languages_foreach_cb, store);
/* And now the "Other..." selection */
gtk_list_store_append (store, &iter); gtk_list_store_append (store, &iter);
gtk_list_store_set (store, &iter, LOCALE_COL, NULL, DISPLAY_LOCALE_COL, _("Other..."), -1); gtk_list_store_set (store, &iter, LOCALE_COL, NULL, DISPLAY_LOCALE_COL, _("Other..."), -1);
g_hash_table_destroy (user_langs);
} }
GtkWidget * GtkWidget *
@ -160,9 +190,8 @@ um_language_chooser_new (void)
GtkWidget *chooser; GtkWidget *chooser;
GtkWidget *list; GtkWidget *list;
GtkWidget *button; GtkWidget *button;
GtkTreeViewColumn *column; GtkTreeModel *model;
GtkCellRenderer *cell; GHashTable *user_langs;
GtkListStore *store;
builder = gtk_builder_new (); builder = gtk_builder_new ();
filename = UIDIR "/language-chooser.ui"; filename = UIDIR "/language-chooser.ui";
@ -185,19 +214,15 @@ um_language_chooser_new (void)
button = (GtkWidget *) gtk_builder_get_object (builder, "ok-button"); button = (GtkWidget *) gtk_builder_get_object (builder, "ok-button");
gtk_widget_grab_default (button); gtk_widget_grab_default (button);
cell = gtk_cell_renderer_text_new (); /* Add user languages */
column = gtk_tree_view_column_new_with_attributes (NULL, cell, "text", DISPLAY_LOCALE_COL, NULL); user_langs = new_ht_for_user_languages ();
gtk_tree_view_append_column (GTK_TREE_VIEW (list), column); cc_common_language_setup_list (list, user_langs);
store = gtk_list_store_new (NUM_COLS, G_TYPE_STRING, G_TYPE_STRING); model = gtk_tree_view_get_model (GTK_TREE_VIEW (list));
gtk_tree_sortable_set_default_sort_func (GTK_TREE_SORTABLE (store),
cc_common_language_sort_languages, NULL, NULL);
gtk_tree_sortable_set_sort_column_id (GTK_TREE_SORTABLE (store),
GTK_TREE_SORTABLE_DEFAULT_SORT_COLUMN_ID,
GTK_SORT_ASCENDING);
gtk_tree_view_set_model (GTK_TREE_VIEW (list), GTK_TREE_MODEL (store)); /* Add the other languages */
cc_common_language_add_available_languages (GTK_LIST_STORE (model), user_langs);
cc_common_language_add_available_languages (store); g_hash_table_destroy (user_langs);
g_object_unref (builder); g_object_unref (builder);