• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // Copyright (c) 2012 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4 
5 #include "chrome/browser/ui/login/login_prompt.h"
6 
7 #include <gtk/gtk.h>
8 
9 #include "base/strings/string16.h"
10 #include "base/strings/utf_string_conversions.h"
11 #include "chrome/browser/password_manager/password_manager.h"
12 #include "chrome/browser/tab_contents/tab_util.h"
13 #include "chrome/browser/ui/gtk/constrained_window_gtk.h"
14 #include "chrome/browser/ui/gtk/gtk_util.h"
15 #include "chrome/browser/ui/login/login_model.h"
16 #include "components/web_modal/web_contents_modal_dialog_manager.h"
17 #include "content/public/browser/browser_thread.h"
18 #include "content/public/browser/web_contents.h"
19 #include "content/public/browser/web_contents_delegate.h"
20 #include "grit/generated_resources.h"
21 #include "net/url_request/url_request.h"
22 #include "ui/base/gtk/gtk_hig_constants.h"
23 #include "ui/base/gtk/gtk_signal.h"
24 #include "ui/base/l10n/l10n_util.h"
25 #include "ui/gfx/gtk_compat.h"
26 #include "ui/gfx/scoped_gobject.h"
27 
28 using autofill::PasswordForm;
29 using content::BrowserThread;
30 using content::WebContents;
31 using web_modal::WebContentsModalDialogManager;
32 
33 // ----------------------------------------------------------------------------
34 // LoginHandlerGtk
35 
36 // This class simply forwards the authentication from the LoginView (on
37 // the UI thread) to the net::URLRequest (on the I/O thread).
38 // This class uses ref counting to ensure that it lives until all InvokeLaters
39 // have been called.
40 class LoginHandlerGtk : public LoginHandler {
41  public:
LoginHandlerGtk(net::AuthChallengeInfo * auth_info,net::URLRequest * request)42   LoginHandlerGtk(net::AuthChallengeInfo* auth_info, net::URLRequest* request)
43       : LoginHandler(auth_info, request),
44         username_entry_(NULL),
45         password_entry_(NULL),
46         ok_(NULL),
47         dialog_(NULL) {
48   }
49 
50   // LoginModelObserver implementation.
OnAutofillDataAvailable(const base::string16 & username,const base::string16 & password)51   virtual void OnAutofillDataAvailable(
52       const base::string16& username,
53       const base::string16& password) OVERRIDE {
54     DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
55 
56     // NOTE: Would be nice to use gtk_entry_get_text_length, but it is fairly
57     // new and not always in our GTK version.
58     if (strlen(gtk_entry_get_text(GTK_ENTRY(username_entry_))) == 0) {
59       gtk_entry_set_text(GTK_ENTRY(username_entry_),
60                          UTF16ToUTF8(username).c_str());
61       gtk_entry_set_text(GTK_ENTRY(password_entry_),
62                          UTF16ToUTF8(password).c_str());
63       gtk_editable_select_region(GTK_EDITABLE(username_entry_), 0, -1);
64     }
65   }
OnLoginModelDestroying()66   virtual void OnLoginModelDestroying() OVERRIDE {}
67 
68   // LoginHandler:
BuildViewForPasswordManager(PasswordManager * manager,const base::string16 & explanation)69   virtual void BuildViewForPasswordManager(
70       PasswordManager* manager,
71       const base::string16& explanation) OVERRIDE {
72     DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
73 
74     root_.reset(gtk_vbox_new(FALSE, ui::kContentAreaBorder));
75     g_object_ref_sink(root_.get());
76     g_signal_connect(root_.get(), "destroy", G_CALLBACK(OnDestroyThunk), this);
77 
78     GtkWidget* label = gtk_label_new(UTF16ToUTF8(explanation).c_str());
79     gtk_label_set_line_wrap(GTK_LABEL(label), TRUE);
80     gtk_box_pack_start(GTK_BOX(root_.get()), label, FALSE, FALSE, 0);
81 
82     username_entry_ = gtk_entry_new();
83     gtk_entry_set_activates_default(GTK_ENTRY(username_entry_), TRUE);
84 
85     password_entry_ = gtk_entry_new();
86     gtk_entry_set_activates_default(GTK_ENTRY(password_entry_), TRUE);
87     gtk_entry_set_visibility(GTK_ENTRY(password_entry_), FALSE);
88 
89     GtkWidget* table = gtk_util::CreateLabeledControlsGroup(NULL,
90         l10n_util::GetStringUTF8(IDS_LOGIN_DIALOG_USERNAME_FIELD).c_str(),
91         username_entry_,
92         l10n_util::GetStringUTF8(IDS_LOGIN_DIALOG_PASSWORD_FIELD).c_str(),
93         password_entry_,
94         NULL);
95     gtk_box_pack_start(GTK_BOX(root_.get()), table, FALSE, FALSE, 0);
96 
97     GtkWidget* hbox = gtk_hbox_new(FALSE, 12);
98     gtk_box_pack_start(GTK_BOX(root_.get()), hbox, FALSE, FALSE, 0);
99 
100     ok_ = gtk_button_new_from_stock(GTK_STOCK_OK);
101     gtk_button_set_label(
102         GTK_BUTTON(ok_),
103         l10n_util::GetStringUTF8(IDS_LOGIN_DIALOG_OK_BUTTON_LABEL).c_str());
104     g_signal_connect(ok_, "clicked", G_CALLBACK(OnOKClickedThunk), this);
105     gtk_box_pack_end(GTK_BOX(hbox), ok_, FALSE, FALSE, 0);
106 
107     GtkWidget* cancel = gtk_button_new_from_stock(GTK_STOCK_CANCEL);
108     g_signal_connect(cancel, "clicked", G_CALLBACK(OnCancelClickedThunk), this);
109     gtk_box_pack_end(GTK_BOX(hbox), cancel, FALSE, FALSE, 0);
110 
111     g_signal_connect(root_.get(), "hierarchy-changed",
112                      G_CALLBACK(OnPromptHierarchyChangedThunk), this);
113 
114     SetModel(manager);
115 
116     // Scary thread safety note: This can potentially be called *after* SetAuth
117     // or CancelAuth (say, if the request was cancelled before the UI thread got
118     // control).  However, that's OK since any UI interaction in those functions
119     // will occur via an InvokeLater on the UI thread, which is guaranteed
120     // to happen after this is called (since this was InvokeLater'd first).
121     WebContents* requesting_contents = GetWebContentsForLogin();
122     DCHECK(requesting_contents);
123 
124     dialog_ = CreateWebContentsModalDialogGtk(root_.get(), username_entry_);
125 
126     WebContentsModalDialogManager* web_contents_modal_dialog_manager =
127         WebContentsModalDialogManager::FromWebContents(requesting_contents);
128     web_contents_modal_dialog_manager->ShowDialog(dialog_);
129 
130     NotifyAuthNeeded();
131   }
132 
CloseDialog()133   virtual void CloseDialog() OVERRIDE {
134     // The hosting dialog may have been freed.
135     if (dialog_)
136       gtk_widget_destroy(dialog_);
137   }
138 
139  protected:
~LoginHandlerGtk()140   virtual ~LoginHandlerGtk() {
141   }
142 
143  private:
144   friend class LoginPrompt;
145 
146   CHROMEGTK_CALLBACK_0(LoginHandlerGtk, void, OnOKClicked);
147   CHROMEGTK_CALLBACK_0(LoginHandlerGtk, void, OnCancelClicked);
148   CHROMEGTK_CALLBACK_1(LoginHandlerGtk, void, OnPromptHierarchyChanged,
149                        GtkWidget*);
150   CHROMEGTK_CALLBACK_0(LoginHandlerGtk, void, OnDestroy);
151 
152   // The GtkWidgets that form our visual hierarchy:
153   // The root container we pass to our parent.
154   ui::ScopedGObject<GtkWidget>::Type root_;
155 
156   // GtkEntry widgets that the user types into.
157   GtkWidget* username_entry_;
158   GtkWidget* password_entry_;
159   GtkWidget* ok_;
160 
161   GtkWidget* dialog_;
162 
163   DISALLOW_COPY_AND_ASSIGN(LoginHandlerGtk);
164 };
165 
OnOKClicked(GtkWidget * sender)166 void LoginHandlerGtk::OnOKClicked(GtkWidget* sender) {
167   SetAuth(
168       UTF8ToUTF16(gtk_entry_get_text(GTK_ENTRY(username_entry_))),
169       UTF8ToUTF16(gtk_entry_get_text(GTK_ENTRY(password_entry_))));
170 }
171 
OnCancelClicked(GtkWidget * sender)172 void LoginHandlerGtk::OnCancelClicked(GtkWidget* sender) {
173   CancelAuth();
174 }
175 
OnPromptHierarchyChanged(GtkWidget * sender,GtkWidget * previous_toplevel)176 void LoginHandlerGtk::OnPromptHierarchyChanged(GtkWidget* sender,
177                                                GtkWidget* previous_toplevel) {
178   DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
179 
180   if (!gtk_widget_is_toplevel(gtk_widget_get_toplevel(ok_)))
181     return;
182 
183   // Now that we have attached ourself to the window, we can make our OK
184   // button the default action and mess with the focus.
185   gtk_widget_set_can_default(ok_, TRUE);
186   gtk_widget_grab_default(ok_);
187 }
188 
189 // static
Create(net::AuthChallengeInfo * auth_info,net::URLRequest * request)190 LoginHandler* LoginHandler::Create(net::AuthChallengeInfo* auth_info,
191                                    net::URLRequest* request) {
192   return new LoginHandlerGtk(auth_info, request);
193 }
194 
OnDestroy(GtkWidget * widget)195 void LoginHandlerGtk::OnDestroy(GtkWidget* widget) {
196   DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
197 
198   // The web contents modal dialog is going to delete itself; clear our pointer.
199   dialog_ = NULL;
200   SetModel(NULL);
201 
202   ReleaseSoon();
203 }
204