edcc2fc
diff -up gdm-2.22.0/configure.ac.keyboard-chooser gdm-2.22.0/configure.ac
edcc2fc
--- gdm-2.22.0/configure.ac.keyboard-chooser	2008-05-05 11:29:30.000000000 -0400
edcc2fc
+++ gdm-2.22.0/configure.ac	2008-05-05 11:30:14.000000000 -0400
edcc2fc
@@ -48,6 +48,7 @@ SCROLLKEEPER_REQUIRED_VERSION=0.1.4
edcc2fc
 GCONF_REQUIRED_VERSION=2.6.1
edcc2fc
 POLICYKIT_REQUIRED_VERSION=0.7
edcc2fc
 GNOME_PANEL_REQUIRED_VERSION=2.0.0
edcc2fc
+LIBXKLAVIER_REQUIRED_VERSION=3.5
edcc2fc
 
edcc2fc
 EXTRA_COMPILE_WARNINGS(yes)
edcc2fc
 
edcc2fc
@@ -98,6 +99,7 @@ PKG_CHECK_MODULES(SIMPLE_GREETER,
edcc2fc
         gtk+-2.0 >= $GTK_REQUIRED_VERSION
edcc2fc
         libglade-2.0 >= $LIBGLADE_REQUIRED_VERSION
edcc2fc
         gconf-2.0 >= $GCONF_REQUIRED_VERSION
edcc2fc
+	libxklavier >= $LIBXKLAVIER_REQUIRED_VERSION
edcc2fc
 )
edcc2fc
 AC_SUBST(SIMPLE_GREETER_CFLAGS)
edcc2fc
 AC_SUBST(SIMPLE_GREETER_LIBS)
edcc2fc
diff -up gdm-2.22.0/daemon/gdm-factory-slave.c.keyboard-chooser gdm-2.22.0/daemon/gdm-factory-slave.c
edcc2fc
--- gdm-2.22.0/daemon/gdm-factory-slave.c.keyboard-chooser	2008-04-17 23:29:28.000000000 -0400
edcc2fc
+++ gdm-2.22.0/daemon/gdm-factory-slave.c	2008-05-05 11:28:39.000000000 -0400
edcc2fc
@@ -416,6 +416,14 @@ on_greeter_language_selected (GdmGreeter
edcc2fc
 }
edcc2fc
 
edcc2fc
 static void
edcc2fc
+on_greeter_layout_selected (GdmGreeterServer *greeter_server,
edcc2fc
+                            const char       *text,
edcc2fc
+                            GdmFactorySlave  *slave)
edcc2fc
+{
edcc2fc
+        gdm_session_select_layout (GDM_SESSION (slave->priv->session), text);
edcc2fc
+}
edcc2fc
+
edcc2fc
+static void
edcc2fc
 on_greeter_user_selected (GdmGreeterServer *greeter_server,
edcc2fc
                           const char       *text,
edcc2fc
                           GdmFactorySlave  *slave)
edcc2fc
@@ -505,6 +513,10 @@ run_greeter (GdmFactorySlave *slave)
edcc2fc
                           G_CALLBACK (on_greeter_language_selected),
edcc2fc
                           slave);
edcc2fc
         g_signal_connect (slave->priv->greeter_server,
edcc2fc
+                          "layout-selected",
edcc2fc
+                          G_CALLBACK (on_greeter_layout_selected),
edcc2fc
+                          slave);
edcc2fc
+        g_signal_connect (slave->priv->greeter_server,
edcc2fc
                           "user-selected",
edcc2fc
                           G_CALLBACK (on_greeter_user_selected),
edcc2fc
                           slave);
edcc2fc
diff -up gdm-2.22.0/daemon/gdm-greeter-server.c.keyboard-chooser gdm-2.22.0/daemon/gdm-greeter-server.c
edcc2fc
--- gdm-2.22.0/daemon/gdm-greeter-server.c.keyboard-chooser	2008-04-17 23:29:28.000000000 -0400
edcc2fc
+++ gdm-2.22.0/daemon/gdm-greeter-server.c	2008-05-05 11:28:39.000000000 -0400
edcc2fc
@@ -76,6 +76,7 @@ enum {
edcc2fc
         SESSION_SELECTED,
edcc2fc
         HOSTNAME_SELECTED,
edcc2fc
         LANGUAGE_SELECTED,
edcc2fc
+        LAYOUT_SELECTED,
edcc2fc
         USER_SELECTED,
edcc2fc
         CANCELLED,
edcc2fc
         CONNECTED,
edcc2fc
@@ -266,6 +267,13 @@ gdm_greeter_server_default_language_name
edcc2fc
 }
edcc2fc
 
edcc2fc
 void
edcc2fc
+gdm_greeter_server_default_layout_name_changed (GdmGreeterServer *greeter_server,
edcc2fc
+                                                const char       *layout_name)
edcc2fc
+{
edcc2fc
+        send_dbus_string_signal (greeter_server, "DefaultLayoutNameChanged", layout_name);
edcc2fc
+}
edcc2fc
+
edcc2fc
+void
edcc2fc
 gdm_greeter_server_default_session_name_changed (GdmGreeterServer *greeter_server,
edcc2fc
                                                  const char       *session_name)
