/* * empathy-password-dialog.c - Source for EmpathyPasswordDialog * Copyright (C) 2010 Collabora Ltd. * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA */ #include #include "empathy-password-dialog.h" #include #define DEBUG_FLAG EMPATHY_DEBUG_SASL #include #include G_DEFINE_TYPE (EmpathyPasswordDialog, empathy_password_dialog, GTK_TYPE_MESSAGE_DIALOG) enum { PROP_HANDLER = 1, LAST_PROPERTY, }; typedef struct { EmpathyServerSASLHandler *handler; GtkWidget *entry; GtkWidget *ticky; GtkWidget *ok_button; gboolean grabbing; gboolean dispose_run; } EmpathyPasswordDialogPriv; static void empathy_password_dialog_get_property (GObject *object, guint property_id, GValue *value, GParamSpec *pspec) { EmpathyPasswordDialogPriv *priv = EMPATHY_PASSWORD_DIALOG (object)->priv; switch (property_id) { case PROP_HANDLER: g_value_set_object (value, priv->handler); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec); break; } } static void empathy_password_dialog_set_property (GObject *object, guint property_id, const GValue *value, GParamSpec *pspec) { EmpathyPasswordDialogPriv *priv = EMPATHY_PASSWORD_DIALOG (object)->priv; switch (property_id) { case PROP_HANDLER: g_assert (priv->handler == NULL); /* construct only */ priv->handler = g_value_dup_object (value); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec); break; } } static void empathy_password_dialog_dispose (GObject *object) { EmpathyPasswordDialogPriv *priv = EMPATHY_PASSWORD_DIALOG (object)->priv; if (priv->dispose_run) return; priv->dispose_run = TRUE; tp_clear_object (&priv->handler); G_OBJECT_CLASS (empathy_password_dialog_parent_class)->dispose (object); } static void password_dialog_response_cb (GtkDialog *dialog, gint response, gpointer user_data) { EmpathyPasswordDialogPriv *priv = EMPATHY_PASSWORD_DIALOG (user_data)->priv; if (response == GTK_RESPONSE_OK) { empathy_server_sasl_handler_provide_password (priv->handler, gtk_entry_get_text (GTK_ENTRY (priv->entry)), gtk_toggle_button_get_active (GTK_TOGGLE_BUTTON (priv->ticky))); } else { empathy_server_sasl_handler_cancel (priv->handler); } gtk_widget_destroy (GTK_WIDGET (dialog)); } static void clear_icon_released_cb (GtkEntry *entry, GtkEntryIconPosition icon_pos, GdkEvent *event, gpointer user_data) { gtk_entry_set_text (entry, ""); } static void password_entry_changed_cb (GtkEditable *entry, gpointer user_data) { EmpathyPasswordDialogPriv *priv = EMPATHY_PASSWORD_DIALOG (user_data)->priv; const gchar *str; str = gtk_entry_get_text (GTK_ENTRY (entry)); gtk_entry_set_icon_sensitive (GTK_ENTRY (entry), GTK_ENTRY_ICON_SECONDARY, !EMP_STR_EMPTY (str)); gtk_widget_set_sensitive (priv->ok_button, !EMP_STR_EMPTY (str)); } static void password_entry_activate_cb (GtkEntry *entry, EmpathyPasswordDialog *self) { gtk_dialog_response (GTK_DIALOG (self), GTK_RESPONSE_OK); } static gboolean password_dialog_grab_keyboard (GtkWidget *widget, GdkEvent *event, gpointer user_data) { EmpathyPasswordDialogPriv *priv = EMPATHY_PASSWORD_DIALOG (user_data)->priv; if (!priv->grabbing) { GdkDevice *device = gdk_event_get_device (event); if (device != NULL) { GdkGrabStatus status = gdk_device_grab (device, gtk_widget_get_window (widget), GDK_OWNERSHIP_WINDOW, FALSE, GDK_ALL_EVENTS_MASK, NULL, gdk_event_get_time (event)); if (status != GDK_GRAB_SUCCESS) DEBUG ("Could not grab keyboard; grab status was %u", status); else priv->grabbing = TRUE; } else DEBUG ("Could not get the event device!"); } return FALSE; } static gboolean password_dialog_ungrab_keyboard (GtkWidget *widget, GdkEvent *event, gpointer user_data) { EmpathyPasswordDialogPriv *priv = EMPATHY_PASSWORD_DIALOG (user_data)->priv; if (priv->grabbing) { GdkDevice *device = gdk_event_get_device (event); if (device != NULL) { gdk_device_ungrab (device, gdk_event_get_time (event)); priv->grabbing = FALSE; } else DEBUG ("Could not get the event device!"); } return FALSE; } static gboolean password_dialog_window_state_changed (GtkWidget *widget, GdkEventWindowState *event, gpointer data) { GdkWindowState state = gdk_window_get_state (gtk_widget_get_window (widget)); if (state & GDK_WINDOW_STATE_WITHDRAWN || state & GDK_WINDOW_STATE_ICONIFIED || state & GDK_WINDOW_STATE_FULLSCREEN || state & GDK_WINDOW_STATE_MAXIMIZED) { password_dialog_ungrab_keyboard (widget, (GdkEvent *) event, data); } else { password_dialog_grab_keyboard (widget, (GdkEvent *) event, data); } return FALSE; } static void password_dialog_handler_invalidated_cb (EmpathyServerSASLHandler *handler, EmpathyPasswordDialog *dialog) { gtk_widget_destroy (GTK_WIDGET (dialog)); } static void empathy_password_dialog_constructed (GObject *object) { EmpathyPasswordDialog *dialog; EmpathyPasswordDialogPriv *priv; TpAccount *account; GtkWidget *icon; GtkBox *box; gchar *text; dialog = EMPATHY_PASSWORD_DIALOG (object); priv = dialog->priv; g_assert (priv->handler != NULL); priv->grabbing = FALSE; account = empathy_server_sasl_handler_get_account (priv->handler); tp_g_signal_connect_object (priv->handler, "invalidated", G_CALLBACK (password_dialog_handler_invalidated_cb), object, 0); /* dialog */ gtk_dialog_add_button (GTK_DIALOG (dialog), GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL); priv->ok_button = gtk_dialog_add_button (GTK_DIALOG (dialog), GTK_STOCK_OK, GTK_RESPONSE_OK); gtk_widget_set_sensitive (priv->ok_button, FALSE); text = g_strdup_printf (_("Enter your password for account\n%s"), tp_account_get_display_name (account)); gtk_message_dialog_set_markup (GTK_MESSAGE_DIALOG (dialog), text); g_free (text); gtk_window_set_icon_name (GTK_WINDOW (dialog), GTK_STOCK_DIALOG_AUTHENTICATION); box = GTK_BOX (gtk_dialog_get_content_area (GTK_DIALOG (dialog))); /* dialog icon */ icon = gtk_image_new_from_icon_name (tp_account_get_icon_name (account), GTK_ICON_SIZE_DIALOG); gtk_message_dialog_set_image (GTK_MESSAGE_DIALOG (dialog), icon); gtk_widget_show (icon); /* entry */ priv->entry = gtk_entry_new (); gtk_entry_set_visibility (GTK_ENTRY (priv->entry), FALSE); /* entry clear icon */ gtk_entry_set_icon_from_stock (GTK_ENTRY (priv->entry), GTK_ENTRY_ICON_SECONDARY, GTK_STOCK_CLEAR); gtk_entry_set_icon_sensitive (GTK_ENTRY (priv->entry), GTK_ENTRY_ICON_SECONDARY, FALSE); g_signal_connect (priv->entry, "icon-release", G_CALLBACK (clear_icon_released_cb), NULL); g_signal_connect (priv->entry, "changed", G_CALLBACK (password_entry_changed_cb), dialog); g_signal_connect (priv->entry, "activate", G_CALLBACK (password_entry_activate_cb), dialog); gtk_box_pack_start (box, priv->entry, FALSE, FALSE, 0); gtk_widget_show (priv->entry); /* remember password ticky box */ priv->ticky = gtk_check_button_new_with_label (_("Remember password")); gtk_box_pack_start (box, priv->ticky, FALSE, FALSE, 0); /* only show it if we actually support it */ if (empathy_server_sasl_handler_can_save_response_somewhere (priv->handler)) gtk_widget_show (priv->ticky); g_signal_connect (dialog, "response", G_CALLBACK (password_dialog_response_cb), dialog); g_signal_connect (dialog, "window-state-event", G_CALLBACK (password_dialog_window_state_changed), dialog); g_signal_connect (dialog, "map-event", G_CALLBACK (password_dialog_grab_keyboard), dialog); g_signal_connect (dialog, "unmap-event", G_CALLBACK (password_dialog_ungrab_keyboard), dialog); gtk_widget_grab_focus (priv->entry); gtk_window_set_position (GTK_WINDOW (dialog), GTK_WIN_POS_CENTER_ALWAYS); } static void empathy_password_dialog_init (EmpathyPasswordDialog *self) { self->priv = G_TYPE_INSTANCE_GET_PRIVATE (self, EMPATHY_TYPE_PASSWORD_DIALOG, EmpathyPasswordDialogPriv); } static void empathy_password_dialog_class_init (EmpathyPasswordDialogClass *klass) { GParamSpec *pspec; GObjectClass *oclass = G_OBJECT_CLASS (klass); g_type_class_add_private (klass, sizeof (EmpathyPasswordDialogPriv)); oclass->set_property = empathy_password_dialog_set_property; oclass->get_property = empathy_password_dialog_get_property; oclass->dispose = empathy_password_dialog_dispose; oclass->constructed = empathy_password_dialog_constructed; pspec = g_param_spec_object ("handler", "The EmpathyServerSASLHandler", "The EmpathyServerSASLHandler to be used.", EMPATHY_TYPE_SERVER_SASL_HANDLER, G_PARAM_CONSTRUCT_ONLY | G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS); g_object_class_install_property (oclass, PROP_HANDLER, pspec); } GtkWidget * empathy_password_dialog_new (EmpathyServerSASLHandler *handler) { g_assert (EMPATHY_IS_SERVER_SASL_HANDLER (handler)); return g_object_new (EMPATHY_TYPE_PASSWORD_DIALOG, "handler", handler, NULL); }