1 // Copyright (c) 2013 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.
5 #include "chrome/browser/ui/views/extensions/extension_message_bubble_view.h"
7 #include "base/strings/string_number_conversions.h"
8 #include "base/strings/string_util.h"
9 #include "base/strings/utf_string_conversions.h"
10 #include "chrome/browser/extensions/dev_mode_bubble_controller.h"
11 #include "chrome/browser/extensions/extension_action_manager.h"
12 #include "chrome/browser/extensions/extension_prefs.h"
13 #include "chrome/browser/extensions/extension_service.h"
14 #include "chrome/browser/extensions/extension_system.h"
15 #include "chrome/browser/extensions/suspicious_extension_bubble_controller.h"
16 #include "chrome/browser/profiles/profile.h"
17 #include "chrome/browser/ui/browser.h"
18 #include "chrome/browser/ui/views/frame/browser_view.h"
19 #include "chrome/browser/ui/views/toolbar/browser_actions_container.h"
20 #include "chrome/browser/ui/views/toolbar/toolbar_view.h"
21 #include "grit/locale_settings.h"
22 #include "ui/base/accessibility/accessible_view_state.h"
23 #include "ui/base/resource/resource_bundle.h"
24 #include "ui/views/controls/button/label_button.h"
25 #include "ui/views/controls/label.h"
26 #include "ui/views/controls/link.h"
27 #include "ui/views/layout/grid_layout.h"
28 #include "ui/views/view.h"
29 #include "ui/views/widget/widget.h"
34 const int kExtensionListPadding
= 10;
35 const int kInsetBottomRight
= 13;
36 const int kInsetLeft
= 14;
37 const int kInsetTop
= 9;
38 const int kHeadlineMessagePadding
= 4;
39 const int kHeadlineRowPadding
= 10;
40 const int kMessageBubblePadding
= 11;
42 // How many extensions to show in the bubble (max).
43 const size_t kMaxExtensionsToShow
= 7;
45 // How long to wait until showing the bubble (in seconds).
46 const int kBubbleAppearanceWaitTime
= 5;
50 ////////////////////////////////////////////////////////////////////////////////
51 // ExtensionMessageBubbleView
53 namespace extensions
{
55 ExtensionMessageBubbleView::ExtensionMessageBubbleView(
56 views::View
* anchor_view
,
57 scoped_ptr
<ExtensionMessageBubbleController
> controller
)
58 : BubbleDelegateView(anchor_view
, views::BubbleBorder::TOP_RIGHT
),
60 controller_(controller
.Pass()),
63 dismiss_button_(NULL
),
65 action_taken_(false) {
66 DCHECK(anchor_view
->GetWidget());
67 set_close_on_deactivate(false);
68 set_move_with_anchor(true);
69 set_close_on_esc(true);
71 // Compensate for built-in vertical padding in the anchor view's image.
72 set_anchor_view_insets(gfx::Insets(5, 0, 5, 0));
76 void ExtensionMessageBubbleView::MaybeShow(
78 ToolbarView
* toolbar_view
,
79 views::View
* anchor_view
) {
81 // The list of suspicious extensions takes priority over the dev mode bubble,
82 // since that needs to be shown as soon as we disable something. The dev mode
83 // bubble is not as time sensitive so we'll catch the dev mode extensions on
84 // the next startup/next window that opens. That way, we're not too spammy
86 scoped_ptr
<SuspiciousExtensionBubbleController
> suspicious_extensions(
87 new SuspiciousExtensionBubbleController(browser
->profile()));
88 if (suspicious_extensions
->ShouldShow()) {
89 SuspiciousExtensionBubbleController
* controller
=
90 suspicious_extensions
.get();
91 ExtensionMessageBubbleView
* bubble_delegate
=
92 new ExtensionMessageBubbleView(anchor_view
,
93 suspicious_extensions
.Pass());
94 views::BubbleDelegateView::CreateBubble(bubble_delegate
);
95 controller
->Show(bubble_delegate
);
99 scoped_ptr
<DevModeBubbleController
> dev_mode_extensions(
100 new DevModeBubbleController(browser
->profile()));
101 if (dev_mode_extensions
->ShouldShow()) {
102 views::View
* reference_view
= NULL
;
103 BrowserActionsContainer
* container
= toolbar_view
->browser_actions();
104 if (container
->animating())
107 ExtensionService
* service
= extensions::ExtensionSystem::Get(
108 browser
->profile())->extension_service();
109 extensions::ExtensionActionManager
* extension_action_manager
=
110 extensions::ExtensionActionManager::Get(browser
->profile());
112 const ExtensionIdList extension_list
=
113 dev_mode_extensions
->GetExtensionIdList();
114 ExtensionToolbarModel::Get(
115 browser
->profile())->EnsureVisibility(extension_list
);
116 for (size_t i
= 0; i
< extension_list
.size(); ++i
) {
117 const Extension
* extension
=
118 service
->GetExtensionById(extension_list
[i
], false);
121 reference_view
= container
->GetBrowserActionView(
122 extension_action_manager
->GetBrowserAction(*extension
));
123 if (reference_view
&& reference_view
->visible())
124 break; // Found a good candidate.
126 if (reference_view
) {
127 // If we have a view, it means we found a browser action and we want to
128 // point to the chevron, not the hotdog menu.
129 if (!reference_view
->visible())
130 reference_view
= container
->chevron(); // It's hidden, use the chevron.
132 if (reference_view
&& reference_view
->visible())
133 anchor_view
= reference_view
; // Catch-all is the hotdog menu.
135 DevModeBubbleController
* controller
= dev_mode_extensions
.get();
136 ExtensionMessageBubbleView
* bubble_delegate
=
137 new ExtensionMessageBubbleView(anchor_view
, dev_mode_extensions
.Pass());
138 views::BubbleDelegateView::CreateBubble(bubble_delegate
);
139 controller
->Show(bubble_delegate
);
144 void ExtensionMessageBubbleView::OnActionButtonClicked(
145 const base::Closure
& callback
) {
146 action_callback_
= callback
;
149 void ExtensionMessageBubbleView::OnDismissButtonClicked(
150 const base::Closure
& callback
) {
151 dismiss_callback_
= callback
;
154 void ExtensionMessageBubbleView::OnLinkClicked(
155 const base::Closure
& callback
) {
156 link_callback_
= callback
;
159 void ExtensionMessageBubbleView::Show() {
160 // Not showing the bubble right away (during startup) has a few benefits:
161 // We don't have to worry about focus being lost due to the Omnibox (or to
162 // other things that want focus at startup). This allows Esc to work to close
163 // the bubble and also solves the keyboard accessibility problem that comes
164 // with focus being lost (we don't have a good generic mechanism of injecting
165 // bubbles into the focus cycle). Another benefit of delaying the show is
166 // that fade-in works (the fade-in isn't apparent if the the bubble appears at
168 base::MessageLoop::current()->PostDelayedTask(
170 base::Bind(&ExtensionMessageBubbleView::ShowBubble
,
171 weak_factory_
.GetWeakPtr()),
172 base::TimeDelta::FromSeconds(kBubbleAppearanceWaitTime
));
175 void ExtensionMessageBubbleView::OnWidgetDestroying(views::Widget
* widget
) {
176 // To catch Esc, we monitor destroy message. Unless the link has been clicked,
177 // we assume Dismiss was the action taken.
178 if (!link_clicked_
&& !action_taken_
)
179 dismiss_callback_
.Run();
182 ////////////////////////////////////////////////////////////////////////////////
183 // ExtensionMessageBubbleView - private.
185 ExtensionMessageBubbleView::~ExtensionMessageBubbleView() {
188 void ExtensionMessageBubbleView::ShowBubble() {
192 void ExtensionMessageBubbleView::Init() {
193 ui::ResourceBundle
& rb
= ui::ResourceBundle::GetSharedInstance();
195 views::GridLayout
* layout
= views::GridLayout::CreatePanel(this);
196 layout
->SetInsets(kInsetTop
, kInsetLeft
,
197 kInsetBottomRight
, kInsetBottomRight
);
198 SetLayoutManager(layout
);
200 ExtensionMessageBubbleController::Delegate
* delegate
=
201 controller_
->delegate();
203 const int headline_column_set_id
= 0;
204 views::ColumnSet
* top_columns
= layout
->AddColumnSet(headline_column_set_id
);
205 top_columns
->AddColumn(views::GridLayout::LEADING
, views::GridLayout::CENTER
,
206 0, views::GridLayout::USE_PREF
, 0, 0);
207 top_columns
->AddPaddingColumn(1, 0);
208 layout
->StartRow(0, headline_column_set_id
);
210 headline_
= new views::Label(delegate
->GetTitle(),
211 rb
.GetFontList(ui::ResourceBundle::MediumFont
));
212 layout
->AddView(headline_
);
214 layout
->AddPaddingRow(0, kHeadlineRowPadding
);
216 const int text_column_set_id
= 1;
217 views::ColumnSet
* upper_columns
= layout
->AddColumnSet(text_column_set_id
);
218 upper_columns
->AddColumn(
219 views::GridLayout::LEADING
, views::GridLayout::LEADING
,
220 0, views::GridLayout::USE_PREF
, 0, 0);
221 layout
->StartRow(0, text_column_set_id
);
223 views::Label
* message
= new views::Label();
224 message
->SetMultiLine(true);
225 message
->SetHorizontalAlignment(gfx::ALIGN_LEFT
);
226 message
->SetText(delegate
->GetMessageBody());
227 message
->SizeToFit(views::Widget::GetLocalizedContentsWidth(
228 IDS_EXTENSION_WIPEOUT_BUBBLE_WIDTH_CHARS
));
229 layout
->AddView(message
);
231 if (delegate
->ShouldShowExtensionList()) {
232 const int extension_list_column_set_id
= 2;
233 views::ColumnSet
* middle_columns
=
234 layout
->AddColumnSet(extension_list_column_set_id
);
235 middle_columns
->AddPaddingColumn(0, kExtensionListPadding
);
236 middle_columns
->AddColumn(
237 views::GridLayout::LEADING
, views::GridLayout::CENTER
,
238 0, views::GridLayout::USE_PREF
, 0, 0);
240 layout
->StartRowWithPadding(0, extension_list_column_set_id
,
241 0, kHeadlineMessagePadding
);
242 views::Label
* extensions
= new views::Label();
243 extensions
->SetMultiLine(true);
244 extensions
->SetHorizontalAlignment(gfx::ALIGN_LEFT
);
246 std::vector
<base::string16
> extension_list
;
247 base::char16 bullet_point
= 0x2022;
249 std::vector
<base::string16
> suspicious
= controller_
->GetExtensionList();
251 for (; i
< suspicious
.size() && i
< kMaxExtensionsToShow
; ++i
) {
252 // Add each extension with bullet point.
253 extension_list
.push_back(
254 bullet_point
+ base::ASCIIToUTF16(" ") + suspicious
[i
]);
257 if (i
> kMaxExtensionsToShow
) {
258 base::string16 difference
= base::IntToString16(i
- kMaxExtensionsToShow
);
259 extension_list
.push_back(bullet_point
+ base::ASCIIToUTF16(" ") +
260 delegate
->GetOverflowText(difference
));
263 extensions
->SetText(JoinString(extension_list
, base::ASCIIToUTF16("\n")));
264 extensions
->SizeToFit(views::Widget::GetLocalizedContentsWidth(
265 IDS_EXTENSION_WIPEOUT_BUBBLE_WIDTH_CHARS
));
266 layout
->AddView(extensions
);
269 base::string16 action_button
= delegate
->GetActionButtonLabel();
271 const int action_row_column_set_id
= 3;
272 views::ColumnSet
* bottom_columns
=
273 layout
->AddColumnSet(action_row_column_set_id
);
274 bottom_columns
->AddColumn(views::GridLayout::LEADING
,
275 views::GridLayout::CENTER
, 0, views::GridLayout::USE_PREF
, 0, 0);
276 bottom_columns
->AddPaddingColumn(1, 0);
277 bottom_columns
->AddColumn(views::GridLayout::TRAILING
,
278 views::GridLayout::CENTER
, 0, views::GridLayout::USE_PREF
, 0, 0);
279 if (!action_button
.empty()) {
280 bottom_columns
->AddColumn(views::GridLayout::TRAILING
,
281 views::GridLayout::CENTER
, 0, views::GridLayout::USE_PREF
, 0, 0);
283 layout
->StartRowWithPadding(0, action_row_column_set_id
,
284 0, kMessageBubblePadding
);
286 learn_more_
= new views::Link(delegate
->GetLearnMoreLabel());
287 learn_more_
->set_listener(this);
288 layout
->AddView(learn_more_
);
290 if (!action_button
.empty()) {
291 action_button_
= new views::LabelButton(this, action_button
.c_str());
292 action_button_
->SetStyle(views::Button::STYLE_BUTTON
);
293 layout
->AddView(action_button_
);
296 dismiss_button_
= new views::LabelButton(this,
297 delegate
->GetDismissButtonLabel());
298 dismiss_button_
->SetStyle(views::Button::STYLE_BUTTON
);
299 layout
->AddView(dismiss_button_
);
302 void ExtensionMessageBubbleView::ButtonPressed(views::Button
* sender
,
303 const ui::Event
& event
) {
304 if (sender
== action_button_
) {
305 action_taken_
= true;
306 action_callback_
.Run();
308 DCHECK_EQ(dismiss_button_
, sender
);
310 GetWidget()->Close();
313 void ExtensionMessageBubbleView::LinkClicked(views::Link
* source
,
315 DCHECK_EQ(learn_more_
, source
);
316 link_clicked_
= true;
317 link_callback_
.Run();
318 GetWidget()->Close();
321 void ExtensionMessageBubbleView::GetAccessibleState(
322 ui::AccessibleViewState
* state
) {
323 state
->role
= ui::AccessibilityTypes::ROLE_ALERT
;
326 void ExtensionMessageBubbleView::ViewHierarchyChanged(
327 const ViewHierarchyChangedDetails
& details
) {
328 if (details
.is_add
&& details
.child
== this)
329 NotifyAccessibilityEvent(ui::AccessibilityTypes::EVENT_ALERT
, true);
332 } // namespace extensions