edcc2fc
 {
edcc2fc
@@ -496,6 +504,33 @@ handle_select_language (GdmGreeterServer
edcc2fc
 }
edcc2fc
 
edcc2fc
 static DBusHandlerResult
edcc2fc
+handle_select_layout (GdmGreeterServer *greeter_server,
edcc2fc
+                      DBusConnection  *connection,
edcc2fc
+                      DBusMessage     *message)
edcc2fc
+{
edcc2fc
+        DBusMessage *reply;
edcc2fc
+        DBusError    error;
edcc2fc
+        const char  *text;
edcc2fc
+
edcc2fc
+        dbus_error_init (&error);
edcc2fc
+        if (! dbus_message_get_args (message, &error,
edcc2fc
+                                     DBUS_TYPE_STRING, &text,
edcc2fc
+                                     DBUS_TYPE_INVALID)) {
edcc2fc
+                g_warning ("ERROR: %s", error.message);
edcc2fc
+        }
edcc2fc
+
edcc2fc
+        g_debug ("GreeterServer: SelectLayout: %s", text);
edcc2fc
+
edcc2fc
+        reply = dbus_message_new_method_return (message);
edcc2fc
+        dbus_connection_send (connection, reply, NULL);
edcc2fc
+        dbus_message_unref (reply);
edcc2fc
+
edcc2fc
+        g_signal_emit (greeter_server, signals [LAYOUT_SELECTED], 0, text);
edcc2fc
+
edcc2fc
+        return DBUS_HANDLER_RESULT_HANDLED;
edcc2fc
+}
edcc2fc
+
edcc2fc
+static DBusHandlerResult
edcc2fc
 handle_select_user (GdmGreeterServer *greeter_server,
edcc2fc
                     DBusConnection   *connection,
edcc2fc
                     DBusMessage      *message)
edcc2fc
@@ -624,6 +659,8 @@ greeter_handle_child_message (DBusConnec
edcc2fc
                 return handle_select_hostname (greeter_server, connection, message);
edcc2fc
         } else if (dbus_message_is_method_call (message, GDM_GREETER_SERVER_DBUS_INTERFACE, "SelectLanguage")) {
edcc2fc
                 return handle_select_language (greeter_server, connection, message);
edcc2fc
+        } else if (dbus_message_is_method_call (message, GDM_GREETER_SERVER_DBUS_INTERFACE, "SelectLayout")) {
edcc2fc
+                return handle_select_layout (greeter_server, connection, message);
edcc2fc
         } else if (dbus_message_is_method_call (message, GDM_GREETER_SERVER_DBUS_INTERFACE, "SelectUser")) {
edcc2fc
                 return handle_select_user (greeter_server, connection, message);
edcc2fc
         } else if (dbus_message_is_method_call (message, GDM_GREETER_SERVER_DBUS_INTERFACE, "Cancel")) {
edcc2fc
@@ -711,6 +748,10 @@ do_introspect (DBusConnection *connectio
edcc2fc
                                "    <signal name=\"DefaultLanguageNameChanged\">\n"
edcc2fc
                                "      <arg name=\"language_name\" type=\"s\"/>\n"
edcc2fc
                                "    </signal>\n"
edcc2fc
+                               "    <signal name=\"DefaultLayoutNameChanged\">\n"
edcc2fc
+                               "      <arg name=\"layout_name\" type=\"s\"/>\n"
edcc2fc
+                               "    </signal>\n"
edcc2fc
+                               "    <signal name=\"DefaultSessionNameChanged\">\n"
edcc2fc
                                "    <signal name=\"DefaultSessionNameChanged\">\n"
edcc2fc
                                "      <arg name=\"session_name\" type=\"s\"/>\n"
edcc2fc
                                "    </signal>\n"
edcc2fc
@@ -1157,6 +1198,17 @@ gdm_greeter_server_class_init (GdmGreete
edcc2fc
                               G_TYPE_NONE,
edcc2fc
                               1,
edcc2fc
                               G_TYPE_STRING);
edcc2fc
+        signals [LAYOUT_SELECTED] =
edcc2fc
+                g_signal_new ("layout-selected",
edcc2fc
+                              G_OBJECT_CLASS_TYPE (object_class),
edcc2fc
+                              G_SIGNAL_RUN_FIRST,
edcc2fc
+                              G_STRUCT_OFFSET (GdmGreeterServerClass, layout_selected),
edcc2fc
+                              NULL,
edcc2fc
+                              NULL,
edcc2fc
+                              g_cclosure_marshal_VOID__STRING,
edcc2fc
+                              G_TYPE_NONE,
edcc2fc
+                              1,
edcc2fc
+                              G_TYPE_STRING);
edcc2fc
         signals [USER_SELECTED] =
edcc2fc
                 g_signal_new ("user-selected",
edcc2fc
                               G_OBJECT_CLASS_TYPE (object_class),
edcc2fc
diff -up gdm-2.22.0/daemon/gdm-greeter-server.h.keyboard-chooser gdm-2.22.0/daemon/gdm-greeter-server.h
edcc2fc
--- gdm-2.22.0/daemon/gdm-greeter-server.h.keyboard-chooser	2008-04-17 23:29:28.000000000 -0400
edcc2fc
+++ gdm-2.22.0/daemon/gdm-greeter-server.h	2008-05-05 11:28:39.000000000 -0400
edcc2fc
@@ -57,6 +57,8 @@ typedef struct
edcc2fc
                                              const char        *hostname);
edcc2fc
         void (* language_selected)          (GdmGreeterServer  *greeter_server,
edcc2fc
                                              const char        *name);
edcc2fc
+        void (* layout_selected)            (GdmGreeterServer  *greeter_server,
edcc2fc
+                                             const char        *name);
edcc2fc
         void (* user_selected)              (GdmGreeterServer  *greeter_server,
edcc2fc
                                              const char        *name);
edcc2fc
         void (* cancelled)                  (GdmGreeterServer  *greeter_server);
edcc2fc
@@ -88,6 +90,8 @@ void                gdm_greeter_server_s
edcc2fc
                                                               const char       *text);
edcc2fc
 void                gdm_greeter_server_default_language_name_changed (GdmGreeterServer *greeter_server,
edcc2fc
                                                                       const char       *text);
edcc2fc
+void                gdm_greeter_server_default_layout_name_changed (GdmGreeterServer *greeter_server,
edcc2fc
+                                                                    const char       *text);
edcc2fc
 void                gdm_greeter_server_default_session_name_changed (GdmGreeterServer *greeter_server,
edcc2fc
                                                                      const char       *text);
edcc2fc
 
edcc2fc
diff -up gdm-2.22.0/daemon/gdm-product-slave.c.keyboard-chooser gdm-2.22.0/daemon/gdm-product-slave.c
edcc2fc
--- gdm-2.22.0/daemon/gdm-product-slave.c.keyboard-chooser	2008-04-17 23:29:28.000000000 -0400
edcc2fc
+++ gdm-2.22.0/daemon/gdm-product-slave.c	2008-05-05 11:28:39.000000000 -0400
edcc2fc
@@ -755,6 +755,28 @@ on_relay_language_selected (GdmProductSl
edcc2fc
 }
edcc2fc
 
edcc2fc
 static void
edcc2fc
+on_relay_layout_selected (GdmProductSlave *slave,
edcc2fc
+                          DBusMessage     *message)
edcc2fc
+{
edcc2fc
+        DBusError   error;
edcc2fc
+        const char *text;
edcc2fc
+        dbus_bool_t res;
edcc2fc
+
edcc2fc
+        dbus_error_init (&error);
edcc2fc
+        res = dbus_message_get_args (message,
edcc2fc
+                                     &error,
edcc2fc
+                                     DBUS_TYPE_STRING, &text,
edcc2fc
+                                     DBUS_TYPE_INVALID);
edcc2fc
+        if (res) {
edcc2fc
+                g_debug ("GdmProductSlave: Layout selected %s", text);
edcc2fc
+                gdm_session_select_layout (GDM_SESSION (slave->priv->session), text);
edcc2fc
+        } else {
edcc2fc
+                g_warning ("Unable to get arguments: %s", error.message);
edcc2fc
+                dbus_error_free (&error);
edcc2fc
+        }
edcc2fc
+}
edcc2fc
+
edcc2fc
+static void
edcc2fc
 on_relay_user_selected (GdmProductSlave *slave,
edcc2fc
                         DBusMessage     *message)
edcc2fc
 {
edcc2fc
@@ -963,6 +985,8 @@ relay_dbus_handle_message (DBusConnectio
edcc2fc
                 on_relay_session_selected (slave, message);
edcc2fc
         } else if (dbus_message_is_signal (message, RELAY_SERVER_DBUS_INTERFACE, "LanguageSelected")) {
edcc2fc
                 on_relay_language_selected (slave, message);
edcc2fc
+        } else if (dbus_message_is_signal (message, RELAY_SERVER_DBUS_INTERFACE, "LayoutSelected")) {
edcc2fc
+                on_relay_layout_selected (slave, message);
edcc2fc
         } else if (dbus_message_is_signal (message, RELAY_SERVER_DBUS_INTERFACE, "UserSelected")) {
edcc2fc
                 on_relay_user_selected (slave, message);
edcc2fc
         } else if (dbus_message_is_signal (message, RELAY_SERVER_DBUS_INTERFACE, "StartSession")) {
edcc2fc
diff -up gdm-2.22.0/daemon/gdm-session.c.keyboard-chooser gdm-2.22.0/daemon/gdm-session.c
edcc2fc
--- gdm-2.22.0/daemon/gdm-session.c.keyboard-chooser	2008-04-17 23:29:28.000000000 -0400
edcc2fc
+++ gdm-2.22.0/daemon/gdm-session.c	2008-05-05 11:28:39.000000000 -0400
edcc2fc
@@ -50,6 +50,7 @@ enum {
edcc2fc
         SESSION_DIED,
edcc2fc
         SELECTED_USER_CHANGED,
edcc2fc
         DEFAULT_LANGUAGE_NAME_CHANGED,
edcc2fc
+        DEFAULT_LAYOUT_NAME_CHANGED,
edcc2fc
         DEFAULT_SESSION_NAME_CHANGED,
edcc2fc
         LAST_SIGNAL
edcc2fc
 };
edcc2fc
@@ -164,6 +165,15 @@ gdm_session_select_language (GdmSession 
edcc2fc
 }
edcc2fc
 
edcc2fc
 void
edcc2fc
+gdm_session_select_layout (GdmSession *session,
edcc2fc
+                           const char *text)
edcc2fc
+{
edcc2fc
+        g_return_if_fail (GDM_IS_SESSION (session));
edcc2fc
+
edcc2fc
+        GDM_SESSION_GET_IFACE (session)->select_layout (session, text);
edcc2fc
+}
edcc2fc
+
edcc2fc
+void
edcc2fc
 gdm_session_select_user (GdmSession *session,
edcc2fc
                          const char *text)
edcc2fc
 {
edcc2fc
@@ -429,6 +439,17 @@ gdm_session_class_init (gpointer g_iface
edcc2fc
                               G_TYPE_NONE,
edcc2fc
                               1,
edcc2fc
                               G_TYPE_STRING);
edcc2fc
+        signals [DEFAULT_LAYOUT_NAME_CHANGED] =
edcc2fc
+                g_signal_new ("default-layout-name-changed",
edcc2fc
+                              iface_type,
edcc2fc
+                              G_SIGNAL_RUN_FIRST,
edcc2fc
+                              G_STRUCT_OFFSET (GdmSessionIface, default_layout_name_changed),
edcc2fc
+                              NULL,
edcc2fc
+                              NULL,
edcc2fc
+                              g_cclosure_marshal_VOID__STRING,
edcc2fc
+                              G_TYPE_NONE,
edcc2fc
+                              1,
edcc2fc
+                              G_TYPE_STRING);
edcc2fc
         signals [DEFAULT_SESSION_NAME_CHANGED] =
edcc2fc
                 g_signal_new ("default-session-name-changed",
edcc2fc
                               iface_type,
edcc2fc
@@ -610,6 +631,15 @@ _gdm_session_default_language_name_chang
edcc2fc
 }
edcc2fc
 
edcc2fc
 void
edcc2fc
+_gdm_session_default_layout_name_changed (GdmSession   *session,
edcc2fc
+                                          const char   *layout_name)
edcc2fc
+{
edcc2fc
+        g_return_if_fail (GDM_IS_SESSION (session));
edcc2fc
+
edcc2fc
+        g_signal_emit (session, signals [DEFAULT_LAYOUT_NAME_CHANGED], 0, layout_name);
edcc2fc
+}
edcc2fc
+
edcc2fc
+void
edcc2fc
 _gdm_session_default_session_name_changed (GdmSession   *session,
edcc2fc
                                            const char   *session_name)
edcc2fc
 {
edcc2fc
diff -up gdm-2.22.0/daemon/gdm-session-direct.c.keyboard-chooser gdm-2.22.0/daemon/gdm-session-direct.c
edcc2fc
--- gdm-2.22.0/daemon/gdm-session-direct.c.keyboard-chooser	2008-04-17 23:29:28.000000000 -0400
edcc2fc
+++ gdm-2.22.0/daemon/gdm-session-direct.c	2008-05-05 11:28:39.000000000 -0400
edcc2fc
@@ -70,6 +70,8 @@ struct _GdmSessionDirectPrivate
edcc2fc
         char                *saved_session;
edcc2fc
         char                *selected_language;
edcc2fc
         char                *saved_language;
edcc2fc
+        char                *selected_layout;
edcc2fc
+        char                *saved_layout;
edcc2fc
         char                *selected_user;
edcc2fc
         char                *user_x11_authority_file;
edcc2fc
 
edcc2fc
@@ -596,6 +598,16 @@ get_default_language_name (GdmSessionDir
edcc2fc
     return setlocale (LC_MESSAGES, NULL);
edcc2fc
 }
edcc2fc
 
edcc2fc
+static const char *
edcc2fc
+get_default_layout_name (GdmSessionDirect *session)
edcc2fc
+{
edcc2fc
+    if (session->priv->saved_layout != NULL) {
edcc2fc
+                return session->priv->saved_layout;
edcc2fc
+    }
edcc2fc
+
edcc2fc
+    return "us";
edcc2fc
+}
edcc2fc
+
edcc2fc
 static char *
edcc2fc
 get_fallback_session_name (void)
edcc2fc
 {
edcc2fc
@@ -661,6 +673,8 @@ gdm_session_direct_defaults_changed (Gdm
edcc2fc
 {
edcc2fc
         _gdm_session_default_language_name_changed (GDM_SESSION (session),
edcc2fc
                                                     get_default_language_name (session));
edcc2fc
+        _gdm_session_default_layout_name_changed (GDM_SESSION (session),
edcc2fc
+                                                  get_default_layout_name (session));
edcc2fc
         _gdm_session_default_session_name_changed (GDM_SESSION (session),
edcc2fc
                                                    get_default_session_name (session));
edcc2fc
 }
edcc2fc
@@ -681,6 +695,9 @@ gdm_session_direct_select_user (GdmSessi
edcc2fc
 
edcc2fc
         g_free (impl->priv->saved_language);
edcc2fc
         impl->priv->saved_language = NULL;
edcc2fc
+
edcc2fc
+        g_free (impl->priv->saved_layout);
edcc2fc
+        impl->priv->saved_layout = NULL;
edcc2fc
 }
edcc2fc
 
edcc2fc
 static DBusHandlerResult
edcc2fc
@@ -1035,6 +1052,39 @@ gdm_session_direct_handle_saved_language
edcc2fc
 }
edcc2fc
 
edcc2fc
 static DBusHandlerResult
edcc2fc
+gdm_session_direct_handle_saved_layout_name_read (GdmSessionDirect *session,
edcc2fc
+                                                  DBusConnection   *connection,
edcc2fc
+                                                  DBusMessage      *message)
edcc2fc
+{
edcc2fc
+        DBusMessage *reply;
edcc2fc
+        DBusError    error;
edcc2fc
+        const char  *layout_name;
edcc2fc
+
edcc2fc
+        dbus_error_init (&error);
edcc2fc
+        if (! dbus_message_get_args (message, &error,
edcc2fc
+                                     DBUS_TYPE_STRING, &layout_name,
edcc2fc
+                                     DBUS_TYPE_INVALID)) {
edcc2fc
+                g_warning ("ERROR: %s", error.message);
edcc2fc
+        }
edcc2fc
+
edcc2fc
+        reply = dbus_message_new_method_return (message);
edcc2fc
+        dbus_connection_send (connection, reply, NULL);
edcc2fc
+        dbus_message_unref (reply);
edcc2fc
+
edcc2fc
+        if (strcmp (layout_name,
edcc2fc
+                    get_default_layout_name (session)) != 0) {
edcc2fc
+                g_free (session->priv->saved_layout);
edcc2fc
+                session->priv->saved_layout = g_strdup (layout_name);
edcc2fc
+
edcc2fc
+                _gdm_session_default_layout_name_changed (GDM_SESSION (session),
edcc2fc
+                                                          layout_name);
edcc2fc
+        }
edcc2fc
+
edcc2fc
+
edcc2fc
+        return DBUS_HANDLER_RESULT_HANDLED;
edcc2fc
+}
edcc2fc
+
edcc2fc
+static DBusHandlerResult
edcc2fc
 gdm_session_direct_handle_saved_session_name_read (GdmSessionDirect *session,
edcc2fc
                                                    DBusConnection   *connection,
edcc2fc
                                                    DBusMessage      *message)
edcc2fc
@@ -1135,6 +1185,8 @@ session_worker_message (DBusConnection *
edcc2fc
                 return gdm_session_direct_handle_session_died (session, connection, message);
edcc2fc
         } else if (dbus_message_is_method_call (message, GDM_SESSION_DBUS_INTERFACE, "SavedLanguageNameRead")) {
edcc2fc
                 return gdm_session_direct_handle_saved_language_name_read (session, connection, message);
edcc2fc
+        } else if (dbus_message_is_method_call (message, GDM_SESSION_DBUS_INTERFACE, "SavedLayoutNameRead")) {
edcc2fc
+                return gdm_session_direct_handle_saved_layout_name_read (session, connection, message);
edcc2fc
         } else if (dbus_message_is_method_call (message, GDM_SESSION_DBUS_INTERFACE, "SavedSessionNameRead")) {
edcc2fc
                 return gdm_session_direct_handle_saved_session_name_read (session, connection, message);
edcc2fc
         }
edcc2fc
@@ -1834,6 +1886,16 @@ get_language_name (GdmSessionDirect *ses
edcc2fc
 }
edcc2fc
 
edcc2fc
 static const char *
edcc2fc
+get_layout_name (GdmSessionDirect *session)
edcc2fc
+{
edcc2fc
+        if (session->priv->selected_layout != NULL) {
edcc2fc
+                return session->priv->selected_layout;
edcc2fc
+        }
edcc2fc
+
edcc2fc
+        return get_default_layout_name (session);
edcc2fc
+}
edcc2fc
+
edcc2fc
+static const char *
edcc2fc
 get_session_name (GdmSessionDirect *session)
edcc2fc
 {
edcc2fc
         /* FIXME: test the session names before we use them? */
edcc2fc
@@ -1975,6 +2037,10 @@ setup_session_environment (GdmSessionDir
edcc2fc
                                                      get_language_name (session));
edcc2fc
 
edcc2fc
         gdm_session_direct_set_environment_variable (session,
edcc2fc
+                                                     "GDM_KEYBOARD_LAYOUT",
edcc2fc
+                                                     get_layout_name (session));
edcc2fc
+
edcc2fc
+        gdm_session_direct_set_environment_variable (session,
edcc2fc
                                                      "DISPLAY",
edcc2fc
                                                      session->priv->display_name);
edcc2fc
         if (session_cookie != NULL) {
edcc2fc
@@ -2063,6 +2129,12 @@ gdm_session_direct_close (GdmSession *se
edcc2fc
         g_free (impl->priv->saved_language);
edcc2fc
         impl->priv->saved_language = NULL;
edcc2fc
 
edcc2fc
+        g_free (impl->priv->selected_layout);
edcc2fc
+        impl->priv->selected_layout = NULL;
edcc2fc
+
edcc2fc
+        g_free (impl->priv->saved_layout);
edcc2fc
+        impl->priv->saved_layout = NULL;
edcc2fc
+
edcc2fc
         g_free (impl->priv->user_x11_authority_file);
edcc2fc
         impl->priv->user_x11_authority_file = NULL;
edcc2fc
 
edcc2fc
@@ -2139,6 +2211,24 @@ gdm_session_direct_select_language (GdmS
edcc2fc
 }
edcc2fc
 
edcc2fc
 static void
edcc2fc
+gdm_session_direct_select_layout (GdmSession *session,
edcc2fc
+                                  const char *text)
edcc2fc
+{
edcc2fc
+        GdmSessionDirect *impl = GDM_SESSION_DIRECT (session);
edcc2fc
+
edcc2fc
+        g_free (impl->priv->selected_layout);
edcc2fc
+
edcc2fc
+        if (strcmp (text, "__previous") == 0) {
edcc2fc
+                impl->priv->selected_layout = NULL;
edcc2fc
+        } else {
edcc2fc
+                impl->priv->selected_layout = g_strdup (text);
edcc2fc
+        }
edcc2fc
+
edcc2fc
+        send_dbus_string_signal (impl, "SetLayoutName",
edcc2fc
+                                 get_layout_name (impl));
edcc2fc
+}
edcc2fc
+
edcc2fc
+static void
edcc2fc
 _gdm_session_direct_set_display_id (GdmSessionDirect *session,
edcc2fc
                                     const char       *id)
edcc2fc
 {
edcc2fc
@@ -2330,6 +2420,8 @@ gdm_session_direct_finalize (GObject *ob
edcc2fc
         g_free (session->priv->saved_session);
edcc2fc
         g_free (session->priv->selected_language);
edcc2fc
         g_free (session->priv->saved_language);
edcc2fc
+        g_free (session->priv->selected_layout);
edcc2fc
+        g_free (session->priv->saved_layout);
edcc2fc
 
edcc2fc
         parent_class = G_OBJECT_CLASS (gdm_session_direct_parent_class);
edcc2fc
 
edcc2fc
@@ -2405,6 +2497,7 @@ gdm_session_iface_init (GdmSessionIface 
edcc2fc
         iface->answer_query = gdm_session_direct_answer_query;
edcc2fc
         iface->select_session = gdm_session_direct_select_session;
edcc2fc
         iface->select_language = gdm_session_direct_select_language;
edcc2fc
+        iface->select_layout = gdm_session_direct_select_layout;
edcc2fc
         iface->select_user = gdm_session_direct_select_user;
edcc2fc
 }
edcc2fc
 
edcc2fc
diff -up gdm-2.22.0/daemon/gdm-session.h.keyboard-chooser gdm-2.22.0/daemon/gdm-session.h
edcc2fc
--- gdm-2.22.0/daemon/gdm-session.h.keyboard-chooser	2008-04-17 23:29:28.000000000 -0400
edcc2fc
+++ gdm-2.22.0/daemon/gdm-session.h	2008-05-05 11:28:39.000000000 -0400
edcc2fc
@@ -60,6 +60,8 @@ struct _GdmSessionIface
edcc2fc
                                               const char   *text);
edcc2fc
         void (* select_language)             (GdmSession   *session,
edcc2fc
                                               const char   *text);
edcc2fc
+        void (* select_layout)               (GdmSession   *session,
edcc2fc
+                                              const char   *text);
edcc2fc
         void (* select_session)              (GdmSession   *session,
edcc2fc
                                               const char   *text);
edcc2fc
         void (* select_user)                 (GdmSession   *session,
edcc2fc
@@ -108,6 +110,8 @@ struct _GdmSessionIface
edcc2fc
 
edcc2fc
         void (* default_language_name_changed)    (GdmSession   *session,
edcc2fc
                                                    const char   *text);
edcc2fc
+        void (* default_layout_name_changed)      (GdmSession   *session,
edcc2fc
+                                                   const char   *text);
edcc2fc
         void (* default_session_name_changed)     (GdmSession   *session,
edcc2fc
                                                    const char   *text);
edcc2fc
 };
edcc2fc
@@ -134,6 +138,8 @@ void     gdm_session_select_session     
edcc2fc
                                                   const char *session_name);
edcc2fc
 void     gdm_session_select_language             (GdmSession *session,
edcc2fc
                                                   const char *language);
edcc2fc
+void     gdm_session_select_layout               (GdmSession *session,
edcc2fc
+                                                  const char *language);
edcc2fc
 void     gdm_session_select_user                 (GdmSession *session,
edcc2fc
                                                   const char *username);
edcc2fc
 void     gdm_session_cancel                      (GdmSession *session);
edcc2fc
diff -up gdm-2.22.0/daemon/gdm-session-private.h.keyboard-chooser gdm-2.22.0/daemon/gdm-session-private.h
edcc2fc
--- gdm-2.22.0/daemon/gdm-session-private.h.keyboard-chooser	2008-04-17 23:29:28.000000000 -0400
edcc2fc
+++ gdm-2.22.0/daemon/gdm-session-private.h	2008-05-05 11:28:39.000000000 -0400
edcc2fc
@@ -56,6 +56,8 @@ void             _gdm_session_closed    
edcc2fc
 /* user settings read from ~/.dmrc / system defaults */
edcc2fc
 void             _gdm_session_default_language_name_changed     (GdmSession   *session,
edcc2fc
                                                                  const char   *language_name);
edcc2fc
+void             _gdm_session_default_layout_name_changed       (GdmSession   *session,
edcc2fc
+                                                                 const char   *layout_name);
edcc2fc
 void             _gdm_session_default_session_name_changed      (GdmSession   *session,
edcc2fc
                                                                  const char   *session_name);
edcc2fc
 /* user is selected/changed internally */
edcc2fc
diff -up gdm-2.22.0/daemon/gdm-session-relay.c.keyboard-chooser gdm-2.22.0/daemon/gdm-session-relay.c
edcc2fc
--- gdm-2.22.0/daemon/gdm-session-relay.c.keyboard-chooser	2008-04-17 23:29:28.000000000 -0400
edcc2fc
+++ gdm-2.22.0/daemon/gdm-session-relay.c	2008-05-05 11:28:39.000000000 -0400
edcc2fc
@@ -267,6 +267,14 @@ gdm_session_relay_select_language (GdmSe
edcc2fc
 }
edcc2fc
 
edcc2fc
 static void
edcc2fc
+gdm_session_relay_select_layout (GdmSession *session,
edcc2fc
+                                 const char *text)
edcc2fc
+{
edcc2fc
+        GdmSessionRelay *impl = GDM_SESSION_RELAY (session);
edcc2fc
+        send_dbus_string_signal (impl, "LayoutSelected", text);
edcc2fc
+}
edcc2fc
+
edcc2fc
+static void
edcc2fc
 gdm_session_relay_select_user (GdmSession *session,
edcc2fc
                                const char *text)
edcc2fc
 {
edcc2fc
@@ -814,6 +822,9 @@ do_introspect (DBusConnection *connectio
edcc2fc
                                "    <signal name=\"LanguageSelected\">\n"
edcc2fc
                                "      <arg name=\"language\" type=\"s\"/>\n"
edcc2fc
                                "    </signal>\n"
edcc2fc
+                               "    <signal name=\"LayoutSelected\">\n"
edcc2fc
+                               "      <arg name=\"layout\" type=\"s\"/>\n"
edcc2fc
+                               "    </signal>\n"
edcc2fc
                                "    <signal name=\"SessionSelected\">\n"
edcc2fc
                                "      <arg name=\"session\" type=\"s\"/>\n"
edcc2fc
                                "    </signal>\n"
edcc2fc
@@ -1108,6 +1119,7 @@ gdm_session_iface_init (GdmSessionIface 
edcc2fc
         iface->answer_query = gdm_session_relay_answer_query;
edcc2fc
         iface->select_session = gdm_session_relay_select_session;
edcc2fc
         iface->select_language = gdm_session_relay_select_language;
edcc2fc
+        iface->select_layout = gdm_session_relay_select_layout;
edcc2fc
         iface->select_user = gdm_session_relay_select_user;
edcc2fc
 }
edcc2fc
 
edcc2fc
diff -up gdm-2.22.0/daemon/gdm-session-settings.c.keyboard-chooser gdm-2.22.0/daemon/gdm-session-settings.c
edcc2fc
--- gdm-2.22.0/daemon/gdm-session-settings.c.keyboard-chooser	2008-04-17 23:29:28.000000000 -0400
edcc2fc
+++ gdm-2.22.0/daemon/gdm-session-settings.c	2008-05-05 11:28:39.000000000 -0400
edcc2fc
@@ -36,6 +36,7 @@ struct _GdmSessionSettingsPrivate
edcc2fc
 {
edcc2fc
         char *session_name;
edcc2fc
         char *language_name;
edcc2fc
+        char *layout_name;
edcc2fc
 };
edcc2fc
 
edcc2fc
 static void gdm_session_settings_finalize (GObject *object);
edcc2fc
@@ -55,6 +56,7 @@ enum {
edcc2fc
         PROP_0 = 0,
edcc2fc
         PROP_SESSION_NAME,
edcc2fc
         PROP_LANGUAGE_NAME,
edcc2fc
+        PROP_LAYOUT_NAME,
edcc2fc
 };
edcc2fc
 
edcc2fc
 G_DEFINE_TYPE (GdmSessionSettings, gdm_session_settings, G_TYPE_OBJECT)
edcc2fc
@@ -93,6 +95,11 @@ gdm_session_settings_class_install_prope
edcc2fc
                                         NULL,
edcc2fc
                                        G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY);
edcc2fc
         g_object_class_install_property (object_class, PROP_LANGUAGE_NAME, param_spec);
edcc2fc
+        param_spec = g_param_spec_string ("layout-name", "Keyboard Layout Name",
edcc2fc
+                                        "The name of the keyboard layout",
edcc2fc
+                                        NULL,
edcc2fc
+                                       G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY);
edcc2fc
+        g_object_class_install_property (object_class, PROP_LAYOUT_NAME, param_spec);
edcc2fc
 }
edcc2fc
 
edcc2fc
 static void
edcc2fc
@@ -114,6 +121,7 @@ gdm_session_settings_finalize (GObject *
edcc2fc
 
edcc2fc
         g_free (settings->priv->session_name);
edcc2fc
         g_free (settings->priv->language_name);
edcc2fc
+        g_free (settings->priv->layout_name);
edcc2fc
 
edcc2fc
         parent_class = G_OBJECT_CLASS (gdm_session_settings_parent_class);
edcc2fc
 
edcc2fc
@@ -136,6 +144,19 @@ gdm_session_settings_set_language_name (
edcc2fc
 }
edcc2fc
 
edcc2fc
 void
edcc2fc
+gdm_session_settings_set_layout_name (GdmSessionSettings *settings,
edcc2fc
+                                      const char         *layout_name)
edcc2fc
+{
edcc2fc
+        g_return_if_fail (GDM_IS_SESSION_SETTINGS (settings));
edcc2fc
+
edcc2fc
+        if (settings->priv->layout_name == NULL ||
edcc2fc
+            strcmp (settings->priv->layout_name, layout_name) != 0) {
edcc2fc
+                settings->priv->layout_name = g_strdup (layout_name);
edcc2fc
+                g_object_notify (G_OBJECT (settings), "layout-name");
edcc2fc
+        }
edcc2fc
+}
edcc2fc
+
edcc2fc
+void
edcc2fc
 gdm_session_settings_set_session_name (GdmSessionSettings *settings,
edcc2fc
                                        const char         *session_name)
edcc2fc
 {
edcc2fc
@@ -156,6 +177,13 @@ gdm_session_settings_get_language_name (
edcc2fc
 }
edcc2fc
 
edcc2fc
 char *
edcc2fc
+gdm_session_settings_get_layout_name (GdmSessionSettings *settings)
edcc2fc
+{
edcc2fc
+        g_return_val_if_fail (GDM_IS_SESSION_SETTINGS (settings), NULL);
edcc2fc
+        return g_strdup (settings->priv->layout_name);
edcc2fc
+}
edcc2fc
+
edcc2fc
+char *
edcc2fc
 gdm_session_settings_get_session_name (GdmSessionSettings *settings)
edcc2fc
 {
edcc2fc
         g_return_val_if_fail (GDM_IS_SESSION_SETTINGS (settings), NULL);
edcc2fc
@@ -177,6 +205,10 @@ gdm_session_settings_set_property (GObje
edcc2fc
                         gdm_session_settings_set_language_name (settings, g_value_get_string (value));
edcc2fc
                 break;
edcc2fc
 
edcc2fc
+                case PROP_LAYOUT_NAME:
edcc2fc
+                        gdm_session_settings_set_layout_name (settings, g_value_get_string (value));
edcc2fc
+                break;
edcc2fc
+
edcc2fc
                 case PROP_SESSION_NAME:
edcc2fc
                         gdm_session_settings_set_session_name (settings, g_value_get_string (value));
edcc2fc
                 break;
edcc2fc
@@ -205,6 +237,10 @@ gdm_session_settings_get_property (GObje
edcc2fc
                         g_value_set_string (value, settings->priv->language_name);
edcc2fc
                 break;
edcc2fc
 
edcc2fc
+                case PROP_LAYOUT_NAME:
edcc2fc
+                        g_value_set_string (value, settings->priv->layout_name);
edcc2fc
+                break;
edcc2fc
+
edcc2fc
                 default:
edcc2fc
                         G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
edcc2fc
     }
edcc2fc
@@ -225,7 +261,8 @@ gboolean
edcc2fc
 gdm_session_settings_is_loaded (GdmSessionSettings  *settings)
edcc2fc
 {
edcc2fc
         return settings->priv->session_name != NULL ||
edcc2fc
-               settings->priv->language_name != NULL;
edcc2fc
+               settings->priv->language_name != NULL ||
edcc2fc
+               settings->priv->layout_name != NULL;
edcc2fc
 }
edcc2fc
 
edcc2fc
 gboolean
edcc2fc
@@ -238,6 +275,7 @@ gdm_session_settings_load (GdmSessionSet
edcc2fc
         gboolean  is_loaded;
edcc2fc
         char     *session_name;
edcc2fc
         char     *language_name;
edcc2fc
+        char     *layout_name;
edcc2fc
         char     *filename;
edcc2fc
 
edcc2fc
         g_return_val_if_fail (settings != NULL, FALSE);
edcc2fc
@@ -283,6 +321,20 @@ gdm_session_settings_load (GdmSessionSet
edcc2fc
                 goto out;
edcc2fc
         }
edcc2fc
 
edcc2fc
+        layout_name = g_key_file_get_string (key_file, "Desktop", "Layout",
edcc2fc
+                                             &load_error);
edcc2fc
+
edcc2fc
+        if (layout_name != NULL) {
edcc2fc
+                gdm_session_settings_set_layout_name (settings, layout_name);
edcc2fc
+                g_free (layout_name);
edcc2fc
+        } else if (g_error_matches (load_error, G_KEY_FILE_ERROR, G_KEY_FILE_ERROR_KEY_NOT_FOUND)) {
edcc2fc
+                g_error_free (load_error);
edcc2fc
+                load_error = NULL;
edcc2fc
+        } else {
edcc2fc
+                g_propagate_error (error, load_error);
edcc2fc
+                goto out;
edcc2fc
+        }
edcc2fc
+
edcc2fc
         is_loaded = TRUE;
edcc2fc
 out:
edcc2fc
         g_key_file_free (key_file);
edcc2fc
@@ -327,6 +379,11 @@ gdm_session_settings_save (GdmSessionSet
edcc2fc
                                        settings->priv->language_name);
edcc2fc
         }
edcc2fc
 
edcc2fc
+        if (settings->priv->layout_name != NULL) {
edcc2fc
+                g_key_file_set_string (key_file, "Desktop", "Layout",
edcc2fc
+                                       settings->priv->layout_name);
edcc2fc
+        }
edcc2fc
+
edcc2fc
         contents = g_key_file_to_data (key_file, &length, &file_error);
edcc2fc
 
edcc2fc
         if (contents == NULL) {
edcc2fc
diff -up gdm-2.22.0/daemon/gdm-session-settings.h.keyboard-chooser gdm-2.22.0/daemon/gdm-session-settings.h
edcc2fc
--- gdm-2.22.0/daemon/gdm-session-settings.h.keyboard-chooser	2008-04-17 23:29:28.000000000 -0400
edcc2fc
+++ gdm-2.22.0/daemon/gdm-session-settings.h	2008-05-05 11:28:39.000000000 -0400
edcc2fc
@@ -61,9 +61,12 @@ gboolean            gdm_session_settings
edcc2fc
                                                              GError             **error);
edcc2fc
 gboolean            gdm_session_settings_is_loaded          (GdmSessionSettings  *settings);
edcc2fc
 char               *gdm_session_settings_get_language_name  (GdmSessionSettings *settings);
edcc2fc
+char               *gdm_session_settings_get_layout_name    (GdmSessionSettings *settings);
edcc2fc
 char               *gdm_session_settings_get_session_name   (GdmSessionSettings *settings);
edcc2fc
 void                gdm_session_settings_set_language_name  (GdmSessionSettings *settings,
edcc2fc
                                                              const char         *language_name);
edcc2fc
+void                gdm_session_settings_set_layout_name    (GdmSessionSettings *settings,
edcc2fc
+                                                             const char         *layout_name);
edcc2fc
 void                gdm_session_settings_set_session_name   (GdmSessionSettings *settings,
edcc2fc
                                                              const char         *session_name);
edcc2fc
 
edcc2fc
diff -up gdm-2.22.0/daemon/gdm-session-worker.c.keyboard-chooser gdm-2.22.0/daemon/gdm-session-worker.c
edcc2fc
--- gdm-2.22.0/daemon/gdm-session-worker.c.keyboard-chooser	2008-04-17 23:29:28.000000000 -0400
edcc2fc
+++ gdm-2.22.0/daemon/gdm-session-worker.c	2008-05-05 11:28:39.000000000 -0400
edcc2fc
@@ -1785,6 +1785,14 @@ gdm_session_worker_set_language_name (Gd
edcc2fc
 }
edcc2fc
 
edcc2fc
 static void
edcc2fc
+gdm_session_worker_set_layout_name (GdmSessionWorker *worker,
edcc2fc
+                                    const char       *layout_name)
edcc2fc
+{
edcc2fc
+        gdm_session_settings_set_layout_name (worker->priv->user_settings,
edcc2fc
+                                              layout_name);
edcc2fc
+}
edcc2fc
+
edcc2fc
+static void
edcc2fc
 on_set_language_name (GdmSessionWorker *worker,
edcc2fc
                       DBusMessage      *message)
edcc2fc
 {
edcc2fc
@@ -1807,6 +1815,28 @@ on_set_language_name (GdmSessionWorker *
edcc2fc
 }
edcc2fc
 
edcc2fc
 static void
edcc2fc
+on_set_layout_name (GdmSessionWorker *worker,
edcc2fc
+                    DBusMessage      *message)
edcc2fc
+{
edcc2fc
+        DBusError   error;
edcc2fc
+        const char *layout_name;
edcc2fc
+        dbus_bool_t res;
edcc2fc
+
edcc2fc
+        dbus_error_init (&error);
edcc2fc
+        res = dbus_message_get_args (message,
edcc2fc
+                                     &error,
edcc2fc
+                                     DBUS_TYPE_STRING, &layout_name,
edcc2fc
+                                     DBUS_TYPE_INVALID);
edcc2fc
+        if (res) {
edcc2fc
+                g_debug ("GdmSessionWorker: layout name set to %s", layout_name);
edcc2fc
+                gdm_session_worker_set_layout_name (worker, layout_name);
edcc2fc
+        } else {
edcc2fc
+                g_warning ("Unable to get arguments: %s", error.message);
edcc2fc
+                dbus_error_free (&error);
edcc2fc
+        }
edcc2fc
+}
edcc2fc
+
edcc2fc
+static void
edcc2fc
 on_saved_language_name_read (GdmSessionWorker *worker)
edcc2fc
 {
edcc2fc
         char *language_name;
edcc2fc
@@ -1819,6 +1849,18 @@ on_saved_language_name_read (GdmSessionW
edcc2fc
 }
edcc2fc
 
edcc2fc
 static void
edcc2fc
+on_saved_layout_name_read (GdmSessionWorker *worker)
edcc2fc
+{
edcc2fc
+        char *layout_name;
edcc2fc
+
edcc2fc
+        layout_name = gdm_session_settings_get_layout_name (worker->priv->user_settings);
edcc2fc
+        send_dbus_string_method (worker->priv->connection,
edcc2fc
+                                 "SavedLayoutNameRead",
edcc2fc
+                                 layout_name);
edcc2fc
+        g_free (layout_name);
edcc2fc
+}
edcc2fc
+
edcc2fc
+static void
edcc2fc
 on_saved_session_name_read (GdmSessionWorker *worker)
edcc2fc
 {
edcc2fc
         char *session_name;
edcc2fc
@@ -1844,6 +1886,11 @@ do_setup (GdmSessionWorker *worker)
edcc2fc
                                   worker);
edcc2fc
 
edcc2fc
         g_signal_connect_swapped (worker->priv->user_settings,
edcc2fc
+                                  "notify::layout-name",
edcc2fc
+                                  G_CALLBACK (on_saved_layout_name_read),
edcc2fc
+                                  worker);
edcc2fc
+
edcc2fc
+        g_signal_connect_swapped (worker->priv->user_settings,
edcc2fc
                                   "notify::session-name",
edcc2fc
                                   G_CALLBACK (on_saved_session_name_read),
edcc2fc
                                   worker);
edcc2fc
@@ -2300,6 +2347,8 @@ worker_dbus_handle_message (DBusConnecti
edcc2fc
                 on_set_environment_variable (worker, message);
edcc2fc
         } else if (dbus_message_is_signal (message, GDM_SESSION_DBUS_INTERFACE, "SetLanguageName")) {
edcc2fc
                 on_set_language_name (worker, message);
edcc2fc
+        } else if (dbus_message_is_signal (message, GDM_SESSION_DBUS_INTERFACE, "SetLayoutName")) {
edcc2fc
+                on_set_layout_name (worker, message);
edcc2fc
         } else if (dbus_message_is_signal (message, GDM_SESSION_DBUS_INTERFACE, "SetSessionName")) {
edcc2fc
                 on_set_session_name (worker, message);
edcc2fc
         } else {
edcc2fc
diff -up gdm-2.22.0/daemon/gdm-simple-slave.c.keyboard-chooser gdm-2.22.0/daemon/gdm-simple-slave.c
edcc2fc
--- gdm-2.22.0/daemon/gdm-simple-slave.c.keyboard-chooser	2008-05-01 18:35:36.000000000 -0400
edcc2fc
+++ gdm-2.22.0/daemon/gdm-simple-slave.c	2008-05-05 11:28:39.000000000 -0400
edcc2fc
@@ -485,6 +485,16 @@ on_default_language_name_changed (GdmSes
edcc2fc
 }
edcc2fc
 
edcc2fc
 static void
edcc2fc
+on_default_layout_name_changed (GdmSession     *session,
edcc2fc
+                                const char     *text,
edcc2fc
+                                GdmSimpleSlave *slave)
edcc2fc
+{
edcc2fc
+        g_debug ("GdmSimpleSlave: Default layout name changed: %s", text);
edcc2fc
+
edcc2fc
+        gdm_greeter_server_default_layout_name_changed (slave->priv->greeter_server, text);
edcc2fc
+}
edcc2fc
+
edcc2fc
+static void
edcc2fc
 on_default_session_name_changed (GdmSession     *session,
edcc2fc
                                  const char     *text,
edcc2fc
                                  GdmSimpleSlave *slave)
edcc2fc
@@ -619,6 +629,11 @@ create_new_session (GdmSimpleSlave *slav
edcc2fc
                           slave);
edcc2fc
 
edcc2fc
         g_signal_connect (slave->priv->session,
edcc2fc
+                          "default-layout-name-changed",
edcc2fc
+                          G_CALLBACK (on_default_layout_name_changed),
edcc2fc
+                          slave);
edcc2fc
+
edcc2fc
+        g_signal_connect (slave->priv->session,
edcc2fc
                           "default-session-name-changed",
edcc2fc
                           G_CALLBACK (on_default_session_name_changed),
edcc2fc
                           slave);
edcc2fc
@@ -713,6 +728,14 @@ on_greeter_language_selected (GdmGreeter
edcc2fc
 }
edcc2fc
 
edcc2fc
 static void
edcc2fc
+on_greeter_layout_selected (GdmGreeterServer *greeter_server,
edcc2fc
+                            const char       *text,
edcc2fc
+                            GdmSimpleSlave   *slave)
edcc2fc
+{
edcc2fc
+        gdm_session_select_layout (GDM_SESSION (slave->priv->session), text);
edcc2fc
+}
edcc2fc
+
edcc2fc
+static void
edcc2fc
 on_greeter_user_selected (GdmGreeterServer *greeter_server,
edcc2fc
                           const char       *text,
edcc2fc
                           GdmSimpleSlave   *slave)
edcc2fc
@@ -847,6 +870,10 @@ run_greeter (GdmSimpleSlave *slave)
edcc2fc
                           G_CALLBACK (on_greeter_language_selected),
edcc2fc
                           slave);
edcc2fc
         g_signal_connect (slave->priv->greeter_server,
edcc2fc
+                          "layout-selected",
edcc2fc
+                          G_CALLBACK (on_greeter_layout_selected),
edcc2fc
+                          slave);
edcc2fc
+        g_signal_connect (slave->priv->greeter_server,
edcc2fc
                           "user-selected",
edcc2fc
                           G_CALLBACK (on_greeter_user_selected),
edcc2fc
                           slave);
edcc2fc
diff -up gdm-2.22.0/gui/simple-greeter/gdm-greeter-client.c.keyboard-chooser gdm-2.22.0/gui/simple-greeter/gdm-greeter-client.c
edcc2fc
--- gdm-2.22.0/gui/simple-greeter/gdm-greeter-client.c.keyboard-chooser	2008-04-17 23:29:26.000000000 -0400
edcc2fc
+++ gdm-2.22.0/gui/simple-greeter/gdm-greeter-client.c	2008-05-05 11:28:39.000000000 -0400
edcc2fc
@@ -67,6 +67,7 @@ enum {
edcc2fc
         RESET,
edcc2fc
         SELECTED_USER_CHANGED,
edcc2fc
         DEFAULT_LANGUAGE_NAME_CHANGED,
edcc2fc
+        DEFAULT_LAYOUT_NAME_CHANGED,
edcc2fc
         DEFAULT_SESSION_NAME_CHANGED,
edcc2fc
         TIMED_LOGIN_REQUESTED,
edcc2fc
         USER_AUTHORIZED,
edcc2fc
@@ -175,6 +176,13 @@ on_default_language_name_changed (GdmGre
edcc2fc
 }
edcc2fc
 
edcc2fc
 static void
edcc2fc
+on_default_layout_name_changed (GdmGreeterClient *client,
edcc2fc
+                                DBusMessage      *message)
edcc2fc
+{
edcc2fc
+        emit_string_signal_for_message (client, "DefaultLayoutNameChanged", message, DEFAULT_LAYOUT_NAME_CHANGED);
edcc2fc
+}
edcc2fc
+
edcc2fc
+static void
edcc2fc
 on_default_session_name_changed (GdmGreeterClient *client,
edcc2fc
                                  DBusMessage      *message)
edcc2fc
 {
edcc2fc
@@ -452,6 +460,14 @@ gdm_greeter_client_call_select_language 
edcc2fc
 }
edcc2fc
 
edcc2fc
 void
edcc2fc
+gdm_greeter_client_call_select_layout (GdmGreeterClient *client,
edcc2fc
+                                       const char       *text)
edcc2fc
+{
edcc2fc
+        send_dbus_string_method (client->priv->connection,
edcc2fc
+                                 "SelectLayout",
edcc2fc
+                                 text);
edcc2fc
+}
edcc2fc
+void
edcc2fc
 gdm_greeter_client_call_select_user (GdmGreeterClient *client,
edcc2fc
                                      const char       *text)
edcc2fc
 {
edcc2fc
@@ -633,6 +649,8 @@ client_dbus_handle_message (DBusConnecti
edcc2fc
                 on_selected_user_changed (client, message);
edcc2fc
         } else if (dbus_message_is_signal (message, GREETER_SERVER_DBUS_INTERFACE, "DefaultLanguageNameChanged")) {
edcc2fc
                 on_default_language_name_changed (client, message);
edcc2fc
+        } else if (dbus_message_is_signal (message, GREETER_SERVER_DBUS_INTERFACE, "DefaultLayoutNameChanged")) {
edcc2fc
+                on_default_layout_name_changed (client, message);
edcc2fc
         } else if (dbus_message_is_signal (message, GREETER_SERVER_DBUS_INTERFACE, "DefaultSessionNameChanged")) {
edcc2fc
                 on_default_session_name_changed (client, message);
edcc2fc
         } else if (dbus_message_is_signal (message, GREETER_SERVER_DBUS_INTERFACE, "TimedLoginRequested")) {
edcc2fc
@@ -895,6 +913,16 @@ gdm_greeter_client_class_init (GdmGreete
edcc2fc
                               g_cclosure_marshal_VOID__STRING,
edcc2fc
                               G_TYPE_NONE,
edcc2fc
                               1, G_TYPE_STRING);
edcc2fc
+        gdm_greeter_client_signals[DEFAULT_LAYOUT_NAME_CHANGED] =
edcc2fc
+                g_signal_new ("default-layout-name-changed",
edcc2fc
+                              G_OBJECT_CLASS_TYPE (object_class),
edcc2fc
+                              G_SIGNAL_RUN_FIRST,
edcc2fc
+                              G_STRUCT_OFFSET (GdmGreeterClientClass, default_layout_name_changed),
edcc2fc
+                              NULL,
edcc2fc
+                              NULL,
edcc2fc
+                              g_cclosure_marshal_VOID__STRING,
edcc2fc
+                              G_TYPE_NONE,
edcc2fc
+                              1, G_TYPE_STRING);
edcc2fc
         gdm_greeter_client_signals[DEFAULT_SESSION_NAME_CHANGED] =
edcc2fc
                 g_signal_new ("default-session-name-changed",
edcc2fc
                               G_OBJECT_CLASS_TYPE (object_class),
edcc2fc
diff -up gdm-2.22.0/gui/simple-greeter/gdm-greeter-client.h.keyboard-chooser gdm-2.22.0/gui/simple-greeter/gdm-greeter-client.h
edcc2fc
--- gdm-2.22.0/gui/simple-greeter/gdm-greeter-client.h.keyboard-chooser	2008-04-17 23:29:26.000000000 -0400
edcc2fc
+++ gdm-2.22.0/gui/simple-greeter/gdm-greeter-client.h	2008-05-05 11:28:39.000000000 -0400
edcc2fc
@@ -64,6 +64,8 @@ typedef struct
edcc2fc
                                                 const char        *session_name);
edcc2fc
         void (* default_language_name_changed) (GdmGreeterClient  *client,
edcc2fc
                                                 const char        *language_name);
edcc2fc
+        void (* default_layout_name_changed) (GdmGreeterClient  *client,
edcc2fc
+                                              const char        *layout_name);
edcc2fc
         void (* timed_login_requested)   (GdmGreeterClient  *client,
edcc2fc
                                           const char        *username,
edcc2fc
                                           int                delay);
edcc2fc
@@ -102,6 +104,8 @@ void               gdm_greeter_client_ca
edcc2fc
                                                                       const char       *text);
edcc2fc
 void               gdm_greeter_client_call_select_language           (GdmGreeterClient *client,
edcc2fc
                                                                       const char       *text);
edcc2fc
+void               gdm_greeter_client_call_select_layout             (GdmGreeterClient *client,
edcc2fc
+                                                                      const char       *text);
edcc2fc
 void               gdm_greeter_client_call_select_session            (GdmGreeterClient *client,
edcc2fc
                                                                       const char       *text);
edcc2fc
 void               gdm_greeter_client_call_answer_query              (GdmGreeterClient *client,
edcc2fc
diff -up gdm-2.22.0/gui/simple-greeter/gdm-greeter-panel.c.keyboard-chooser gdm-2.22.0/gui/simple-greeter/gdm-greeter-panel.c
edcc2fc
--- gdm-2.22.0/gui/simple-greeter/gdm-greeter-panel.c.keyboard-chooser	2008-04-29 13:38:43.000000000 -0400
edcc2fc
+++ gdm-2.22.0/gui/simple-greeter/gdm-greeter-panel.c	2008-05-05 11:28:39.000000000 -0400
edcc2fc
@@ -34,9 +34,11 @@
edcc2fc
 #include <gconf/gconf-client.h>
edcc2fc
 
edcc2fc
 #include "gdm-languages.h"
edcc2fc
+#include "gdm-layouts.h"
edcc2fc
 #include "gdm-greeter-panel.h"
edcc2fc
 #include "gdm-clock-widget.h"
edcc2fc
 #include "gdm-language-option-widget.h"
edcc2fc
+#include "gdm-layout-option-widget.h"
edcc2fc
 #include "gdm-session-option-widget.h"
edcc2fc
 #include "gdm-a11y-preferences-dialog.h"
edcc2fc
 #include "gdm-profile.h"
edcc2fc
@@ -59,6 +61,7 @@ struct GdmGreeterPanelPrivate
edcc2fc
         GtkWidget              *hostname_label;
edcc2fc
         GtkWidget              *clock;
edcc2fc
         GtkWidget              *language_option_widget;
edcc2fc
+        GtkWidget              *layout_option_widget;
edcc2fc
         GtkWidget              *session_option_widget;
edcc2fc
 
edcc2fc
         char                   *default_session_name;
edcc2fc
@@ -71,6 +74,7 @@ enum {
edcc2fc
 
edcc2fc
 enum {
edcc2fc
         LANGUAGE_SELECTED,
edcc2fc
+        LAYOUT_SELECTED,
edcc2fc
         SESSION_SELECTED,
edcc2fc
         NUMBER_OF_SIGNALS
edcc2fc
 };
edcc2fc
@@ -398,6 +402,17 @@ gdm_greeter_panel_class_init (GdmGreeter
edcc2fc
                               G_TYPE_NONE,
edcc2fc
                               1, G_TYPE_STRING);
edcc2fc
 
edcc2fc
+        signals[LAYOUT_SELECTED] =
edcc2fc
+                g_signal_new ("layout-selected",
edcc2fc
+                              G_TYPE_FROM_CLASS (object_class),
edcc2fc
+                              G_SIGNAL_RUN_LAST,
edcc2fc
+                              G_STRUCT_OFFSET (GdmGreeterPanelClass, layout_selected),
edcc2fc
+                              NULL,
edcc2fc
+                              NULL,
edcc2fc
+                              g_cclosure_marshal_VOID__STRING,
edcc2fc
+                              G_TYPE_NONE,
edcc2fc
+                              1, G_TYPE_STRING);
edcc2fc
+
edcc2fc
         signals[SESSION_SELECTED] =
edcc2fc
                 g_signal_new ("session-selected",
edcc2fc
                               G_TYPE_FROM_CLASS (object_class),
edcc2fc
@@ -431,6 +446,26 @@ on_language_activated (GdmLanguageOption
edcc2fc
 }
edcc2fc
 
edcc2fc
 static void
edcc2fc
+on_layout_activated (GdmLayoutOptionWidget *widget,
edcc2fc
+                     GdmGreeterPanel       *panel)
edcc2fc
+{
edcc2fc
+
edcc2fc
+        char *layout;
edcc2fc
+
edcc2fc
+        layout = gdm_layout_option_widget_get_current_layout_name (GDM_LAYOUT_OPTION_WIDGET (panel->priv->layout_option_widget));
edcc2fc
+
edcc2fc
+        if (layout == NULL) {
edcc2fc
+                return;
edcc2fc
+        }
edcc2fc
+
edcc2fc
+	g_debug ("GdmGreeterPanel: activating selected layout %s", layout);
edcc2fc
+	gdm_layout_activate (layout);
edcc2fc
+
edcc2fc
+        g_signal_emit (panel, signals[LAYOUT_SELECTED], 0, layout);
edcc2fc
+
edcc2fc
+        g_free (layout);
edcc2fc
+}
edcc2fc
+static void
edcc2fc
 on_session_activated (GdmSessionOptionWidget *widget,
edcc2fc
                       GdmGreeterPanel        *panel)
edcc2fc
 {
edcc2fc
@@ -562,6 +597,12 @@ gdm_greeter_panel_init (GdmGreeterPanel 
edcc2fc
         gtk_box_pack_start (GTK_BOX (panel->priv->option_hbox), panel->priv->language_option_widget, FALSE, FALSE, 6);
edcc2fc
         gdm_profile_end ("creating option widget");
edcc2fc
 
edcc2fc
+        panel->priv->layout_option_widget = gdm_layout_option_widget_new ();
edcc2fc
+        g_signal_connect (G_OBJECT (panel->priv->layout_option_widget),
edcc2fc
+                          "layout-activated",
edcc2fc
+                          G_CALLBACK (on_layout_activated), panel);
edcc2fc
+        gtk_box_pack_start (GTK_BOX (panel->priv->option_hbox), panel->priv->layout_option_widget, FALSE, FALSE, 6);
edcc2fc
+
edcc2fc
         panel->priv->session_option_widget = gdm_session_option_widget_new ();
edcc2fc
         g_signal_connect (G_OBJECT (panel->priv->session_option_widget),
edcc2fc
                           "session-activated",
edcc2fc
@@ -626,6 +667,7 @@ gdm_greeter_panel_show_user_options (Gdm
edcc2fc
 {
edcc2fc
         gtk_widget_show (panel->priv->session_option_widget);
edcc2fc
         gtk_widget_show (panel->priv->language_option_widget);
edcc2fc
+        gtk_widget_show (panel->priv->layout_option_widget);
edcc2fc
 }
edcc2fc
 
edcc2fc
 void
edcc2fc
@@ -633,12 +675,17 @@ gdm_greeter_panel_hide_user_options (Gdm
edcc2fc
 {
edcc2fc
         gtk_widget_hide (panel->priv->session_option_widget);
edcc2fc
         gtk_widget_hide (panel->priv->language_option_widget);
edcc2fc
+        gtk_widget_hide (panel->priv->layout_option_widget);
edcc2fc
+
edcc2fc
+	g_debug ("GdmGreeterPanel: activating default layout");
edcc2fc
+	gdm_layout_activate (NULL); 
edcc2fc
 }
edcc2fc
 
edcc2fc
 void
edcc2fc
 gdm_greeter_panel_reset (GdmGreeterPanel *panel)
edcc2fc
 {
edcc2fc
         gdm_greeter_panel_set_default_language_name (panel, NULL);
edcc2fc
+        gdm_greeter_panel_set_default_layout_name (panel, NULL);
edcc2fc
         gdm_greeter_panel_set_default_session_name (panel, NULL);
edcc2fc
         gdm_greeter_panel_hide_user_options (panel);
edcc2fc
 }
edcc2fc
@@ -671,6 +718,26 @@ gdm_greeter_panel_set_default_language_n
edcc2fc
 }
edcc2fc
 
edcc2fc
 void
edcc2fc
+gdm_greeter_panel_set_default_layout_name (GdmGreeterPanel *panel,
edcc2fc
+                                           const char      *layout_name)
edcc2fc
+{
edcc2fc
+        g_return_if_fail (GDM_IS_GREETER_PANEL (panel));
edcc2fc
+
edcc2fc
+        if (layout_name != NULL &&
edcc2fc
+            !gdm_option_widget_lookup_item (GDM_OPTION_WIDGET (panel->priv->layout_option_widget),
edcc2fc
+                                            layout_name, NULL, NULL, NULL)) {
edcc2fc
+                gdm_recent_option_widget_add_item (GDM_RECENT_OPTION_WIDGET (panel->priv->layout_option_widget),
edcc2fc
+                                                   layout_name);
edcc2fc
+        }
edcc2fc
+
edcc2fc
+        gdm_option_widget_set_default_item (GDM_OPTION_WIDGET (panel->priv->layout_option_widget),
edcc2fc
+                                            layout_name);
edcc2fc
+
edcc2fc
+	g_debug ("GdmGreeterPanel: activating layout: %s", layout_name);
edcc2fc
+	gdm_layout_activate (layout_name);
edcc2fc
+}
edcc2fc
+
edcc2fc
+void
edcc2fc
 gdm_greeter_panel_set_default_session_name (GdmGreeterPanel *panel,
edcc2fc
                                             const char      *session_name)
edcc2fc
 {
edcc2fc
diff -up gdm-2.22.0/gui/simple-greeter/gdm-greeter-panel.h.keyboard-chooser gdm-2.22.0/gui/simple-greeter/gdm-greeter-panel.h
edcc2fc
--- gdm-2.22.0/gui/simple-greeter/gdm-greeter-panel.h.keyboard-chooser	2008-04-17 23:29:26.000000000 -0400
edcc2fc
+++ gdm-2.22.0/gui/simple-greeter/gdm-greeter-panel.h	2008-05-05 11:28:39.000000000 -0400
edcc2fc
@@ -48,6 +48,9 @@ typedef struct
edcc2fc
         void (* language_selected)           (GdmGreeterPanel *panel,
edcc2fc
                                               const char      *text);
edcc2fc
 
edcc2fc
+        void (* layout_selected)             (GdmGreeterPanel *panel,
edcc2fc
+                                              const char      *text);
edcc2fc
+
edcc2fc
         void (* session_selected)            (GdmGreeterPanel *panel,
edcc2fc
                                               const char      *text);
edcc2fc
 } GdmGreeterPanelClass;
edcc2fc
@@ -62,6 +65,8 @@ void                   gdm_greeter_panel
edcc2fc
 
edcc2fc
 void                   gdm_greeter_panel_set_default_language_name      (GdmGreeterPanel *panel,
edcc2fc
                                                                          const char      *language_name);
edcc2fc
+void                   gdm_greeter_panel_set_default_layout_name        (GdmGreeterPanel *panel,
edcc2fc
+                                                                         const char      *layout_name);
edcc2fc
 void                   gdm_greeter_panel_set_default_session_name       (GdmGreeterPanel *panel,
edcc2fc
                                                                          const char      *session_name);
edcc2fc
 G_END_DECLS
edcc2fc
diff -up gdm-2.22.0/gui/simple-greeter/gdm-greeter-session.c.keyboard-chooser gdm-2.22.0/gui/simple-greeter/gdm-greeter-session.c
edcc2fc
--- gdm-2.22.0/gui/simple-greeter/gdm-greeter-session.c.keyboard-chooser	2008-04-17 23:29:26.000000000 -0400
edcc2fc
+++ gdm-2.22.0/gui/simple-greeter/gdm-greeter-session.c	2008-05-05 11:28:39.000000000 -0400
edcc2fc
@@ -139,6 +139,16 @@ on_default_language_name_changed (GdmGre
edcc2fc
 }
edcc2fc
 
edcc2fc
 static void
edcc2fc
+on_default_layout_name_changed (GdmGreeterClient  *client,
edcc2fc
+                                const char        *text,
edcc2fc
+                                GdmGreeterSession *session)
edcc2fc
+{
edcc2fc
+        g_debug ("GdmGreeterSession: default layout name changed: %s", text);
edcc2fc
+        gdm_greeter_panel_set_default_layout_name (GDM_GREETER_PANEL (session->priv->panel),
edcc2fc
+                                                     text);
edcc2fc
+}
edcc2fc
+
edcc2fc
+static void
edcc2fc
 on_default_session_name_changed (GdmGreeterClient  *client,
edcc2fc
                                  const char        *text,
edcc2fc
                                  GdmGreeterSession *session)
edcc2fc
@@ -237,6 +247,14 @@ on_select_language (GdmGreeterSession   
edcc2fc
 }
edcc2fc
 
edcc2fc
 static void
edcc2fc
+on_select_layout (GdmGreeterSession      *session,
edcc2fc
+                  const char             *text)
edcc2fc
+{
edcc2fc
+        gdm_greeter_client_call_select_layout (session->priv->client,
edcc2fc
+                                               text);
edcc2fc
+}
edcc2fc
+
edcc2fc
+static void
edcc2fc
 on_select_user (GdmGreeterLoginWindow *login_window,
edcc2fc
                 const char            *text,
edcc2fc
                 GdmGreeterSession     *session)
edcc2fc
@@ -284,6 +302,11 @@ toggle_panel (GdmSessionManager *manager
edcc2fc
                                           session);
edcc2fc
 
edcc2fc
                 g_signal_connect_swapped (session->priv->panel,
edcc2fc
+                                          "layout-selected",
edcc2fc
+                                          G_CALLBACK (on_select_layout),
edcc2fc
+                                          session);
edcc2fc
+
edcc2fc
+                g_signal_connect_swapped (session->priv->panel,
edcc2fc
                                           "session-selected",
edcc2fc
                                           G_CALLBACK (on_select_session),
edcc2fc
                                           session);
edcc2fc
@@ -883,6 +906,10 @@ gdm_greeter_session_init (GdmGreeterSess
edcc2fc
                           G_CALLBACK (on_default_language_name_changed),
edcc2fc
                           session);
edcc2fc
         g_signal_connect (session->priv->client,
edcc2fc
+                          "default-layout-name-changed",
edcc2fc
+                          G_CALLBACK (on_default_layout_name_changed),
edcc2fc
+                          session);
edcc2fc
+        g_signal_connect (session->priv->client,
edcc2fc
                           "default-session-name-changed",
edcc2fc
                           G_CALLBACK (on_default_session_name_changed),
edcc2fc
                           session);
edcc2fc
diff -up /dev/null gdm-2.22.0/gui/simple-greeter/gdm-layout-chooser-dialog.c
edcc2fc
--- /dev/null	2008-05-05 08:19:33.312003896 -0400
edcc2fc
+++ gdm-2.22.0/gui/simple-greeter/gdm-layout-chooser-dialog.c	2008-05-05 11:28:39.000000000 -0400
edcc2fc
@@ -0,0 +1,216 @@
edcc2fc
+/* -*- Mode: C; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 8 -*-
edcc2fc
+ *
edcc2fc
+ * Copyright (C) 2008 Matthias Clasen <mclasen@redhat.com>
edcc2fc
+ *
edcc2fc
+ * This program is free software; you can redistribute it and/or modify
edcc2fc
+ * it under the terms of the GNU General Public License as published by
edcc2fc
+ * the Free Software Foundation; either version 2 of the License, or
edcc2fc
+ * (at your option) any later version.
edcc2fc
+ *
edcc2fc
+ * This program is distributed in the hope that it will be useful,
edcc2fc
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
edcc2fc
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
edcc2fc
+ * GNU General Public License for more details.
edcc2fc
+ *
edcc2fc
+ * You should have received a copy of the GNU General Public License
edcc2fc
+ * along with this program; if not, write to the Free Software
edcc2fc
+ * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
edcc2fc
+ *
edcc2fc
+ */
edcc2fc
+
edcc2fc
+#include "config.h"
edcc2fc
+
edcc2fc
+#include <stdlib.h>
edcc2fc
+#include <stdio.h>
edcc2fc
+#include <unistd.h>
edcc2fc
+#include <string.h>
edcc2fc
+
edcc2fc
+#include <locale.h>
edcc2fc
+
edcc2fc
+#include <glib.h>
edcc2fc
+#include <glib/gi18n.h>
edcc2fc
+#include <glib-object.h>
edcc2fc
+#include <gtk/gtk.h>
edcc2fc
+
edcc2fc
+#include "gdm-layout-chooser-widget.h"
edcc2fc
+#include "gdm-layout-chooser-dialog.h"
edcc2fc
+
edcc2fc
+#define GDM_LAYOUT_CHOOSER_DIALOG_GET_PRIVATE(o) (G_TYPE_INSTANCE_GET_PRIVATE ((o), GDM_TYPE_LAYOUT_CHOOSER_DIALOG, GdmLayoutChooserDialogPrivate))
edcc2fc
+
edcc2fc
+struct GdmLayoutChooserDialogPrivate
edcc2fc
+{
edcc2fc
+        GtkWidget *chooser_widget;
edcc2fc
+};
edcc2fc
+
edcc2fc
+
edcc2fc
+static void     gdm_layout_chooser_dialog_class_init  (GdmLayoutChooserDialogClass *klass);
edcc2fc
+static void     gdm_layout_chooser_dialog_init        (GdmLayoutChooserDialog      *layout_chooser_dialog);
edcc2fc
+static void     gdm_layout_chooser_dialog_finalize    (GObject                       *object);
edcc2fc
+
edcc2fc
+G_DEFINE_TYPE (GdmLayoutChooserDialog, gdm_layout_chooser_dialog, GTK_TYPE_DIALOG)
edcc2fc
+
edcc2fc
+char *
edcc2fc
+gdm_layout_chooser_dialog_get_current_layout_name (GdmLayoutChooserDialog *dialog)
edcc2fc
+{
edcc2fc
+        char *layout_name;
edcc2fc
+
edcc2fc
+        g_return_val_if_fail (GDM_IS_LAYOUT_CHOOSER_DIALOG (dialog), NULL);
edcc2fc
+
edcc2fc
+        layout_name = gdm_layout_chooser_widget_get_current_layout_name (GDM_LAYOUT_CHOOSER_WIDGET (dialog->priv->chooser_widget));
edcc2fc
+
edcc2fc
+        return layout_name;
edcc2fc
+}
edcc2fc
+
edcc2fc
+void
edcc2fc
+gdm_layout_chooser_dialog_set_current_layout_name (GdmLayoutChooserDialog *dialog,
edcc2fc
+                                                       const char               *layout_name)
edcc2fc
+{
edcc2fc
+
edcc2fc
+        g_return_if_fail (GDM_IS_LAYOUT_CHOOSER_DIALOG (dialog));
edcc2fc
+
edcc2fc
+        gdm_layout_chooser_widget_set_current_layout_name (GDM_LAYOUT_CHOOSER_WIDGET (dialog->priv->chooser_widget), layout_name);
edcc2fc
+}
edcc2fc
+
edcc2fc
+static void
edcc2fc
+gdm_layout_chooser_dialog_size_request (GtkWidget      *widget,
edcc2fc
+                                       GtkRequisition *requisition)
edcc2fc
+{
edcc2fc
+        int            screen_w;
edcc2fc
+        int            screen_h;
edcc2fc
+        GtkRequisition child_requisition;
edcc2fc
+
edcc2fc
+        if (GTK_WIDGET_CLASS (gdm_layout_chooser_dialog_parent_class)->size_request) {
edcc2fc
+                GTK_WIDGET_CLASS (gdm_layout_chooser_dialog_parent_class)->size_request (widget, requisition);
edcc2fc
+        }
edcc2fc
+
edcc2fc
+        screen_w = gdk_screen_get_width (gtk_widget_get_screen (widget));
edcc2fc
+        screen_h = gdk_screen_get_height (gtk_widget_get_screen (widget));
edcc2fc
+
edcc2fc
+        gtk_widget_get_child_requisition (GTK_BIN (widget)->child, &child_requisition);
edcc2fc
+        *requisition = child_requisition;
edcc2fc
+
edcc2fc
+        requisition->width += 2 * GTK_CONTAINER (widget)->border_width;
edcc2fc
+        requisition->height += 2 * GTK_CONTAINER (widget)->border_width;
edcc2fc
+
edcc2fc
+        requisition->width = MIN (requisition->width, .50 * screen_w);
edcc2fc
+        requisition->height = MIN (requisition->height, .80 * screen_h);
edcc2fc
+}
edcc2fc
+
edcc2fc
+static void
edcc2fc
+gdm_layout_chooser_dialog_response (GtkDialog *dialog,
edcc2fc
+                                      int        response_id)
edcc2fc
+{
edcc2fc
+        GdmLayoutChooserDialog *chooser_dialog;
edcc2fc
+
edcc2fc
+        chooser_dialog = GDM_LAYOUT_CHOOSER_DIALOG (dialog);
edcc2fc
+
edcc2fc
+        if (response_id == GTK_RESPONSE_OK) {
edcc2fc
+                gdm_chooser_widget_activate_selected_item (GDM_CHOOSER_WIDGET (chooser_dialog->priv->chooser_widget));
edcc2fc
+        }
edcc2fc
+}
edcc2fc
+
edcc2fc
+static void
edcc2fc
+gdm_layout_chooser_dialog_realize (GtkWidget *widget)
edcc2fc
+{
edcc2fc
+        GdmLayoutChooserDialog *chooser_dialog;
edcc2fc
+
edcc2fc
+        chooser_dialog = GDM_LAYOUT_CHOOSER_DIALOG (widget);
edcc2fc
+
edcc2fc
+        gtk_widget_show (chooser_dialog->priv->chooser_widget);
edcc2fc
+
edcc2fc
+        GTK_WIDGET_CLASS (gdm_layout_chooser_dialog_parent_class)->realize (widget);
edcc2fc
+}
edcc2fc
+
edcc2fc
+static void
edcc2fc
+gdm_layout_chooser_dialog_class_init (GdmLayoutChooserDialogClass *klass)
edcc2fc
+{
edcc2fc
+        GObjectClass   *object_class = G_OBJECT_CLASS (klass);
edcc2fc
+        GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (klass);
edcc2fc
+#ifdef I_COULD_GO_BACK_IN_TIME_AND_MAKE_RESPONSE_RUN_FIRST
edcc2fc
+        GtkDialogClass *dialog_class = GTK_DIALOG_CLASS (klass);
edcc2fc
+#endif
edcc2fc
+
edcc2fc
+        object_class->finalize = gdm_layout_chooser_dialog_finalize;
edcc2fc
+        widget_class->size_request = gdm_layout_chooser_dialog_size_request;
edcc2fc
+        widget_class->realize = gdm_layout_chooser_dialog_realize;
edcc2fc
+#ifdef I_COULD_GO_BACK_IN_TIME_AND_MAKE_RESPONSE_RUN_FIRST
edcc2fc
+        dialog_class->response = gdm_layout_chooser_dialog_response;
edcc2fc
+#endif
edcc2fc
+
edcc2fc
+        g_type_class_add_private (klass, sizeof (GdmLayoutChooserDialogPrivate));
edcc2fc
+}
edcc2fc
+
edcc2fc
+static gboolean
edcc2fc
+respond (GdmLayoutChooserDialog *dialog)
edcc2fc
+{
edcc2fc
+        gtk_dialog_response (GTK_DIALOG (dialog), GTK_RESPONSE_OK);
edcc2fc
+        return FALSE;
edcc2fc
+}
edcc2fc
+
edcc2fc
+static void
edcc2fc
+queue_response (GdmLayoutChooserDialog *dialog)
edcc2fc
+{
edcc2fc
+        g_idle_add ((GSourceFunc) respond, dialog);
edcc2fc
+}
edcc2fc
+
edcc2fc
+static void
edcc2fc
+gdm_layout_chooser_dialog_init (GdmLayoutChooserDialog *dialog)
edcc2fc
+{
edcc2fc
+
edcc2fc
+        dialog->priv = GDM_LAYOUT_CHOOSER_DIALOG_GET_PRIVATE (dialog);
edcc2fc
+
edcc2fc
+        dialog->priv->chooser_widget = gdm_layout_chooser_widget_new ();
edcc2fc
+        gdm_chooser_widget_set_hide_inactive_items (GDM_CHOOSER_WIDGET (dialog->priv->chooser_widget),
edcc2fc
+                                                    FALSE);
edcc2fc
+
edcc2fc
+#ifndef I_COULD_GO_BACK_IN_TIME_AND_MAKE_RESPONSE_RUN_FIRST
edcc2fc
+        g_signal_connect (G_OBJECT (dialog), "response", G_CALLBACK (gdm_layout_chooser_dialog_response), NULL);
edcc2fc
+#endif
edcc2fc
+
edcc2fc
+        gdm_layout_chooser_widget_set_current_layout_name (GDM_LAYOUT_CHOOSER_WIDGET (dialog->priv->chooser_widget),
edcc2fc
+                                                               setlocale (LC_MESSAGES, NULL));
edcc2fc
+        gtk_container_add (GTK_CONTAINER (GTK_DIALOG (dialog)->vbox), dialog->priv->chooser_widget);
edcc2fc
+
edcc2fc
+        g_signal_connect_swapped (G_OBJECT (dialog->priv->chooser_widget),
edcc2fc
+                                  "activated", G_CALLBACK (queue_response),
edcc2fc
+                                  dialog);
edcc2fc
+        gtk_dialog_add_buttons (GTK_DIALOG (dialog),
edcc2fc
+                                GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL,
edcc2fc
+                                GTK_STOCK_OK, GTK_RESPONSE_OK,
edcc2fc
+                                NULL);
edcc2fc
+        gtk_window_set_icon_name (GTK_WINDOW (dialog), "keyboard");
edcc2fc
+        gtk_dialog_set_has_separator (GTK_DIALOG (dialog), FALSE);
edcc2fc
+        gtk_container_set_border_width (GTK_CONTAINER (dialog), 12);
edcc2fc
+        gtk_container_set_border_width (GTK_CONTAINER (dialog->priv->chooser_widget), 5);
edcc2fc
+        gtk_window_set_position (GTK_WINDOW (dialog), GTK_WIN_POS_CENTER_ALWAYS);
edcc2fc
+        gtk_window_set_default_size (GTK_WINDOW (dialog), 512, 440);
edcc2fc
+}
edcc2fc
+
edcc2fc
+static void
edcc2fc
+gdm_layout_chooser_dialog_finalize (GObject *object)
edcc2fc
+{
edcc2fc
+        GdmLayoutChooserDialog *layout_chooser_dialog;
edcc2fc
+
edcc2fc
+        g_return_if_fail (object != NULL);
edcc2fc
+        g_return_if_fail (GDM_IS_LAYOUT_CHOOSER_DIALOG (object));
edcc2fc
+
edcc2fc
+        layout_chooser_dialog = GDM_LAYOUT_CHOOSER_DIALOG (object);
edcc2fc
+
edcc2fc
+        g_return_if_fail (layout_chooser_dialog->priv != NULL);
edcc2fc
+
edcc2fc
+        G_OBJECT_CLASS (gdm_layout_chooser_dialog_parent_class)->finalize (object);
edcc2fc
+}
edcc2fc
+
edcc2fc
+GtkWidget *
edcc2fc
+gdm_layout_chooser_dialog_new (void)
edcc2fc
+{
edcc2fc
+        GObject *object;
edcc2fc
+
edcc2fc
+        object = g_object_new (GDM_TYPE_LAYOUT_CHOOSER_DIALOG,
edcc2fc
+                               "title", _("Keyboard layouts"),
edcc2fc
+                               "border-width", 8,
edcc2fc
+                               "modal", TRUE,
edcc2fc
+                               NULL);
edcc2fc
+
edcc2fc
+        return GTK_WIDGET (object);
edcc2fc
+}
edcc2fc
diff -up /dev/null gdm-2.22.0/gui/simple-greeter/gdm-layout-chooser-dialog.h
edcc2fc
--- /dev/null	2008-05-05 08:19:33.312003896 -0400
edcc2fc
+++ gdm-2.22.0/gui/simple-greeter/gdm-layout-chooser-dialog.h	2008-05-05 11:28:39.000000000 -0400
edcc2fc
@@ -0,0 +1,59 @@
edcc2fc
+/* -*- Mode: C; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 8 -*-
edcc2fc
+ *
edcc2fc
+ * Copyright (C) 2008 Matthias Clasen <mclasen@redhat.com>
edcc2fc
+ *
edcc2fc
+ * This program is free software; you can redistribute it and/or modify
edcc2fc
+ * it under the terms of the GNU General Public License as published by
edcc2fc
+ * the Free Software Foundation; either version 2 of the License, or
edcc2fc
+ * (at your option) any later version.
edcc2fc
+ *
edcc2fc
+ * This program is distributed in the hope that it will be useful,
edcc2fc
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
edcc2fc
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
edcc2fc
+ * GNU General Public License for more details.
edcc2fc
+ *
edcc2fc
+ * You should have received a copy of the GNU General Public License
edcc2fc
+ * along with this program; if not, write to the Free Software
edcc2fc
+ * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
edcc2fc
+ *
edcc2fc
+ */
edcc2fc
+
edcc2fc
+#ifndef __GDM_LAYOUT_CHOOSER_DIALOG_H
edcc2fc
+#define __GDM_LAYOUT_CHOOSER_DIALOG_H
edcc2fc
+
edcc2fc
+#include <glib-object.h>
edcc2fc
+#include <gtk/gtkdialog.h>
edcc2fc
+
edcc2fc
+G_BEGIN_DECLS
edcc2fc
+
edcc2fc
+#define GDM_TYPE_LAYOUT_CHOOSER_DIALOG         (gdm_layout_chooser_dialog_get_type ())
edcc2fc
+#define GDM_LAYOUT_CHOOSER_DIALOG(o)           (G_TYPE_CHECK_INSTANCE_CAST ((o), GDM_TYPE_LAYOUT_CHOOSER_DIALOG, GdmLayoutChooserDialog))
edcc2fc
+#define GDM_LAYOUT_CHOOSER_DIALOG_CLASS(k)     (G_TYPE_CHECK_CLASS_CAST((k), GDM_TYPE_LAYOUT_CHOOSER_DIALOG, GdmLayoutChooserDialogClass))
edcc2fc
+#define GDM_IS_LAYOUT_CHOOSER_DIALOG(o)        (G_TYPE_CHECK_INSTANCE_TYPE ((o), GDM_TYPE_LAYOUT_CHOOSER_DIALOG))
edcc2fc
+#define GDM_IS_LAYOUT_CHOOSER_DIALOG_CLASS(k)  (G_TYPE_CHECK_CLASS_TYPE ((k), GDM_TYPE_LAYOUT_CHOOSER_DIALOG))
edcc2fc
+#define GDM_LAYOUT_CHOOSER_DIALOG_GET_CLASS(o) (G_TYPE_INSTANCE_GET_CLASS ((o), GDM_TYPE_LAYOUT_CHOOSER_DIALOG, GdmLayoutChooserDialogClass))
edcc2fc
+
edcc2fc
+typedef struct GdmLayoutChooserDialogPrivate GdmLayoutChooserDialogPrivate;
edcc2fc
+
edcc2fc
+typedef struct
edcc2fc
+{
edcc2fc
+        GtkDialog                        parent;
edcc2fc
+        GdmLayoutChooserDialogPrivate *priv;
edcc2fc
+} GdmLayoutChooserDialog;
edcc2fc
+
edcc2fc
+typedef struct
edcc2fc
+{
edcc2fc
+        GtkDialogClass   parent_class;
edcc2fc
+} GdmLayoutChooserDialogClass;
edcc2fc
+
edcc2fc
+GType                  gdm_layout_chooser_dialog_get_type                       (void);
edcc2fc
+
edcc2fc
+GtkWidget            * gdm_layout_chooser_dialog_new                            (void);
edcc2fc
+
edcc2fc
+char *                 gdm_layout_chooser_dialog_get_current_layout_name      (GdmLayoutChooserDialog *dialog);
edcc2fc
+void                   gdm_layout_chooser_dialog_set_current_layout_name      (GdmLayoutChooserDialog *dialog,
edcc2fc
+     const char               *layout_name);
edcc2fc
+
edcc2fc
+G_END_DECLS
edcc2fc
+
edcc2fc
+#endif /* __GDM_LAYOUT_CHOOSER_DIALOG_H */
edcc2fc
diff -up /dev/null gdm-2.22.0/gui/simple-greeter/gdm-layout-chooser-widget.c
edcc2fc
--- /dev/null	2008-05-05 08:19:33.312003896 -0400
edcc2fc
+++ gdm-2.22.0/gui/simple-greeter/gdm-layout-chooser-widget.c	2008-05-05 11:28:39.000000000 -0400
edcc2fc
@@ -0,0 +1,202 @@
edcc2fc
+/* -*- Mode: C; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 8 -*-
edcc2fc
+ *
edcc2fc
+ * Copyright (C) 2008 Matthias Clasen <mclasen@redhat.com>
edcc2fc
+ *
edcc2fc
+ * This program is free software; you can redistribute it and/or modify
edcc2fc
+ * it under the terms of the GNU General Public License as published by
edcc2fc
+ * the Free Software Foundation; either version 2 of the License, or
edcc2fc
+ * (at your option) any later version.
edcc2fc
+ *
edcc2fc
+ * This program is distributed in the hope that it will be useful,
edcc2fc
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
edcc2fc
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
edcc2fc
+ * GNU General Public License for more details.
edcc2fc
+ *
edcc2fc
+ * You should have received a copy of the GNU General Public License
edcc2fc
+ * along with this program; if not, write to the Free Software
edcc2fc
+ * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
edcc2fc
+ *
edcc2fc
+ */
edcc2fc
+
edcc2fc
+#include "config.h"
edcc2fc
+
edcc2fc
+#include <stdlib.h>
edcc2fc
+#include <stdio.h>
edcc2fc
+#include <unistd.h>
edcc2fc
+#include <string.h>
edcc2fc
+#include <errno.h>
edcc2fc
+#include <dirent.h>
edcc2fc
+#include <locale.h>
edcc2fc
+#include <sys/stat.h>
edcc2fc
+
edcc2fc
+#include <fontconfig/fontconfig.h>
edcc2fc
+
edcc2fc
+#include <glib.h>
edcc2fc
+#include <glib/gi18n.h>
edcc2fc
+#include <glib/gstdio.h>
edcc2fc
+#include <gtk/gtk.h>
edcc2fc
+
edcc2fc
+#include "gdm-layout-chooser-widget.h"
edcc2fc
+#include "gdm-chooser-widget.h"
edcc2fc
+#include "gdm-layouts.h"
edcc2fc
+
edcc2fc
+#define GDM_LAYOUT_CHOOSER_WIDGET_GET_PRIVATE(o) (G_TYPE_INSTANCE_GET_PRIVATE ((o), GDM_TYPE_LAYOUT_CHOOSER_WIDGET, GdmLayoutChooserWidgetPrivate))
edcc2fc
+
edcc2fc
+struct GdmLayoutChooserWidgetPrivate
edcc2fc
+{
edcc2fc
+        guint               layouts_added : 1;
edcc2fc
+};
edcc2fc
+
edcc2fc
+static void     gdm_layout_chooser_widget_class_init  (GdmLayoutChooserWidgetClass *klass);
edcc2fc
+static void     gdm_layout_chooser_widget_init        (GdmLayoutChooserWidget      *layout_chooser_widget);
edcc2fc
+static void     gdm_layout_chooser_widget_finalize    (GObject                       *object);
edcc2fc
+
edcc2fc
+G_DEFINE_TYPE (GdmLayoutChooserWidget, gdm_layout_chooser_widget, GDM_TYPE_CHOOSER_WIDGET)
edcc2fc
+
edcc2fc
+enum {
edcc2fc
+        CHOOSER_LIST_TITLE_COLUMN = 0,
edcc2fc
+        CHOOSER_LIST_TRANSLATED_COLUMN,
edcc2fc
+        CHOOSER_LIST_LOCALE_COLUMN
edcc2fc
+};
edcc2fc
+
edcc2fc
+char *
edcc2fc
+gdm_layout_chooser_widget_get_current_layout_name (GdmLayoutChooserWidget *widget)
edcc2fc
+{
edcc2fc
+        char *id;
edcc2fc
+
edcc2fc
+        g_return_val_if_fail (GDM_IS_LAYOUT_CHOOSER_WIDGET (widget), NULL);
edcc2fc
+
edcc2fc
+        id = gdm_chooser_widget_get_active_item (GDM_CHOOSER_WIDGET (widget));
edcc2fc
+
edcc2fc
+        if (id == NULL) {
edcc2fc
+                id = g_strdup ("us");
edcc2fc
+        }
edcc2fc
+
edcc2fc
+        return id;
edcc2fc
+}
edcc2fc
+
edcc2fc
+void
edcc2fc
+gdm_layout_chooser_widget_set_current_layout_name (GdmLayoutChooserWidget *widget,
edcc2fc
+                                                   const char             *id)
edcc2fc
+{
edcc2fc
+        g_return_if_fail (GDM_IS_LAYOUT_CHOOSER_WIDGET (widget));
edcc2fc
+
edcc2fc
+        if (id == NULL) {
edcc2fc
+                gdm_chooser_widget_set_active_item (GDM_CHOOSER_WIDGET (widget),
edcc2fc
+                                                   NULL);
edcc2fc
+                return;
edcc2fc
+        }
edcc2fc
+
edcc2fc
+        gdm_chooser_widget_set_active_item (GDM_CHOOSER_WIDGET (widget), id);
edcc2fc
+}
edcc2fc
+
edcc2fc
+static void
edcc2fc
+gdm_layout_chooser_widget_add_layout (GdmLayoutChooserWidget *widget,
edcc2fc
+                                          const char         *name)
edcc2fc
+{
edcc2fc
+        char *layout;
edcc2fc
+	char *escaped;
edcc2fc
+
edcc2fc
+        layout = gdm_get_layout_from_name (name);
edcc2fc
+
edcc2fc
+        if (layout != NULL) {
edcc2fc
+		escaped = g_markup_escape_text (layout, -1);
edcc2fc
+                gdm_chooser_widget_add_item (GDM_CHOOSER_WIDGET (widget),
edcc2fc
+                                             name, 
edcc2fc
+                                             NULL,
edcc2fc
+                                             escaped, 
edcc2fc
+                                             NULL,
edcc2fc
+                                             0,
edcc2fc
+                                             FALSE,
edcc2fc
+                                             FALSE);
edcc2fc
+		g_free (escaped);
edcc2fc
+                g_free (layout);
edcc2fc
+        }
edcc2fc
+}
edcc2fc
+
edcc2fc
+void
edcc2fc
+add_available_layouts (GdmLayoutChooserWidget *widget)
edcc2fc
+{
edcc2fc
+        char **layout_names;
edcc2fc
+        int    i;
edcc2fc
+
edcc2fc
+        layout_names = gdm_get_all_layout_names ();
edcc2fc
+
edcc2fc
+        for (i = 0; layout_names[i] != NULL; i++) {
edcc2fc
+                gdm_layout_chooser_widget_add_layout (widget,
edcc2fc
+                                                          layout_names[i]);
edcc2fc
+        }
edcc2fc
+
edcc2fc
+        g_strfreev (layout_names);
edcc2fc
+}
edcc2fc
+
edcc2fc
+static void
edcc2fc
+gdm_layout_chooser_widget_dispose (GObject *object)
edcc2fc
+{
edcc2fc
+        G_OBJECT_CLASS (gdm_layout_chooser_widget_parent_class)->dispose (object);
edcc2fc
+}
edcc2fc
+
edcc2fc
+static void
edcc2fc
+gdm_layout_chooser_widget_realize (GtkWidget *widget)
edcc2fc
+{
edcc2fc
+        GdmLayoutChooserWidget *chooser;
edcc2fc
+
edcc2fc
+        chooser = GDM_LAYOUT_CHOOSER_WIDGET (widget);
edcc2fc
+
edcc2fc
+        GTK_WIDGET_CLASS (gdm_layout_chooser_widget_parent_class)->realize (widget);
edcc2fc
+
edcc2fc
+        if (!chooser->priv->layouts_added) {
edcc2fc
+                add_available_layouts (chooser);
edcc2fc
+                chooser->priv->layouts_added = TRUE;
edcc2fc
+        }
edcc2fc
+}
edcc2fc
+
edcc2fc
+static void
edcc2fc
+gdm_layout_chooser_widget_class_init (GdmLayoutChooserWidgetClass *klass)
edcc2fc
+{
edcc2fc
+        GObjectClass   *object_class = G_OBJECT_CLASS (klass);
edcc2fc
+        GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (klass);
edcc2fc
+
edcc2fc
+        object_class->dispose = gdm_layout_chooser_widget_dispose;
edcc2fc
+        object_class->finalize = gdm_layout_chooser_widget_finalize;
edcc2fc
+        widget_class->realize = gdm_layout_chooser_widget_realize;
edcc2fc
+
edcc2fc
+        g_type_class_add_private (klass, sizeof (GdmLayoutChooserWidgetPrivate));
edcc2fc
+}
edcc2fc
+
edcc2fc
+static void
edcc2fc
+gdm_layout_chooser_widget_init (GdmLayoutChooserWidget *widget)
edcc2fc
+{
edcc2fc
+        widget->priv = GDM_LAYOUT_CHOOSER_WIDGET_GET_PRIVATE (widget);
edcc2fc
+
edcc2fc
+        gdm_chooser_widget_set_separator_position (GDM_CHOOSER_WIDGET (widget),
edcc2fc
+                                                   GDM_CHOOSER_WIDGET_POSITION_TOP);
edcc2fc
+}
edcc2fc
+
edcc2fc
+static void
edcc2fc
+gdm_layout_chooser_widget_finalize (GObject *object)
edcc2fc
+{
edcc2fc
+        GdmLayoutChooserWidget *layout_chooser_widget;
edcc2fc
+
edcc2fc
+        g_return_if_fail (object != NULL);
edcc2fc
+        g_return_if_fail (GDM_IS_LAYOUT_CHOOSER_WIDGET (object));
edcc2fc
+
edcc2fc
+        layout_chooser_widget = GDM_LAYOUT_CHOOSER_WIDGET (object);
edcc2fc
+
edcc2fc
+        g_return_if_fail (layout_chooser_widget->priv != NULL);
edcc2fc
+
edcc2fc
+        G_OBJECT_CLASS (gdm_layout_chooser_widget_parent_class)->finalize (object);
edcc2fc
+}
edcc2fc
+
edcc2fc
+GtkWidget *
edcc2fc
+gdm_layout_chooser_widget_new (void)
edcc2fc
+{
edcc2fc
+        GObject *object;
edcc2fc
+
edcc2fc
+        object = g_object_new (GDM_TYPE_LAYOUT_CHOOSER_WIDGET,
edcc2fc
+                               "inactive-text", _("_Keyboard:"),
edcc2fc
+                               "active-text", _("_Keyboard:"),
edcc2fc
+                               NULL);
edcc2fc
+
edcc2fc
+        return GTK_WIDGET (object);
edcc2fc
+}
edcc2fc
diff -up /dev/null gdm-2.22.0/gui/simple-greeter/gdm-layout-chooser-widget.h
edcc2fc
--- /dev/null	2008-05-05 08:19:33.312003896 -0400
edcc2fc
+++ gdm-2.22.0/gui/simple-greeter/gdm-layout-chooser-widget.h	2008-05-05 11:28:39.000000000 -0400
edcc2fc
@@ -0,0 +1,58 @@
edcc2fc
+/* -*- Mode: C; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 8 -*-
edcc2fc
+ *
edcc2fc
+ * Copyright (C) 2008 Matthias Clasen <mclasen@redhat.com>
edcc2fc
+ *
edcc2fc
+ * This program is free software; you can redistribute it and/or modify
edcc2fc
+ * it under the terms of the GNU General Public License as published by
edcc2fc
+ * the Free Software Foundation; either version 2 of the License, or
edcc2fc
+ * (at your option) any later version.
edcc2fc
+ *
edcc2fc
+ * This program is distributed in the hope that it will be useful,
edcc2fc
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
edcc2fc
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
edcc2fc
+ * GNU General Public License for more details.
edcc2fc
+ *
edcc2fc
+ * You should have received a copy of the GNU General Public License
edcc2fc
+ * along with this program; if not, write to the Free Software
edcc2fc
+ * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
edcc2fc
+ *
edcc2fc
+ */
edcc2fc
+
edcc2fc
+#ifndef __GDM_LAYOUT_CHOOSER_WIDGET_H
edcc2fc
+#define __GDM_LAYOUT_CHOOSER_WIDGET_H
edcc2fc
+
edcc2fc
+#include <glib-object.h>
edcc2fc
+#include "gdm-chooser-widget.h"
edcc2fc
+
edcc2fc
+G_BEGIN_DECLS
edcc2fc
+
edcc2fc
+#define GDM_TYPE_LAYOUT_CHOOSER_WIDGET         (gdm_layout_chooser_widget_get_type ())
edcc2fc
+#define GDM_LAYOUT_CHOOSER_WIDGET(o)           (G_TYPE_CHECK_INSTANCE_CAST ((o), GDM_TYPE_LAYOUT_CHOOSER_WIDGET, GdmLayoutChooserWidget))
edcc2fc
+#define GDM_LAYOUT_CHOOSER_WIDGET_CLASS(k)     (G_TYPE_CHECK_CLASS_CAST((k), GDM_TYPE_LAYOUT_CHOOSER_WIDGET, GdmLayoutChooserWidgetClass))
edcc2fc
+#define GDM_IS_LAYOUT_CHOOSER_WIDGET(o)        (G_TYPE_CHECK_INSTANCE_TYPE ((o), GDM_TYPE_LAYOUT_CHOOSER_WIDGET))
edcc2fc
+#define GDM_IS_LAYOUT_CHOOSER_WIDGET_CLASS(k)  (G_TYPE_CHECK_CLASS_TYPE ((k), GDM_TYPE_LAYOUT_CHOOSER_WIDGET))
edcc2fc
+#define GDM_LAYOUT_CHOOSER_WIDGET_GET_CLASS(o) (G_TYPE_INSTANCE_GET_CLASS ((o), GDM_TYPE_LAYOUT_CHOOSER_WIDGET, GdmLayoutChooserWidgetClass))
edcc2fc
+
edcc2fc
+typedef struct GdmLayoutChooserWidgetPrivate GdmLayoutChooserWidgetPrivate;
edcc2fc
+
edcc2fc
+typedef struct
edcc2fc
+{
edcc2fc
+        GdmChooserWidget                 parent;
edcc2fc
+        GdmLayoutChooserWidgetPrivate *priv;
edcc2fc
+} GdmLayoutChooserWidget;
edcc2fc
+
edcc2fc
+typedef struct
edcc2fc
+{
edcc2fc
+        GdmChooserWidgetClass   parent_class;
edcc2fc
+} GdmLayoutChooserWidgetClass;
edcc2fc
+
edcc2fc
+GType                  gdm_layout_chooser_widget_get_type                       (void);
edcc2fc
+GtkWidget *            gdm_layout_chooser_widget_new                            (void);
edcc2fc
+
edcc2fc
+char *                 gdm_layout_chooser_widget_get_current_layout_name      (GdmLayoutChooserWidget *widget);
edcc2fc
+void                   gdm_layout_chooser_widget_set_current_layout_name      (GdmLayoutChooserWidget *widget,
edcc2fc
+                                                                                   const char               *name);
edcc2fc
+
edcc2fc
+G_END_DECLS
edcc2fc
+
edcc2fc
+#endif /* __GDM_LAYOUT_CHOOSER_WIDGET_H */
edcc2fc
diff -up /dev/null gdm-2.22.0/gui/simple-greeter/gdm-layout-option-widget.c
edcc2fc
--- /dev/null	2008-05-05 08:19:33.312003896 -0400
edcc2fc
+++ gdm-2.22.0/gui/simple-greeter/gdm-layout-option-widget.c	2008-05-05 11:28:39.000000000 -0400
edcc2fc
@@ -0,0 +1,268 @@
edcc2fc
+/* -*- Mode: C; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 8 -*-
edcc2fc
+ *
edcc2fc
+ * Copyright (C) 2008 Red Hat, Inc.
edcc2fc
+ *
edcc2fc
+ * This program is free software; you can redistribute it and/or modify
edcc2fc
+ * it under the terms of the GNU General Public License as published by
edcc2fc
+ * the Free Software Foundation; either version 2 of the License, or
edcc2fc
+ * (at your option) any later version.
edcc2fc
+ *
edcc2fc
+ * This program is distributed in the hope that it will be useful,
edcc2fc
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
edcc2fc
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
edcc2fc
+ * GNU General Public License for more details.
edcc2fc
+ *
edcc2fc
+ * You should have received a copy of the GNU General Public License
edcc2fc
+ * along with this program; if not, write to the Free Software
edcc2fc
+ * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
edcc2fc
+ *
edcc2fc
+ * Written by: Matthias Clasen <mclasen@redhat.com>
edcc2fc
+ *
edcc2fc
+ */
edcc2fc
+
edcc2fc
+#include "config.h"
edcc2fc
+
edcc2fc
+#include <stdlib.h>
edcc2fc
+#include <stdio.h>
edcc2fc
+#include <unistd.h>
edcc2fc
+#include <string.h>
edcc2fc
+#include <errno.h>
edcc2fc
+#include <dirent.h>
edcc2fc
+#include <sys/stat.h>
edcc2fc
+
edcc2fc
+#include <glib.h>
edcc2fc
+#include <glib/gi18n.h>
edcc2fc
+#include <glib/gstdio.h>
edcc2fc
+#include <gtk/gtk.h>
edcc2fc
+
edcc2fc
+#include "gdm-profile.h"
edcc2fc
+#include "gdm-layouts.h"
edcc2fc
+#include "gdm-layout-option-widget.h"
edcc2fc
+#include "gdm-recent-option-widget.h"
edcc2fc
+#include "gdm-layout-chooser-dialog.h"
edcc2fc
+
edcc2fc
+#define GDM_LAYOUT_OPTION_WIDGET_GET_PRIVATE(o) (G_TYPE_INSTANCE_GET_PRIVATE ((o), GDM_TYPE_LAYOUT_OPTION_WIDGET, GdmLayoutOptionWidgetPrivate))
edcc2fc
+
edcc2fc
+struct GdmLayoutOptionWidgetPrivate
edcc2fc
+{
edcc2fc
+        GtkWidget *dialog;
edcc2fc
+};
edcc2fc
+
edcc2fc
+enum {
edcc2fc
+        LAYOUT_ACTIVATED,
edcc2fc
+        NUMBER_OF_SIGNALS
edcc2fc
+};
edcc2fc
+
edcc2fc
+static guint signals [NUMBER_OF_SIGNALS] = { 0, };
edcc2fc
+
edcc2fc
+static void     gdm_layout_option_widget_class_init  (GdmLayoutOptionWidgetClass *klass);
edcc2fc
+static void     gdm_layout_option_widget_init        (GdmLayoutOptionWidget      *layout_option_widget);
edcc2fc
+static void     gdm_layout_option_widget_finalize    (GObject                     *object);
edcc2fc
+
edcc2fc
+G_DEFINE_TYPE (GdmLayoutOptionWidget, gdm_layout_option_widget, GDM_TYPE_RECENT_OPTION_WIDGET)
edcc2fc
+
edcc2fc
+static void
edcc2fc
+gdm_layout_option_widget_set_layout_from_dialog (GdmLayoutOptionWidget *widget)
edcc2fc
+{
edcc2fc
+        char *layout_name;
edcc2fc
+
edcc2fc
+        layout_name = gdm_layout_chooser_dialog_get_current_layout_name (GDM_LAYOUT_CHOOSER_DIALOG (widget->priv->dialog));
edcc2fc
+        gdm_layout_option_widget_set_current_layout_name (widget, layout_name);
edcc2fc
+        g_free (layout_name);
edcc2fc
+}
edcc2fc
+
edcc2fc
+static void
edcc2fc
+on_dialog_response (GtkDialog               *dialog,
edcc2fc
+                    int                      response_id,
edcc2fc
+                    GdmLayoutOptionWidget *widget)
edcc2fc
+{
edcc2fc
+        switch (response_id) {
edcc2fc
+                case GTK_RESPONSE_OK:
edcc2fc
+                        gdm_layout_option_widget_set_layout_from_dialog (widget);
edcc2fc
+                break;
edcc2fc
+
edcc2fc
+                default:
edcc2fc
+                break;
edcc2fc
+        }
edcc2fc
+        gtk_widget_hide (GTK_WIDGET (dialog));
edcc2fc
+}
edcc2fc
+
edcc2fc
+static void
edcc2fc
+gdm_layout_option_widget_show_dialog (GdmLayoutOptionWidget *widget)
edcc2fc
+{
edcc2fc
+        gdm_layout_option_widget_set_layout_from_dialog (widget);
edcc2fc
+        gtk_widget_show_all (GTK_WIDGET (widget->priv->dialog));
edcc2fc
+}
edcc2fc
+
edcc2fc
+static void
edcc2fc
+gdm_layout_option_widget_activated (GdmOptionWidget *widget)
edcc2fc
+{
edcc2fc
+        char *active_item_id;
edcc2fc
+
edcc2fc
+        active_item_id = gdm_option_widget_get_active_item (GDM_OPTION_WIDGET (widget));
edcc2fc
+        if (active_item_id == NULL) {
edcc2fc
+                return;
edcc2fc
+        }
edcc2fc
+
edcc2fc
+        if (strcmp (active_item_id, "__other") == 0) {
edcc2fc
+                gdm_layout_option_widget_show_dialog (GDM_LAYOUT_OPTION_WIDGET (widget));
edcc2fc
+                return;
edcc2fc
+        }
edcc2fc
+
edcc2fc
+        gdm_layout_chooser_dialog_set_current_layout_name (GDM_LAYOUT_CHOOSER_DIALOG (GDM_LAYOUT_OPTION_WIDGET (widget)->priv->dialog),
edcc2fc
+                                                               active_item_id);
edcc2fc
+
edcc2fc
+        g_signal_emit (G_OBJECT (widget), signals[LAYOUT_ACTIVATED], 0);
edcc2fc
+}
edcc2fc
+
edcc2fc
+static void
edcc2fc
+gdm_layout_option_widget_class_init (GdmLayoutOptionWidgetClass *klass)
edcc2fc
+{
edcc2fc
+        GObjectClass   *object_class = G_OBJECT_CLASS (klass);
edcc2fc
+        GdmOptionWidgetClass *option_widget_class = GDM_OPTION_WIDGET_CLASS (klass);
edcc2fc
+
edcc2fc
+        object_class->finalize = gdm_layout_option_widget_finalize;
edcc2fc
+
edcc2fc
+        option_widget_class->activated = gdm_layout_option_widget_activated;
edcc2fc
+
edcc2fc
+        signals[LAYOUT_ACTIVATED] = g_signal_new ("layout-activated",
edcc2fc
+                                                     G_TYPE_FROM_CLASS (object_class),
edcc2fc
+                                                     G_SIGNAL_RUN_FIRST,
edcc2fc
+                                                     G_STRUCT_OFFSET (GdmLayoutOptionWidgetClass, layout_activated),
edcc2fc
+                                                     NULL,
edcc2fc
+                                                     NULL,
edcc2fc
+                                                     g_cclosure_marshal_VOID__VOID,
edcc2fc
+                                                     G_TYPE_NONE,
edcc2fc
+                                                     0);
edcc2fc
+
edcc2fc
+        g_type_class_add_private (klass, sizeof (GdmLayoutOptionWidgetPrivate));
edcc2fc
+}
edcc2fc
+
edcc2fc
+static gboolean
edcc2fc
+gdm_layout_option_widget_lookup_item (GdmRecentOptionWidget *widget,
edcc2fc
+                                      const char            *id,
edcc2fc
+                                      char                 **name,
edcc2fc
+                                      char                 **comment)
edcc2fc
+{
edcc2fc
+	char *layout;
edcc2fc
+
edcc2fc
+        layout = gdm_get_layout_from_name (id);
edcc2fc
+
edcc2fc
+        if (layout == NULL) {
edcc2fc
+                return FALSE;
edcc2fc
+        }
edcc2fc
+
edcc2fc
+        *name = layout;
edcc2fc
+        *comment = NULL;
edcc2fc
+
edcc2fc
+        return TRUE;
edcc2fc
+}
edcc2fc
+
edcc2fc
+static void
edcc2fc
+create_dialog (GdmLayoutOptionWidget *widget)
edcc2fc
+{
edcc2fc
+        gdm_profile_start (NULL);
edcc2fc
+
edcc2fc
+        g_assert (widget->priv->dialog == NULL);
edcc2fc
+
edcc2fc
+        widget->priv->dialog = gdm_layout_chooser_dialog_new ();
edcc2fc
+
edcc2fc
+        g_signal_connect (GTK_DIALOG (widget->priv->dialog),
edcc2fc
+                          "response",
edcc2fc
+                          G_CALLBACK (on_dialog_response),
edcc2fc
+                          widget);
edcc2fc
+
edcc2fc
+        gdm_profile_end (NULL);
edcc2fc
+}
edcc2fc
+
edcc2fc
+static void
edcc2fc
+gdm_layout_option_widget_init (GdmLayoutOptionWidget *widget)
edcc2fc
+{
edcc2fc
+        GError *error;
edcc2fc
+
edcc2fc
+        widget->priv = GDM_LAYOUT_OPTION_WIDGET_GET_PRIVATE (widget);
edcc2fc
+
edcc2fc
+	error = NULL; 
edcc2fc
+        gdm_recent_option_widget_set_gconf_key (GDM_RECENT_OPTION_WIDGET (widget),
edcc2fc
+                                                "/apps/gdm/simple-greeter/recent-layouts",
edcc2fc
+                                                gdm_layout_option_widget_lookup_item,
edcc2fc
+                                                &error);
edcc2fc
+
edcc2fc
+        if (error != NULL) {
edcc2fc
+                g_warning ("Could not read recent layouts from gconf: %s",
edcc2fc
+                           error->message);
edcc2fc
+                g_error_free (error);
edcc2fc
+        }
edcc2fc
+
edcc2fc
+        gdm_option_widget_add_item (GDM_OPTION_WIDGET (widget),
edcc2fc
+                                    "__other",
edcc2fc
+                                    _("Other..."),
edcc2fc
+                                    _("Choose a keyboard layout from the "
edcc2fc
+                                      "full list of available layouts."),
edcc2fc
+                                    GDM_OPTION_WIDGET_POSITION_BOTTOM);
edcc2fc
+
edcc2fc
+        create_dialog (widget);
edcc2fc
+}
edcc2fc
+
edcc2fc
+static void
edcc2fc
+gdm_layout_option_widget_finalize (GObject *object)
edcc2fc
+{
edcc2fc
+        GdmLayoutOptionWidget *layout_option_widget;
edcc2fc
+
edcc2fc
+        g_return_if_fail (object != NULL);
edcc2fc
+        g_return_if_fail (GDM_IS_LAYOUT_OPTION_WIDGET (object));
edcc2fc
+
edcc2fc
+        layout_option_widget = GDM_LAYOUT_OPTION_WIDGET (object);
edcc2fc
+
edcc2fc
+        g_return_if_fail (layout_option_widget->priv != NULL);
edcc2fc
+
edcc2fc
+        G_OBJECT_CLASS (gdm_layout_option_widget_parent_class)->finalize (object);
edcc2fc
+}
edcc2fc
+
edcc2fc
+GtkWidget *
edcc2fc
+gdm_layout_option_widget_new (void)
edcc2fc
+{
edcc2fc
+        GObject *object;
edcc2fc
+
edcc2fc
+        object = g_object_new (GDM_TYPE_LAYOUT_OPTION_WIDGET,
edcc2fc
+                               "label-text", _("_Keyboard:"),
edcc2fc
+                               "icon-name", "keyboard",
edcc2fc
+                               "max-item-count", 8,
edcc2fc
+                               NULL);
edcc2fc
+
edcc2fc
+        return GTK_WIDGET (object);
edcc2fc
+}
edcc2fc
+
edcc2fc
+char *
edcc2fc
+gdm_layout_option_widget_get_current_layout_name (GdmLayoutOptionWidget *widget)
edcc2fc
+{
edcc2fc
+        char *active_item_id;
edcc2fc
+
edcc2fc
+        active_item_id = gdm_option_widget_get_active_item (GDM_OPTION_WIDGET (widget));
edcc2fc
+        if (active_item_id == NULL) {
edcc2fc
+                return NULL;
edcc2fc
+        }
edcc2fc
+
edcc2fc
+        if (strcmp (active_item_id, "__other") == 0) {
edcc2fc
+                g_free (active_item_id);
edcc2fc
+                return NULL;
edcc2fc
+        }
edcc2fc
+
edcc2fc
+        return active_item_id;
edcc2fc
+}
edcc2fc
+
edcc2fc
+void
edcc2fc
+gdm_layout_option_widget_set_current_layout_name (GdmLayoutOptionWidget *widget,
edcc2fc
+                                                  const char            *id)
edcc2fc
+{
edcc2fc
+        g_return_if_fail (GDM_IS_LAYOUT_OPTION_WIDGET (widget));
edcc2fc
+
edcc2fc
+        if (id != NULL &&
edcc2fc
+            !gdm_option_widget_lookup_item (GDM_OPTION_WIDGET (widget),
edcc2fc
+                                            id, NULL, NULL, NULL)) {
edcc2fc
+                gdm_recent_option_widget_add_item (GDM_RECENT_OPTION_WIDGET (widget),
edcc2fc
+                                                   id);
edcc2fc
+        }
edcc2fc
+
edcc2fc
+        gdm_option_widget_set_active_item (GDM_OPTION_WIDGET (widget), id);
edcc2fc
+}
edcc2fc
diff -up /dev/null gdm-2.22.0/gui/simple-greeter/gdm-layout-option-widget.h
edcc2fc
--- /dev/null	2008-05-05 08:19:33.312003896 -0400
edcc2fc
+++ gdm-2.22.0/gui/simple-greeter/gdm-layout-option-widget.h	2008-05-05 11:28:39.000000000 -0400
edcc2fc
@@ -0,0 +1,62 @@
edcc2fc
+/* -*- Mode: C; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 8 -*-
edcc2fc
+ *
edcc2fc
+ * Copyright (C) 2008 Red Hat, Inc.
edcc2fc
+ *
edcc2fc
+ * This program is free software; you can redistribute it and/or modify
edcc2fc
+ * it under the terms of the GNU General Public License as published by
edcc2fc
+ * the Free Software Foundation; either version 2 of the License, or
edcc2fc
+ * (at your option) any later version.
edcc2fc
+ *
edcc2fc
+ * This program is distributed in the hope that it will be useful,
edcc2fc
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
edcc2fc
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
edcc2fc
+ * GNU General Public License for more details.
edcc2fc
+ *
edcc2fc
+ * You should have received a copy of the GNU General Public License
edcc2fc
+ * along with this program; if not, write to the Free Software
edcc2fc
+ * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
edcc2fc
+ *
edcc2fc
+ *  Written by: Ray Strode <rstrode@redhat.com>
edcc2fc
+ */
edcc2fc
+
edcc2fc
+#ifndef __GDM_LAYOUT_OPTION_WIDGET_H
edcc2fc
+#define __GDM_LAYOUT_OPTION_WIDGET_H
edcc2fc
+
edcc2fc
+#include <glib-object.h>
edcc2fc
+
edcc2fc
+#include "gdm-recent-option-widget.h"
edcc2fc
+
edcc2fc
+G_BEGIN_DECLS
edcc2fc
+
edcc2fc
+#define GDM_TYPE_LAYOUT_OPTION_WIDGET         (gdm_layout_option_widget_get_type ())
edcc2fc
+#define GDM_LAYOUT_OPTION_WIDGET(o)           (G_TYPE_CHECK_INSTANCE_CAST ((o), GDM_TYPE_LAYOUT_OPTION_WIDGET, GdmLayoutOptionWidget))
edcc2fc
+#define GDM_LAYOUT_OPTION_WIDGET_CLASS(k)     (G_TYPE_CHECK_CLASS_CAST((k), GDM_TYPE_LAYOUT_OPTION_WIDGET, GdmLayoutOptionWidgetClass))
edcc2fc
+#define GDM_IS_LAYOUT_OPTION_WIDGET(o)        (G_TYPE_CHECK_INSTANCE_TYPE ((o), GDM_TYPE_LAYOUT_OPTION_WIDGET))
edcc2fc
+#define GDM_IS_LAYOUT_OPTION_WIDGET_CLASS(k)  (G_TYPE_CHECK_CLASS_TYPE ((k), GDM_TYPE_LAYOUT_OPTION_WIDGET))
edcc2fc
+#define GDM_LAYOUT_OPTION_WIDGET_GET_CLASS(o) (G_TYPE_INSTANCE_GET_CLASS ((o), GDM_TYPE_LAYOUT_OPTION_WIDGET, GdmLayoutOptionWidgetClass))
edcc2fc
+
edcc2fc
+typedef struct GdmLayoutOptionWidgetPrivate GdmLayoutOptionWidgetPrivate;
edcc2fc
+
edcc2fc
+typedef struct
edcc2fc
+{
edcc2fc
+        GdmRecentOptionWidget          parent;
edcc2fc
+        GdmLayoutOptionWidgetPrivate *priv;
edcc2fc
+} GdmLayoutOptionWidget;
edcc2fc
+
edcc2fc
+typedef struct
edcc2fc
+{
edcc2fc
+        GdmRecentOptionWidgetClass    parent_class;
edcc2fc
+
edcc2fc
+        void (* layout_activated)        (GdmLayoutOptionWidget *widget);
edcc2fc
+} GdmLayoutOptionWidgetClass;
edcc2fc
+
edcc2fc
+GType                  gdm_layout_option_widget_get_type               (void);
edcc2fc
+GtkWidget *            gdm_layout_option_widget_new                    (void);
edcc2fc
+
edcc2fc
+char *                 gdm_layout_option_widget_get_current_layout_name      (GdmLayoutOptionWidget *widget);
edcc2fc
+void                   gdm_layout_option_widget_set_current_layout_name      (GdmLayoutOptionWidget *widget,
edcc2fc
+                                                                                  const char              *name);
edcc2fc
+
edcc2fc
+
edcc2fc
+
edcc2fc
+#endif /* __GDM_LAYOUT_OPTION_WIDGET_H */
edcc2fc
diff -up /dev/null gdm-2.22.0/gui/simple-greeter/gdm-layouts.c
edcc2fc
--- /dev/null	2008-05-05 08:19:33.312003896 -0400
edcc2fc
+++ gdm-2.22.0/gui/simple-greeter/gdm-layouts.c	2008-05-05 11:28:39.000000000 -0400
edcc2fc
@@ -0,0 +1,204 @@
edcc2fc
+/* -*- Mode: C; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 8 -*-
edcc2fc
+ *
edcc2fc
+ * Copyright 2008  Red Hat, Inc,
edcc2fc
+ *
edcc2fc
+ * This program is free software; you can redistribute it and/or modify
edcc2fc
+ * it under the terms of the GNU General Public License as published by
edcc2fc
+ * the Free Software Foundation; either version 2 of the License, or
edcc2fc
+ * (at your option) any later version.
edcc2fc
+ *
edcc2fc
+ * This program is distributed in the hope that it will be useful,
edcc2fc
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
edcc2fc
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
edcc2fc
+ * GNU General Public License for more details.
edcc2fc
+ *
edcc2fc
+ * You should have received a copy of the GNU General Public License
edcc2fc
+ * along with this program; if not, write to the Free Software
edcc2fc
+ * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
edcc2fc
+ *
edcc2fc
+ * Written by : Matthias Clasen
edcc2fc
+ */
edcc2fc
+
edcc2fc
+#include "config.h"
edcc2fc
+
edcc2fc
+#include <string.h>
edcc2fc
+
edcc2fc
+#include <glib.h>
edcc2fc
+
edcc2fc
+#include <gdk/gdkx.h>
edcc2fc
+#include <libxklavier/xklavier.h>
edcc2fc
+#include <gconf/gconf-client.h>
edcc2fc
+
edcc2fc
+#include "gdm-layouts.h"
edcc2fc
+
edcc2fc
+static XklEngine *engine = NULL;
edcc2fc
+static XklConfigRegistry *config_registry = NULL;
edcc2fc
+static XklConfigRec *initial_config = NULL;
edcc2fc
+
edcc2fc
+static void
edcc2fc
+init_xkl (void)
edcc2fc
+{
edcc2fc
+	if (config_registry == NULL) {
edcc2fc
+		engine = xkl_engine_get_instance (GDK_DISPLAY ());
edcc2fc
+		xkl_engine_backup_names_prop (engine);
edcc2fc
+		config_registry = xkl_config_registry_get_instance (engine);
edcc2fc
+		xkl_config_registry_load (config_registry);
edcc2fc
+
edcc2fc
+		initial_config = xkl_config_rec_new ();
edcc2fc
+		if (!xkl_config_rec_get_from_backup (initial_config, engine)) {
edcc2fc
+			g_warning ("failed to load XKB configuration");
edcc2fc
+			initial_config->model = g_strdup ("pc105");
edcc2fc
+		}
edcc2fc
+	}
edcc2fc
+}
edcc2fc
+
edcc2fc
+static char *
edcc2fc
+xci_desc_to_utf8 (XklConfigItem * ci)
edcc2fc
+{
edcc2fc
+        char *sd = g_strstrip (ci->description);
edcc2fc
+        return sd[0] == 0 ? g_strdup (ci->name) :
edcc2fc
+            g_locale_to_utf8 (sd, -1, NULL, NULL, NULL);
edcc2fc
+}
edcc2fc
+
edcc2fc
+gchar *
edcc2fc
+gdm_get_layout_from_name (const char *name)
edcc2fc
+{
edcc2fc
+	XklConfigItem *item;
edcc2fc
+	gchar *layout, *variant, *result;
edcc2fc
+	char *id1, *id2, *p;
edcc2fc
+
edcc2fc
+	init_xkl ();
edcc2fc
+
edcc2fc
+	id1 = g_strdup (name);
edcc2fc
+	p = strchr (id1, '\t');
edcc2fc
+
edcc2fc
+	if (p) {
edcc2fc
+		id2 = p + 1;
edcc2fc
+		*p = 0;
edcc2fc
+	}
edcc2fc
+	else 
edcc2fc
+		id2 = NULL;
edcc2fc
+
edcc2fc
+	item = xkl_config_item_new ();
edcc2fc
+
edcc2fc
+	g_snprintf (item->name, XKL_MAX_CI_NAME_LENGTH, id1);
edcc2fc
+	if (xkl_config_registry_find_layout (config_registry, item))
edcc2fc
+		layout = xci_desc_to_utf8 (item);
edcc2fc
+	else
edcc2fc
+		layout =  g_strdup_printf ("Layout %s", id1);
edcc2fc
+
edcc2fc
+	if (id2) {
edcc2fc
+		g_snprintf (item->name, XKL_MAX_CI_NAME_LENGTH, id2);
edcc2fc
+		if (xkl_config_registry_find_variant (config_registry, id1, item))
edcc2fc
+			variant = xci_desc_to_utf8 (item);
edcc2fc
+		else
edcc2fc
+			variant = g_strdup_printf ("Variant %s", id2);
edcc2fc
+	}
edcc2fc
+	else 
edcc2fc
+		variant = NULL;
edcc2fc
+
edcc2fc
+	g_object_unref (item);
edcc2fc
+
edcc2fc
+	g_free (id1);
edcc2fc
+
edcc2fc
+	if (variant) {
edcc2fc
+		result = g_strdup_printf ("%s (%s)", layout, variant);
edcc2fc
+		g_free (layout);
edcc2fc
+		g_free (variant);
edcc2fc
+	}
edcc2fc
+	else
edcc2fc
+		result = layout;
edcc2fc
+
edcc2fc
+	return result;
edcc2fc
+}
edcc2fc
+
edcc2fc
+typedef struct {
edcc2fc
+	GSList *list;
edcc2fc
+	char *layout;
edcc2fc
+} LayoutData;
edcc2fc
+
edcc2fc
+static void
edcc2fc
+add_variant (XklConfigRegistry   *config,
edcc2fc
+             const XklConfigItem *item,
edcc2fc
+             gpointer             data)
edcc2fc
+{
edcc2fc
+	LayoutData *ldata = data;
edcc2fc
+
edcc2fc
+	ldata->list = g_slist_prepend (ldata->list, g_strdup_printf  ("%s\t%s", ldata->layout, item->name));
edcc2fc
+}
edcc2fc
+
edcc2fc
+static void
edcc2fc
+add_layout (XklConfigRegistry   *config,
edcc2fc
+            const XklConfigItem *item,
edcc2fc
+            gpointer             data)
edcc2fc
+{
edcc2fc
+	LayoutData *ldata = data;
edcc2fc
+
edcc2fc
+	ldata->layout = item->name;
edcc2fc
+	ldata->list = g_slist_prepend (ldata->list, g_strdup (item->name));
edcc2fc
+	xkl_config_registry_foreach_layout_variant (config, item->name, add_variant, data);
edcc2fc
+	ldata->layout = NULL;
edcc2fc
+}
edcc2fc
+
edcc2fc
+char **
edcc2fc
+gdm_get_all_layout_names (void)
edcc2fc
+{
edcc2fc
+	GSList *l;
edcc2fc
+	int len, i;
edcc2fc
+        char **layouts;
edcc2fc
+	LayoutData data;
edcc2fc
+
edcc2fc
+	init_xkl ();
edcc2fc
+
edcc2fc
+	data.list = NULL;
edcc2fc
+	data.layout = NULL;
edcc2fc
+
edcc2fc
+	xkl_config_registry_foreach_layout (config_registry, add_layout, &data);
edcc2fc
+
edcc2fc
+	len = g_slist_length (data.list);
edcc2fc
+
edcc2fc
+        layouts = g_new (char *, len + 1);
edcc2fc
+	layouts[len] = NULL;
edcc2fc
+
edcc2fc
+	for (i = 0, l = data.list; i < len; i++, l = l->next) 
edcc2fc
+		layouts[len - i - 1] = l->data;
edcc2fc
+
edcc2fc
+	g_slist_free (data.list);
edcc2fc
+
edcc2fc
+        return layouts;
edcc2fc
+}
edcc2fc
+
edcc2fc
+void
edcc2fc
+gdm_layout_activate (const char *layout)
edcc2fc
+{
edcc2fc
+	XklConfigRec *config;
edcc2fc
+	char *p;
edcc2fc
+
edcc2fc
+	init_xkl ();
edcc2fc
+
edcc2fc
+	config = xkl_config_rec_new ();
edcc2fc
+	config->model = g_strdup (initial_config->model);
edcc2fc
+
edcc2fc
+	if (layout == NULL) {
edcc2fc
+		config->layouts = g_strdupv (initial_config->layouts);
edcc2fc
+		config->variants = g_strdupv (initial_config->variants);
edcc2fc
+		config->options = g_strdupv (initial_config->options);
edcc2fc
+	}
edcc2fc
+	else {
edcc2fc
+		config->layouts = g_new0 (gchar *, 2);
edcc2fc
+		config->layouts[0] = g_strdup (layout);
edcc2fc
+
edcc2fc
+		p = strchr (config->layouts[0], '\t');
edcc2fc
+		if (p) {
edcc2fc
+				
edcc2fc
+			config->variants = g_new0 (gchar *, 2);
edcc2fc
+			config->layouts[0][p - config->layouts[0]] = 0;
edcc2fc
+			config->variants[0] = g_strdup (p + 1);	
edcc2fc
+		}
edcc2fc
+	}
edcc2fc
+
edcc2fc
+	xkl_config_rec_activate (config, engine);
edcc2fc
+	
edcc2fc
+	g_object_unref (config);
edcc2fc
+}
edcc2fc
+
edcc2fc
diff -up /dev/null gdm-2.22.0/gui/simple-greeter/gdm-layouts.h
edcc2fc
--- /dev/null	2008-05-05 08:19:33.312003896 -0400
edcc2fc
+++ gdm-2.22.0/gui/simple-greeter/gdm-layouts.h	2008-05-05 11:28:39.000000000 -0400
edcc2fc
@@ -0,0 +1,33 @@
edcc2fc
+/* -*- Modex: C; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 8 -*-
edcc2fc
+ *
edcc2fc
+ * Copyright 2008 Red Hat, Inc.
edcc2fc
+ *
edcc2fc
+ * This program is free software; you can redistribute it and/or modify
edcc2fc
+ * it under the terms of the GNU General Public License as published by
edcc2fc
+ * the Free Software Foundation; either version 2 of the License, or
edcc2fc
+ * (at your option) any later version.
edcc2fc
+ *
edcc2fc
+ * This program is distributed in the hope that it will be useful,
edcc2fc
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
edcc2fc
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
edcc2fc
+ * GNU General Public License for more details.
edcc2fc
+ *
edcc2fc
+ * You should have received a copy of the GNU General Public License
edcc2fc
+ * along with this program; if not, write to the Free Software
edcc2fc
+ * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
edcc2fc
+ *
edcc2fc
+ * Written by: Matthias Clasen
edcc2fc
+ */
edcc2fc
+
edcc2fc
+#ifndef __GDM_LAYOUTS_H
edcc2fc
+#define __GDM_LAYOUTS_H
edcc2fc
+
edcc2fc
+G_BEGIN_DECLS
edcc2fc
+
edcc2fc
+char *        gdm_get_layout_from_name  (const char *name);
edcc2fc
+char **       gdm_get_all_layout_names  (void);
edcc2fc
+void          gdm_layout_activate       (const char *layout);
edcc2fc
+
edcc2fc
+G_END_DECLS
edcc2fc
+
edcc2fc
+#endif /* __GDM_LAYOUT_CHOOSER_WIDGET_H */
edcc2fc
diff -up gdm-2.22.0/gui/simple-greeter/gdm-simple-greeter.schemas.in.keyboard-chooser gdm-2.22.0/gui/simple-greeter/gdm-simple-greeter.schemas.in
edcc2fc
--- gdm-2.22.0/gui/simple-greeter/gdm-simple-greeter.schemas.in.keyboard-chooser	2008-05-01 18:35:35.000000000 -0400
edcc2fc
+++ gdm-2.22.0/gui/simple-greeter/gdm-simple-greeter.schemas.in	2008-05-05 11:28:39.000000000 -0400
edcc2fc
@@ -226,5 +226,29 @@
edcc2fc
       </locale>
edcc2fc
     </schema>
edcc2fc
 
edcc2fc
+    <schema>
edcc2fc
+      <key>/schemas/apps/gdm/simple-greeter/settings-manager-plugins/keyboard/active</key>
edcc2fc
+      <applyto>/apps/gdm/simple-greeter/settings-manager-plugins/keyboard/active</applyto>
edcc2fc
+      <owner>gdm-simple-greeter</owner>
edcc2fc
+      <type>bool</type>
edcc2fc
+      <default>TRUE</default>
edcc2fc
+      <locale name="C">
edcc2fc
+        <short>True if the keyboard settings manager plugin is enabled.</short>
edcc2fc
+        <long>Set to True to enable the keyboard  settings manager plugin.</long>
edcc2fc
+      </locale>
edcc2fc
+    </schema>
edcc2fc
+
edcc2fc
+    <schema>
edcc2fc
+      <key>/schemas/apps/gdm/simple-greeter/settings-manager-plugins/keyboard/priority</key>
edcc2fc
+      <applyto>/apps/gdm/simple-greeter/settings-manager-plugins/keyboard/priority</applyto>
edcc2fc
+      <owner>gdm-simple-greeter</owner>
edcc2fc
+      <type>int</type>
edcc2fc
+      <default>6</default>
edcc2fc
+      <locale name="C">
edcc2fc
+        <short>True if the keyboard settings manager plugin is enabled.</short>
edcc2fc
+        <long>Set to True to enable the keyboard  settings manager plugin.</long>
edcc2fc
+      </locale>
edcc2fc
+    </schema>
edcc2fc
+
edcc2fc
   </schemalist>
edcc2fc
 </gconfschemafile>
edcc2fc
diff -up gdm-2.22.0/gui/simple-greeter/Makefile.am.keyboard-chooser gdm-2.22.0/gui/simple-greeter/Makefile.am
edcc2fc
--- gdm-2.22.0/gui/simple-greeter/Makefile.am.keyboard-chooser	2008-04-17 23:29:26.000000000 -0400
edcc2fc
+++ gdm-2.22.0/gui/simple-greeter/Makefile.am	2008-05-05 11:29:03.000000000 -0400
edcc2fc
@@ -123,6 +123,14 @@ test_greeter_panel_SOURCES = 	\
edcc2fc
 	gdm-language-chooser-dialog.c	\
edcc2fc
 	gdm-language-option-widget.h	\
edcc2fc
 	gdm-language-option-widget.c	\
edcc2fc
+	gdm-layout-chooser-widget.h	\
edcc2fc
+	gdm-layout-chooser-widget.c	\
edcc2fc
+	gdm-layout-chooser-dialog.h	\
edcc2fc
+	gdm-layout-chooser-dialog.c	\
edcc2fc
+	gdm-layout-option-widget.h	\
edcc2fc
+	gdm-layout-option-widget.c	\
edcc2fc
+	gdm-layouts.h			\
edcc2fc
+	gdm-layouts.c			\
edcc2fc
 	gdm-sessions.h			\
edcc2fc
 	gdm-sessions.c			\
edcc2fc
 	gdm-session-option-widget.h	\
edcc2fc
@@ -276,6 +284,14 @@ gdm_simple_greeter_SOURCES =  		\
edcc2fc
 	gdm-languages.c			\
edcc2fc
 	gdm-language-chooser-widget.h	\
edcc2fc
 	gdm-language-chooser-widget.c	\
edcc2fc
+	gdm-layout-chooser-widget.h	\
edcc2fc
+	gdm-layout-chooser-widget.c	\
edcc2fc
+	gdm-layout-chooser-dialog.h	\
edcc2fc
+	gdm-layout-chooser-dialog.c	\
edcc2fc
+	gdm-layout-option-widget.h	\
edcc2fc
+	gdm-layout-option-widget.c	\
edcc2fc
+	gdm-layouts.h			\
edcc2fc
+	gdm-layouts.c			\
edcc2fc
 	locarchive.h			\
edcc2fc
 	gdm-language-chooser-dialog.h	\
edcc2fc
 	gdm-language-chooser-dialog.c	\