sad_tab_view.cc revision 5d1f7b1de12d16ceb2c938c56701a3e8bfa558f7
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/sad_tab_view.h"
6
7#include <string>
8
9#include "base/metrics/field_trial.h"
10#include "base/metrics/histogram.h"
11#include "base/strings/utf_string_conversions.h"
12#include "chrome/browser/feedback/feedback_util.h"
13#include "chrome/browser/ui/browser.h"
14#include "chrome/browser/ui/browser_finder.h"
15#include "chrome/browser/ui/chrome_pages.h"
16#include "chrome/common/url_constants.h"
17#include "content/public/browser/navigation_controller.h"
18#include "content/public/browser/web_contents.h"
19#include "content/public/browser/web_contents_view.h"
20#include "grit/generated_resources.h"
21#include "grit/theme_resources.h"
22#include "ui/base/l10n/l10n_util.h"
23#include "ui/base/resource/resource_bundle.h"
24#include "ui/views/background.h"
25#include "ui/views/controls/button/label_button.h"
26#include "ui/views/controls/button/label_button_border.h"
27#include "ui/views/controls/image_view.h"
28#include "ui/views/controls/label.h"
29#include "ui/views/controls/link.h"
30#include "ui/views/layout/grid_layout.h"
31#include "ui/views/widget/widget.h"
32
33using content::OpenURLParams;
34using content::WebContents;
35
36namespace {
37
38const int kPadding = 20;
39const float kMessageSize = 0.65f;
40const SkColor kTextColor = SK_ColorWHITE;
41const SkColor kCrashColor = SkColorSetRGB(35, 48, 64);
42const SkColor kKillColor = SkColorSetRGB(57, 48, 88);
43
44const char kCategoryTagCrash[] = "Crash";
45
46}  // namespace
47
48SadTabView::SadTabView(WebContents* web_contents, chrome::SadTabKind kind)
49    : web_contents_(web_contents),
50      kind_(kind),
51      painted_(false),
52      message_(NULL),
53      help_link_(NULL),
54      feedback_link_(NULL),
55      reload_button_(NULL) {
56  DCHECK(web_contents);
57
58  // Sometimes the user will never see this tab, so keep track of the total
59  // number of creation events to compare to display events.
60  // TODO(jamescook): Remove this after R20 stable.  Keep it for now so we can
61  // compare R20 to earlier versions.
62  UMA_HISTOGRAM_COUNTS("SadTab.Created", kind_);
63
64  // These stats should use the same counting approach and bucket size used for
65  // tab discard events in chromeos::OomPriorityManager so they can be
66  // directly compared.
67  // TODO(jamescook): Maybe track time between sad tabs?
68  switch (kind_) {
69    case chrome::SAD_TAB_KIND_CRASHED: {
70      static int crashed = 0;
71      crashed++;
72      UMA_HISTOGRAM_CUSTOM_COUNTS(
73          "Tabs.SadTab.CrashCreated", crashed, 1, 1000, 50);
74      break;
75    }
76    case chrome::SAD_TAB_KIND_KILLED: {
77      static int killed = 0;
78      killed++;
79      UMA_HISTOGRAM_CUSTOM_COUNTS(
80          "Tabs.SadTab.KillCreated", killed, 1, 1000, 50);
81      break;
82    }
83    default:
84      NOTREACHED();
85  }
86
87  // Set the background color.
88  set_background(views::Background::CreateSolidBackground(
89      (kind_ == chrome::SAD_TAB_KIND_CRASHED) ? kCrashColor : kKillColor));
90
91  views::GridLayout* layout = new views::GridLayout(this);
92  SetLayoutManager(layout);
93
94  const int column_set_id = 0;
95  views::ColumnSet* columns = layout->AddColumnSet(column_set_id);
96  columns->AddPaddingColumn(1, kPadding);
97  columns->AddColumn(views::GridLayout::CENTER, views::GridLayout::CENTER,
98                     0, views::GridLayout::USE_PREF, 0, 0);
99  columns->AddPaddingColumn(1, kPadding);
100
101  views::ImageView* image = new views::ImageView();
102  image->SetImage(ui::ResourceBundle::GetSharedInstance().GetImageSkiaNamed(
103      (kind_ == chrome::SAD_TAB_KIND_CRASHED) ? IDR_SAD_TAB : IDR_KILLED_TAB));
104  layout->StartRowWithPadding(0, column_set_id, 1, kPadding);
105  layout->AddView(image);
106
107  views::Label* title = CreateLabel(l10n_util::GetStringUTF16(
108      (kind_ == chrome::SAD_TAB_KIND_CRASHED) ?
109          IDS_SAD_TAB_TITLE : IDS_KILLED_TAB_TITLE));
110  ui::ResourceBundle& rb = ui::ResourceBundle::GetSharedInstance();
111  title->SetFontList(rb.GetFontList(ui::ResourceBundle::MediumFont));
112  layout->StartRowWithPadding(0, column_set_id, 0, kPadding);
113  layout->AddView(title);
114
115  message_ = CreateLabel(l10n_util::GetStringUTF16(
116      (kind_ == chrome::SAD_TAB_KIND_CRASHED) ?
117          IDS_SAD_TAB_MESSAGE : IDS_KILLED_TAB_MESSAGE));
118  message_->SetMultiLine(true);
119  layout->StartRowWithPadding(0, column_set_id, 0, kPadding);
120  layout->AddView(message_);
121
122  if (web_contents_) {
123    layout->StartRowWithPadding(0, column_set_id, 0, kPadding);
124    reload_button_ = new views::LabelButton(
125        this,
126        l10n_util::GetStringUTF16(IDS_SAD_TAB_RELOAD_LABEL));
127    reload_button_->SetStyle(views::Button::STYLE_BUTTON);
128    // Always render the reload button with chrome style borders; never rely on
129    // native styles.
130    reload_button_->SetBorder(scoped_ptr<views::Border>(
131        new views::LabelButtonBorder(reload_button_->style())));
132    layout->AddView(reload_button_);
133
134    help_link_ = CreateLink(l10n_util::GetStringUTF16(
135        (kind_ == chrome::SAD_TAB_KIND_CRASHED) ?
136            IDS_SAD_TAB_HELP_LINK : IDS_LEARN_MORE));
137
138    if (kind_ == chrome::SAD_TAB_KIND_CRASHED) {
139      size_t offset = 0;
140      base::string16 help_text(
141          l10n_util::GetStringFUTF16(IDS_SAD_TAB_HELP_MESSAGE,
142                                     base::string16(), &offset));
143      views::Label* help_prefix = CreateLabel(help_text.substr(0, offset));
144      views::Label* help_suffix = CreateLabel(help_text.substr(offset));
145
146      const int help_column_set_id = 1;
147      views::ColumnSet* help_columns = layout->AddColumnSet(help_column_set_id);
148      help_columns->AddPaddingColumn(1, kPadding);
149      // Center three middle columns for the help's [prefix][link][suffix].
150      for (size_t column = 0; column < 3; column++)
151        help_columns->AddColumn(views::GridLayout::CENTER,
152            views::GridLayout::CENTER, 0, views::GridLayout::USE_PREF, 0, 0);
153      help_columns->AddPaddingColumn(1, kPadding);
154
155      layout->StartRowWithPadding(0, help_column_set_id, 0, kPadding);
156      layout->AddView(help_prefix);
157      layout->AddView(help_link_);
158      layout->AddView(help_suffix);
159    } else {
160      layout->StartRowWithPadding(0, column_set_id, 0, kPadding);
161      layout->AddView(help_link_);
162
163      feedback_link_ = CreateLink(
164          l10n_util::GetStringUTF16(IDS_KILLED_TAB_FEEDBACK_LINK));
165      layout->StartRowWithPadding(0, column_set_id, 0, kPadding);
166      layout->AddView(feedback_link_);
167    }
168  }
169  layout->AddPaddingRow(1, kPadding);
170}
171
172SadTabView::~SadTabView() {}
173
174void SadTabView::LinkClicked(views::Link* source, int event_flags) {
175  DCHECK(web_contents_);
176  if (source == help_link_) {
177    GURL help_url((kind_ == chrome::SAD_TAB_KIND_CRASHED) ?
178        chrome::kCrashReasonURL : chrome::kKillReasonURL);
179    OpenURLParams params(
180        help_url, content::Referrer(), CURRENT_TAB,
181        content::PAGE_TRANSITION_LINK, false);
182    web_contents_->OpenURL(params);
183  } else if (source == feedback_link_) {
184    chrome::ShowFeedbackPage(
185        chrome::FindBrowserWithWebContents(web_contents_),
186        l10n_util::GetStringUTF8(IDS_KILLED_TAB_FEEDBACK_MESSAGE),
187        std::string(kCategoryTagCrash));
188  }
189}
190
191void SadTabView::ButtonPressed(views::Button* sender,
192                               const ui::Event& event) {
193  DCHECK(web_contents_);
194  DCHECK_EQ(reload_button_, sender);
195  web_contents_->GetController().Reload(true);
196}
197
198void SadTabView::Layout() {
199  // Specify the maximum message width explicitly.
200  message_->SizeToFit(static_cast<int>(width() * kMessageSize));
201  View::Layout();
202}
203
204void SadTabView::OnPaint(gfx::Canvas* canvas) {
205  if (!painted_) {
206    // User actually saw the error, keep track for user experience stats.
207    // TODO(jamescook): Remove this after R20 stable.  Keep it for now so we can
208    // compare R20 to earlier versions.
209    UMA_HISTOGRAM_COUNTS("SadTab.Displayed", kind_);
210
211    // These stats should use the same counting approach and bucket size used
212    // for tab discard events in chromeos::OomPriorityManager so they
213    // can be directly compared.
214    switch (kind_) {
215      case chrome::SAD_TAB_KIND_CRASHED: {
216        static int crashed = 0;
217        UMA_HISTOGRAM_CUSTOM_COUNTS(
218            "Tabs.SadTab.CrashDisplayed", ++crashed, 1, 1000, 50);
219        break;
220      }
221      case chrome::SAD_TAB_KIND_KILLED: {
222        static int killed = 0;
223        UMA_HISTOGRAM_CUSTOM_COUNTS(
224            "Tabs.SadTab.KillDisplayed", ++killed, 1, 1000, 50);
225        break;
226      }
227      default:
228        NOTREACHED();
229    }
230    painted_ = true;
231  }
232  View::OnPaint(canvas);
233}
234
235void SadTabView::Show() {
236  views::Widget::InitParams sad_tab_params(
237      views::Widget::InitParams::TYPE_CONTROL);
238
239  // It is not possible to create a native_widget_win that has no parent in
240  // and later re-parent it.
241  // TODO(avi): This is a cheat. Can this be made cleaner?
242  sad_tab_params.parent = web_contents_->GetView()->GetNativeView();
243
244  set_owned_by_client();
245
246  views::Widget* sad_tab = new views::Widget;
247  sad_tab->Init(sad_tab_params);
248  sad_tab->SetContentsView(this);
249
250  views::Widget::ReparentNativeView(sad_tab->GetNativeView(),
251                                    web_contents_->GetView()->GetNativeView());
252  gfx::Rect bounds;
253  web_contents_->GetView()->GetContainerBounds(&bounds);
254  sad_tab->SetBounds(gfx::Rect(bounds.size()));
255}
256
257void SadTabView::Close() {
258  if (GetWidget())
259    GetWidget()->Close();
260}
261
262views::Label* SadTabView::CreateLabel(const base::string16& text) {
263  views::Label* label = new views::Label(text);
264  label->SetBackgroundColor(background()->get_color());
265  label->SetEnabledColor(kTextColor);
266  return label;
267}
268
269views::Link* SadTabView::CreateLink(const base::string16& text) {
270  views::Link* link = new views::Link(text);
271  link->SetBackgroundColor(background()->get_color());
272  link->SetEnabledColor(kTextColor);
273  link->set_listener(this);
274  return link;
275}
276
277namespace chrome {
278
279SadTab* SadTab::Create(content::WebContents* web_contents,
280                       SadTabKind kind) {
281  return new SadTabView(web_contents, kind);
282}
283
284}  // namespace chrome
285