OLD | NEW |
1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. | 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 | 2 // Use of this source code is governed by a BSD-style license that can be |
3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
4 | 4 |
5 #include "chrome/browser/ui/views/sad_tab_view.h" | 5 #include "components/sad_tab/views/sad_tab_view.h" |
6 | 6 |
7 #include <string> | 7 #include <string> |
8 | 8 |
9 #include "base/metrics/field_trial.h" | 9 #include "base/metrics/field_trial.h" |
10 #include "base/metrics/histogram.h" | 10 #include "base/metrics/histogram.h" |
11 #include "base/strings/utf_string_conversions.h" | 11 #include "base/strings/utf_string_conversions.h" |
12 #include "chrome/browser/ui/browser.h" | 12 #include "components/sad_tab/sad_tab_client.h" |
13 #include "chrome/browser/ui/browser_finder.h" | 13 #include "components/sad_tab/url_constants.h" |
14 #include "chrome/browser/ui/chrome_pages.h" | |
15 #include "chrome/common/url_constants.h" | |
16 #include "chrome/grit/generated_resources.h" | |
17 #include "components/feedback/feedback_util.h" | |
18 #include "content/public/browser/navigation_controller.h" | 14 #include "content/public/browser/navigation_controller.h" |
19 #include "content/public/browser/web_contents.h" | 15 #include "content/public/browser/web_contents.h" |
| 16 #include "grit/components_scaled_resources.h" |
20 #include "grit/components_strings.h" | 17 #include "grit/components_strings.h" |
21 #include "grit/theme_resources.h" | |
22 #include "ui/base/l10n/l10n_util.h" | 18 #include "ui/base/l10n/l10n_util.h" |
23 #include "ui/base/resource/resource_bundle.h" | 19 #include "ui/base/resource/resource_bundle.h" |
24 #include "ui/views/background.h" | 20 #include "ui/views/background.h" |
25 #include "ui/views/controls/button/label_button.h" | 21 #include "ui/views/controls/button/label_button.h" |
26 #include "ui/views/controls/button/label_button_border.h" | 22 #include "ui/views/controls/button/label_button_border.h" |
27 #include "ui/views/controls/image_view.h" | 23 #include "ui/views/controls/image_view.h" |
28 #include "ui/views/controls/label.h" | 24 #include "ui/views/controls/label.h" |
29 #include "ui/views/controls/link.h" | 25 #include "ui/views/controls/link.h" |
30 #include "ui/views/layout/grid_layout.h" | 26 #include "ui/views/layout/grid_layout.h" |
31 #include "ui/views/widget/widget.h" | 27 #include "ui/views/widget/widget.h" |
32 | 28 |
33 using content::OpenURLParams; | 29 using content::OpenURLParams; |
34 using content::WebContents; | 30 using content::WebContents; |
35 | 31 |
| 32 namespace sad_tab { |
| 33 |
36 namespace { | 34 namespace { |
37 | 35 |
38 const int kPadding = 20; | 36 const int kPadding = 20; |
39 const float kMessageSize = 0.65f; | 37 const float kMessageSize = 0.65f; |
40 const SkColor kTextColor = SK_ColorWHITE; | 38 const SkColor kTextColor = SK_ColorWHITE; |
41 const SkColor kCrashColor = SkColorSetRGB(35, 48, 64); | 39 const SkColor kCrashColor = SkColorSetRGB(35, 48, 64); |
42 const SkColor kKillColor = SkColorSetRGB(57, 48, 88); | 40 const SkColor kKillColor = SkColorSetRGB(57, 48, 88); |
43 | 41 |
44 const char kCategoryTagCrash[] = "Crash"; | 42 const char kCategoryTagCrash[] = "Crash"; |
45 | 43 |
46 } // namespace | 44 } // namespace |
47 | 45 |
48 SadTabView::SadTabView(WebContents* web_contents, chrome::SadTabKind kind) | 46 SadTabView::SadTabView(WebContents* web_contents, |
| 47 SadTabKind kind, |
| 48 SadTabClient* client) |
49 : web_contents_(web_contents), | 49 : web_contents_(web_contents), |
50 kind_(kind), | 50 kind_(kind), |
| 51 client_(client), |
51 painted_(false), | 52 painted_(false), |
52 message_(NULL), | 53 message_(NULL), |
53 help_link_(NULL), | 54 help_link_(NULL), |
54 feedback_link_(NULL), | 55 feedback_link_(NULL), |
55 reload_button_(NULL) { | 56 reload_button_(NULL) { |
56 DCHECK(web_contents); | 57 DCHECK(web_contents); |
57 | 58 |
58 // Sometimes the user will never see this tab, so keep track of the total | 59 // 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 // 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 // TODO(jamescook): Remove this after R20 stable. Keep it for now so we can |
61 // compare R20 to earlier versions. | 62 // compare R20 to earlier versions. |
62 UMA_HISTOGRAM_COUNTS("SadTab.Created", kind_); | 63 UMA_HISTOGRAM_COUNTS("SadTab.Created", kind_); |
63 | 64 |
64 // These stats should use the same counting approach and bucket size used for | 65 // 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 // tab discard events in chromeos::OomPriorityManager so they can be |
66 // directly compared. | 67 // directly compared. |
67 // TODO(jamescook): Maybe track time between sad tabs? | 68 // TODO(jamescook): Maybe track time between sad tabs? |
68 switch (kind_) { | 69 switch (kind_) { |
69 case chrome::SAD_TAB_KIND_CRASHED: { | 70 case SAD_TAB_KIND_CRASHED: { |
70 static int crashed = 0; | 71 static int crashed = 0; |
71 crashed++; | 72 crashed++; |
72 UMA_HISTOGRAM_CUSTOM_COUNTS( | 73 UMA_HISTOGRAM_CUSTOM_COUNTS( |
73 "Tabs.SadTab.CrashCreated", crashed, 1, 1000, 50); | 74 "Tabs.SadTab.CrashCreated", crashed, 1, 1000, 50); |
74 break; | 75 break; |
75 } | 76 } |
76 case chrome::SAD_TAB_KIND_KILLED: { | 77 case SAD_TAB_KIND_KILLED: { |
77 static int killed = 0; | 78 static int killed = 0; |
78 killed++; | 79 killed++; |
79 UMA_HISTOGRAM_CUSTOM_COUNTS( | 80 UMA_HISTOGRAM_CUSTOM_COUNTS( |
80 "Tabs.SadTab.KillCreated", killed, 1, 1000, 50); | 81 "Tabs.SadTab.KillCreated", killed, 1, 1000, 50); |
81 break; | 82 break; |
82 } | 83 } |
83 default: | 84 default: |
84 NOTREACHED(); | 85 NOTREACHED(); |
85 } | 86 } |
86 | 87 |
87 // Set the background color. | 88 // Set the background color. |
88 set_background(views::Background::CreateSolidBackground( | 89 set_background(views::Background::CreateSolidBackground( |
89 (kind_ == chrome::SAD_TAB_KIND_CRASHED) ? kCrashColor : kKillColor)); | 90 (kind_ == SAD_TAB_KIND_CRASHED) ? kCrashColor : kKillColor)); |
90 | 91 |
91 views::GridLayout* layout = new views::GridLayout(this); | 92 views::GridLayout* layout = new views::GridLayout(this); |
92 SetLayoutManager(layout); | 93 SetLayoutManager(layout); |
93 | 94 |
94 const int column_set_id = 0; | 95 const int column_set_id = 0; |
95 views::ColumnSet* columns = layout->AddColumnSet(column_set_id); | 96 views::ColumnSet* columns = layout->AddColumnSet(column_set_id); |
96 columns->AddPaddingColumn(1, kPadding); | 97 columns->AddPaddingColumn(1, kPadding); |
97 columns->AddColumn(views::GridLayout::CENTER, views::GridLayout::CENTER, | 98 columns->AddColumn(views::GridLayout::CENTER, views::GridLayout::CENTER, |
98 0, views::GridLayout::USE_PREF, 0, 0); | 99 0, views::GridLayout::USE_PREF, 0, 0); |
99 columns->AddPaddingColumn(1, kPadding); | 100 columns->AddPaddingColumn(1, kPadding); |
100 | 101 |
101 views::ImageView* image = new views::ImageView(); | 102 views::ImageView* image = new views::ImageView(); |
102 image->SetImage(ui::ResourceBundle::GetSharedInstance().GetImageSkiaNamed( | 103 image->SetImage(ui::ResourceBundle::GetSharedInstance().GetImageSkiaNamed( |
103 (kind_ == chrome::SAD_TAB_KIND_CRASHED) ? IDR_SAD_TAB : IDR_KILLED_TAB)); | 104 (kind_ == SAD_TAB_KIND_CRASHED) ? IDR_SAD_TAB : IDR_KILLED_TAB)); |
104 layout->StartRowWithPadding(0, column_set_id, 1, kPadding); | 105 layout->StartRowWithPadding(0, column_set_id, 1, kPadding); |
105 layout->AddView(image); | 106 layout->AddView(image); |
106 | 107 |
107 views::Label* title = CreateLabel(l10n_util::GetStringUTF16( | 108 views::Label* title = CreateLabel(l10n_util::GetStringUTF16( |
108 (kind_ == chrome::SAD_TAB_KIND_CRASHED) ? | 109 (kind_ == SAD_TAB_KIND_CRASHED) ? |
109 IDS_SAD_TAB_TITLE : IDS_KILLED_TAB_TITLE)); | 110 IDS_SAD_TAB_TITLE : IDS_KILLED_TAB_TITLE)); |
110 ui::ResourceBundle& rb = ui::ResourceBundle::GetSharedInstance(); | 111 ui::ResourceBundle& rb = ui::ResourceBundle::GetSharedInstance(); |
111 title->SetFontList(rb.GetFontList(ui::ResourceBundle::MediumFont)); | 112 title->SetFontList(rb.GetFontList(ui::ResourceBundle::MediumFont)); |
112 layout->StartRowWithPadding(0, column_set_id, 0, kPadding); | 113 layout->StartRowWithPadding(0, column_set_id, 0, kPadding); |
113 layout->AddView(title); | 114 layout->AddView(title); |
114 | 115 |
115 message_ = CreateLabel(l10n_util::GetStringUTF16( | 116 message_ = CreateLabel(l10n_util::GetStringUTF16( |
116 (kind_ == chrome::SAD_TAB_KIND_CRASHED) ? | 117 (kind_ == SAD_TAB_KIND_CRASHED) ? |
117 IDS_SAD_TAB_MESSAGE : IDS_KILLED_TAB_MESSAGE)); | 118 IDS_SAD_TAB_MESSAGE : IDS_KILLED_TAB_MESSAGE)); |
118 message_->SetMultiLine(true); | 119 message_->SetMultiLine(true); |
119 layout->StartRowWithPadding(0, column_set_id, 0, kPadding); | 120 layout->StartRowWithPadding(0, column_set_id, 0, kPadding); |
120 layout->AddView(message_); | 121 layout->AddView(message_); |
121 | 122 |
122 if (web_contents_) { | 123 if (web_contents_) { |
123 layout->StartRowWithPadding(0, column_set_id, 0, kPadding); | 124 layout->StartRowWithPadding(0, column_set_id, 0, kPadding); |
124 reload_button_ = new views::LabelButton( | 125 reload_button_ = new views::LabelButton( |
125 this, | 126 this, |
126 l10n_util::GetStringUTF16(IDS_SAD_TAB_RELOAD_LABEL)); | 127 l10n_util::GetStringUTF16(IDS_SAD_TAB_RELOAD_LABEL)); |
127 reload_button_->SetStyle(views::Button::STYLE_BUTTON); | 128 reload_button_->SetStyle(views::Button::STYLE_BUTTON); |
128 // Always render the reload button with chrome style borders; never rely on | 129 // Always render the reload button with chrome style borders; never rely on |
129 // native styles. | 130 // native styles. |
130 reload_button_->SetBorder(scoped_ptr<views::Border>( | 131 reload_button_->SetBorder(scoped_ptr<views::Border>( |
131 new views::LabelButtonBorder(reload_button_->style()))); | 132 new views::LabelButtonBorder(reload_button_->style()))); |
132 layout->AddView(reload_button_); | 133 layout->AddView(reload_button_); |
133 | 134 |
134 help_link_ = CreateLink(l10n_util::GetStringUTF16( | 135 help_link_ = CreateLink(l10n_util::GetStringUTF16( |
135 (kind_ == chrome::SAD_TAB_KIND_CRASHED) ? | 136 (kind_ == SAD_TAB_KIND_CRASHED) ? |
136 IDS_SAD_TAB_HELP_LINK : IDS_LEARN_MORE)); | 137 IDS_SAD_TAB_HELP_LINK : IDS_LEARN_MORE)); |
137 | 138 |
138 if (kind_ == chrome::SAD_TAB_KIND_CRASHED) { | 139 if (kind_ == SAD_TAB_KIND_CRASHED) { |
139 size_t offset = 0; | 140 size_t offset = 0; |
140 base::string16 help_text( | 141 base::string16 help_text( |
141 l10n_util::GetStringFUTF16(IDS_SAD_TAB_HELP_MESSAGE, | 142 l10n_util::GetStringFUTF16(IDS_SAD_TAB_HELP_MESSAGE, |
142 base::string16(), &offset)); | 143 base::string16(), &offset)); |
143 views::Label* help_prefix = CreateLabel(help_text.substr(0, offset)); | 144 views::Label* help_prefix = CreateLabel(help_text.substr(0, offset)); |
144 views::Label* help_suffix = CreateLabel(help_text.substr(offset)); | 145 views::Label* help_suffix = CreateLabel(help_text.substr(offset)); |
145 | 146 |
146 const int help_column_set_id = 1; | 147 const int help_column_set_id = 1; |
147 views::ColumnSet* help_columns = layout->AddColumnSet(help_column_set_id); | 148 views::ColumnSet* help_columns = layout->AddColumnSet(help_column_set_id); |
148 help_columns->AddPaddingColumn(1, kPadding); | 149 help_columns->AddPaddingColumn(1, kPadding); |
(...skipping 18 matching lines...) Expand all Loading... |
167 } | 168 } |
168 } | 169 } |
169 layout->AddPaddingRow(1, kPadding); | 170 layout->AddPaddingRow(1, kPadding); |
170 } | 171 } |
171 | 172 |
172 SadTabView::~SadTabView() {} | 173 SadTabView::~SadTabView() {} |
173 | 174 |
174 void SadTabView::LinkClicked(views::Link* source, int event_flags) { | 175 void SadTabView::LinkClicked(views::Link* source, int event_flags) { |
175 DCHECK(web_contents_); | 176 DCHECK(web_contents_); |
176 if (source == help_link_) { | 177 if (source == help_link_) { |
177 GURL help_url((kind_ == chrome::SAD_TAB_KIND_CRASHED) ? | 178 GURL help_url((kind_ == SAD_TAB_KIND_CRASHED) ? |
178 chrome::kCrashReasonURL : chrome::kKillReasonURL); | 179 kCrashReasonURL : kKillReasonURL); |
179 OpenURLParams params( | 180 OpenURLParams params( |
180 help_url, content::Referrer(), CURRENT_TAB, | 181 help_url, content::Referrer(), CURRENT_TAB, |
181 ui::PAGE_TRANSITION_LINK, false); | 182 ui::PAGE_TRANSITION_LINK, false); |
182 web_contents_->OpenURL(params); | 183 web_contents_->OpenURL(params); |
183 } else if (source == feedback_link_) { | 184 } else if (source == feedback_link_) { |
184 chrome::ShowFeedbackPage( | 185 client_->ShowFeedbackPage( |
185 chrome::FindBrowserWithWebContents(web_contents_), | 186 web_contents_, |
186 l10n_util::GetStringUTF8(IDS_KILLED_TAB_FEEDBACK_MESSAGE), | 187 l10n_util::GetStringUTF8(IDS_KILLED_TAB_FEEDBACK_MESSAGE), |
187 std::string(kCategoryTagCrash)); | 188 std::string(kCategoryTagCrash)); |
188 } | 189 } |
189 } | 190 } |
190 | 191 |
191 void SadTabView::ButtonPressed(views::Button* sender, | 192 void SadTabView::ButtonPressed(views::Button* sender, |
192 const ui::Event& event) { | 193 const ui::Event& event) { |
193 DCHECK(web_contents_); | 194 DCHECK(web_contents_); |
194 DCHECK_EQ(reload_button_, sender); | 195 DCHECK_EQ(reload_button_, sender); |
195 web_contents_->GetController().Reload(true); | 196 web_contents_->GetController().Reload(true); |
196 } | 197 } |
197 | 198 |
198 void SadTabView::Layout() { | 199 void SadTabView::Layout() { |
199 // Specify the maximum message width explicitly. | 200 // Specify the maximum message width explicitly. |
200 message_->SizeToFit(static_cast<int>(width() * kMessageSize)); | 201 message_->SizeToFit(static_cast<int>(width() * kMessageSize)); |
201 View::Layout(); | 202 View::Layout(); |
202 } | 203 } |
203 | 204 |
204 void SadTabView::OnPaint(gfx::Canvas* canvas) { | 205 void SadTabView::OnPaint(gfx::Canvas* canvas) { |
205 if (!painted_) { | 206 if (!painted_) { |
206 // User actually saw the error, keep track for user experience stats. | 207 // 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 // TODO(jamescook): Remove this after R20 stable. Keep it for now so we can |
208 // compare R20 to earlier versions. | 209 // compare R20 to earlier versions. |
209 UMA_HISTOGRAM_COUNTS("SadTab.Displayed", kind_); | 210 UMA_HISTOGRAM_COUNTS("SadTab.Displayed", kind_); |
210 | 211 |
211 // These stats should use the same counting approach and bucket size used | 212 // These stats should use the same counting approach and bucket size used |
212 // for tab discard events in chromeos::OomPriorityManager so they | 213 // for tab discard events in chromeos::OomPriorityManager so they |
213 // can be directly compared. | 214 // can be directly compared. |
214 switch (kind_) { | 215 switch (kind_) { |
215 case chrome::SAD_TAB_KIND_CRASHED: { | 216 case SAD_TAB_KIND_CRASHED: { |
216 static int crashed = 0; | 217 static int crashed = 0; |
217 UMA_HISTOGRAM_CUSTOM_COUNTS( | 218 UMA_HISTOGRAM_CUSTOM_COUNTS( |
218 "Tabs.SadTab.CrashDisplayed", ++crashed, 1, 1000, 50); | 219 "Tabs.SadTab.CrashDisplayed", ++crashed, 1, 1000, 50); |
219 break; | 220 break; |
220 } | 221 } |
221 case chrome::SAD_TAB_KIND_KILLED: { | 222 case SAD_TAB_KIND_KILLED: { |
222 static int killed = 0; | 223 static int killed = 0; |
223 UMA_HISTOGRAM_CUSTOM_COUNTS( | 224 UMA_HISTOGRAM_CUSTOM_COUNTS( |
224 "Tabs.SadTab.KillDisplayed", ++killed, 1, 1000, 50); | 225 "Tabs.SadTab.KillDisplayed", ++killed, 1, 1000, 50); |
225 break; | 226 break; |
226 } | 227 } |
227 default: | 228 default: |
228 NOTREACHED(); | 229 NOTREACHED(); |
229 } | 230 } |
230 painted_ = true; | 231 painted_ = true; |
231 } | 232 } |
(...skipping 34 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
266 } | 267 } |
267 | 268 |
268 views::Link* SadTabView::CreateLink(const base::string16& text) { | 269 views::Link* SadTabView::CreateLink(const base::string16& text) { |
269 views::Link* link = new views::Link(text); | 270 views::Link* link = new views::Link(text); |
270 link->SetBackgroundColor(background()->get_color()); | 271 link->SetBackgroundColor(background()->get_color()); |
271 link->SetEnabledColor(kTextColor); | 272 link->SetEnabledColor(kTextColor); |
272 link->set_listener(this); | 273 link->set_listener(this); |
273 return link; | 274 return link; |
274 } | 275 } |
275 | 276 |
276 namespace chrome { | |
277 | |
278 SadTab* SadTab::Create(content::WebContents* web_contents, | 277 SadTab* SadTab::Create(content::WebContents* web_contents, |
279 SadTabKind kind) { | 278 SadTabKind kind, |
280 return new SadTabView(web_contents, kind); | 279 SadTabClient* client) { |
| 280 return new SadTabView(web_contents, kind, client); |
281 } | 281 } |
282 | 282 |
283 } // namespace chrome | 283 } // namespace sad_tab |
OLD | NEW |