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