Home | History | Annotate | Download | only in views
      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/views/external_protocol_dialog.h"
      6 
      7 #include "base/metrics/histogram.h"
      8 #include "base/strings/string_util.h"
      9 #include "base/strings/utf_string_conversions.h"
     10 #include "chrome/browser/external_protocol/external_protocol_handler.h"
     11 #include "chrome/browser/tab_contents/tab_util.h"
     12 #include "chrome/browser/ui/external_protocol_dialog_delegate.h"
     13 #include "chrome/browser/ui/views/constrained_window_views.h"
     14 #include "content/public/browser/web_contents.h"
     15 #include "grit/chromium_strings.h"
     16 #include "grit/generated_resources.h"
     17 #include "ui/base/l10n/l10n_util.h"
     18 #include "ui/gfx/text_elider.h"
     19 #include "ui/views/controls/message_box_view.h"
     20 #include "ui/views/widget/widget.h"
     21 
     22 using content::WebContents;
     23 
     24 namespace {
     25 
     26 const int kMessageWidth = 400;
     27 
     28 }  // namespace
     29 
     30 ///////////////////////////////////////////////////////////////////////////////
     31 // ExternalProtocolHandler
     32 
     33 // static
     34 void ExternalProtocolHandler::RunExternalProtocolDialog(
     35     const GURL& url, int render_process_host_id, int routing_id) {
     36   scoped_ptr<ExternalProtocolDialogDelegate> delegate(
     37       new ExternalProtocolDialogDelegate(url,
     38                                          render_process_host_id,
     39                                          routing_id));
     40   if (delegate->program_name().empty()) {
     41     // ShellExecute won't do anything. Don't bother warning the user.
     42     return;
     43   }
     44 
     45   // Windowing system takes ownership.
     46   new ExternalProtocolDialog(delegate.PassAs<const ProtocolDialogDelegate>(),
     47                              render_process_host_id,
     48                              routing_id);
     49 }
     50 
     51 ///////////////////////////////////////////////////////////////////////////////
     52 // ExternalProtocolDialog
     53 
     54 ExternalProtocolDialog::~ExternalProtocolDialog() {
     55 }
     56 
     57 //////////////////////////////////////////////////////////////////////////////
     58 // ExternalProtocolDialog, views::DialogDelegate implementation:
     59 
     60 int ExternalProtocolDialog::GetDefaultDialogButton() const {
     61   return ui::DIALOG_BUTTON_CANCEL;
     62 }
     63 
     64 base::string16 ExternalProtocolDialog::GetDialogButtonLabel(
     65     ui::DialogButton button) const {
     66   if (button == ui::DIALOG_BUTTON_OK)
     67     return l10n_util::GetStringUTF16(IDS_EXTERNAL_PROTOCOL_OK_BUTTON_TEXT);
     68   else
     69     return l10n_util::GetStringUTF16(IDS_EXTERNAL_PROTOCOL_CANCEL_BUTTON_TEXT);
     70 }
     71 
     72 base::string16 ExternalProtocolDialog::GetWindowTitle() const {
     73   return delegate_->GetTitleText();
     74 }
     75 
     76 void ExternalProtocolDialog::DeleteDelegate() {
     77   delete this;
     78 }
     79 
     80 bool ExternalProtocolDialog::Cancel() {
     81   // We also get called back here if the user closes the dialog or presses
     82   // escape. In these cases it would be preferable to ignore the state of the
     83   // check box but MessageBox doesn't distinguish this from pressing the cancel
     84   // button.
     85   delegate_->DoCancel(delegate_->url(),
     86                       message_box_view_->IsCheckBoxSelected());
     87 
     88   // Returning true closes the dialog.
     89   return true;
     90 }
     91 
     92 bool ExternalProtocolDialog::Accept() {
     93   // We record how long it takes the user to accept an external protocol.  If
     94   // users start accepting these dialogs too quickly, we should worry about
     95   // clickjacking.
     96   UMA_HISTOGRAM_LONG_TIMES("clickjacking.launch_url",
     97                            base::TimeTicks::Now() - creation_time_);
     98 
     99   delegate_->DoAccept(delegate_->url(),
    100                       message_box_view_->IsCheckBoxSelected());
    101 
    102   // Returning true closes the dialog.
    103   return true;
    104 }
    105 
    106 views::View* ExternalProtocolDialog::GetContentsView() {
    107   return message_box_view_;
    108 }
    109 
    110 views::Widget* ExternalProtocolDialog::GetWidget() {
    111   return message_box_view_->GetWidget();
    112 }
    113 
    114 const views::Widget* ExternalProtocolDialog::GetWidget() const {
    115   return message_box_view_->GetWidget();
    116 }
    117 
    118 ///////////////////////////////////////////////////////////////////////////////
    119 // ExternalProtocolDialog, private:
    120 
    121 ExternalProtocolDialog::ExternalProtocolDialog(
    122     scoped_ptr<const ProtocolDialogDelegate> delegate,
    123     int render_process_host_id,
    124     int routing_id)
    125     : delegate_(delegate.Pass()),
    126       render_process_host_id_(render_process_host_id),
    127       routing_id_(routing_id),
    128       creation_time_(base::TimeTicks::Now()) {
    129   views::MessageBoxView::InitParams params(delegate_->GetMessageText());
    130   params.message_width = kMessageWidth;
    131   message_box_view_ = new views::MessageBoxView(params);
    132   message_box_view_->SetCheckBoxLabel(delegate_->GetCheckboxText());
    133 
    134   // Dialog is top level if we don't have a web_contents associated with us.
    135   WebContents* web_contents = tab_util::GetWebContentsByID(
    136       render_process_host_id_, routing_id_);
    137   gfx::NativeWindow parent_window = NULL;
    138   if (web_contents)
    139     parent_window = web_contents->GetTopLevelNativeWindow();
    140   CreateBrowserModalDialogViews(this, parent_window)->Show();
    141 }
    142