blob: 42b99e431d65983c0806155d01fd6aa981f62ea4 [file] [log] [blame]
// Copyright (c) 2012 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "chrome/browser/ui/login/login_prompt.h"
#include "base/strings/string16.h"
#include "base/strings/utf_string_conversions.h"
#include "chrome/browser/password_manager/password_manager.h"
#include "chrome/browser/tab_contents/tab_util.h"
#include "chrome/browser/ui/views/constrained_window_views.h"
#include "chrome/browser/ui/views/login_view.h"
#include "chrome/common/chrome_switches.h"
#include "components/web_modal/web_contents_modal_dialog_host.h"
#include "components/web_modal/web_contents_modal_dialog_manager.h"
#include "components/web_modal/web_contents_modal_dialog_manager_delegate.h"
#include "content/public/browser/browser_thread.h"
#include "content/public/browser/render_view_host.h"
#include "content/public/browser/web_contents.h"
#include "content/public/browser/web_contents_view.h"
#include "grit/generated_resources.h"
#include "net/url_request/url_request.h"
#include "ui/base/l10n/l10n_util.h"
#include "ui/views/widget/widget.h"
#include "ui/views/window/dialog_delegate.h"
using autofill::PasswordForm;
using content::BrowserThread;
using content::WebContents;
using web_modal::WebContentsModalDialogManager;
using web_modal::WebContentsModalDialogManagerDelegate;
// ----------------------------------------------------------------------------
// LoginHandlerViews
// This class simply forwards the authentication from the LoginView (on
// the UI thread) to the net::URLRequest (on the I/O thread).
// This class uses ref counting to ensure that it lives until all InvokeLaters
// have been called.
class LoginHandlerViews : public LoginHandler,
public views::DialogDelegate {
public:
LoginHandlerViews(net::AuthChallengeInfo* auth_info, net::URLRequest* request)
: LoginHandler(auth_info, request),
login_view_(NULL),
dialog_(NULL) {
}
// LoginModelObserver implementation.
virtual void OnAutofillDataAvailable(const string16& username,
const string16& password) OVERRIDE {
// Nothing to do here since LoginView takes care of autofill for win.
}
virtual void OnLoginModelDestroying() OVERRIDE {}
// views::DialogDelegate methods:
virtual string16 GetDialogButtonLabel(
ui::DialogButton button) const OVERRIDE {
if (button == ui::DIALOG_BUTTON_OK)
return l10n_util::GetStringUTF16(IDS_LOGIN_DIALOG_OK_BUTTON_LABEL);
return DialogDelegate::GetDialogButtonLabel(button);
}
virtual string16 GetWindowTitle() const OVERRIDE {
return l10n_util::GetStringUTF16(IDS_LOGIN_DIALOG_TITLE);
}
virtual void WindowClosing() OVERRIDE {
DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
WebContents* tab = GetWebContentsForLogin();
if (tab)
tab->GetRenderViewHost()->SetIgnoreInputEvents(false);
// Reference is no longer valid.
dialog_ = NULL;
CancelAuth();
}
virtual void DeleteDelegate() OVERRIDE {
DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
// The widget is going to delete itself; clear our pointer.
dialog_ = NULL;
SetModel(NULL);
ReleaseSoon();
}
virtual ui::ModalType GetModalType() const OVERRIDE {
#if defined(USE_ASH)
return ui::MODAL_TYPE_CHILD;
#else
return views::WidgetDelegate::GetModalType();
#endif
}
virtual bool Cancel() OVERRIDE {
DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
CancelAuth();
return true;
}
virtual bool Accept() OVERRIDE {
DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
SetAuth(login_view_->GetUsername(), login_view_->GetPassword());
return true;
}
// TODO(wittman): Remove this override once we move to the new style frame
// view on all dialogs.
virtual views::NonClientFrameView* CreateNonClientFrameView(
views::Widget* widget) OVERRIDE {
return CreateConstrainedStyleNonClientFrameView(
widget,
GetWebContentsForLogin()->GetBrowserContext());
}
virtual views::View* GetInitiallyFocusedView() OVERRIDE {
return login_view_->GetInitiallyFocusedView();
}
virtual views::View* GetContentsView() OVERRIDE {
return login_view_;
}
virtual views::Widget* GetWidget() OVERRIDE {
return login_view_->GetWidget();
}
virtual const views::Widget* GetWidget() const OVERRIDE {
return login_view_->GetWidget();
}
// LoginHandler:
virtual void BuildViewForPasswordManager(
PasswordManager* manager,
const string16& explanation) OVERRIDE {
DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
// Create a new LoginView and set the model for it. The model (password
// manager) is owned by the WebContents, but the view is parented to the
// browser window, so the view may be destroyed after the password
// manager. The view listens for model destruction and unobserves
// accordingly.
login_view_ = new LoginView(explanation, manager);
// Scary thread safety note: This can potentially be called *after* SetAuth
// or CancelAuth (say, if the request was cancelled before the UI thread got
// control). However, that's OK since any UI interaction in those functions
// will occur via an InvokeLater on the UI thread, which is guaranteed
// to happen after this is called (since this was InvokeLater'd first).
WebContents* requesting_contents = GetWebContentsForLogin();
WebContentsModalDialogManager* web_contents_modal_dialog_manager =
WebContentsModalDialogManager::FromWebContents(requesting_contents);
WebContentsModalDialogManagerDelegate* modal_delegate =
web_contents_modal_dialog_manager->delegate();
CHECK(modal_delegate);
dialog_ = views::Widget::CreateWindowAsFramelessChild(
this,
requesting_contents->GetView()->GetNativeView(),
modal_delegate->GetWebContentsModalDialogHost()->GetHostView());
web_contents_modal_dialog_manager->ShowDialog(dialog_->GetNativeView());
web_contents_modal_dialog_manager->SetCloseOnInterstitialWebUI(
dialog_->GetNativeView(), true);
NotifyAuthNeeded();
}
virtual void CloseDialog() OVERRIDE {
// The hosting widget may have been freed.
if (dialog_)
dialog_->Close();
}
private:
friend class base::RefCountedThreadSafe<LoginHandlerViews>;
friend class LoginPrompt;
virtual ~LoginHandlerViews() {}
// The LoginView that contains the user's login information
LoginView* login_view_;
views::Widget* dialog_;
DISALLOW_COPY_AND_ASSIGN(LoginHandlerViews);
};
// static
LoginHandler* LoginHandler::Create(net::AuthChallengeInfo* auth_info,
net::URLRequest* request) {
return new LoginHandlerViews(auth_info, request);
}