Upstream version 5.34.104.0
[platform/framework/web/crosswalk.git] / src / chrome / browser / ui / views / extensions / extension_message_bubble_view.cc
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.
4
5 #include "chrome/browser/ui/views/extensions/extension_message_bubble_view.h"
6
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_service.h"
13 #include "chrome/browser/extensions/suspicious_extension_bubble_controller.h"
14 #include "chrome/browser/profiles/profile.h"
15 #include "chrome/browser/ui/browser.h"
16 #include "chrome/browser/ui/views/frame/browser_view.h"
17 #include "chrome/browser/ui/views/toolbar/browser_actions_container.h"
18 #include "chrome/browser/ui/views/toolbar/toolbar_view.h"
19 #include "extensions/browser/extension_prefs.h"
20 #include "extensions/browser/extension_system.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"
30
31 namespace {
32
33 // Layout constants.
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;
41
42 // How many extensions to show in the bubble (max).
43 const size_t kMaxExtensionsToShow = 7;
44
45 // How long to wait until showing the bubble (in seconds).
46 const int kBubbleAppearanceWaitTime = 5;
47
48 }  // namespace
49
50 ////////////////////////////////////////////////////////////////////////////////
51 // ExtensionMessageBubbleView
52
53 namespace extensions {
54
55 ExtensionMessageBubbleView::ExtensionMessageBubbleView(
56     views::View* anchor_view,
57     scoped_ptr<ExtensionMessageBubbleController> controller)
58     : BubbleDelegateView(anchor_view, views::BubbleBorder::TOP_RIGHT),
59       weak_factory_(this),
60       controller_(controller.Pass()),
61       headline_(NULL),
62       learn_more_(NULL),
63       dismiss_button_(NULL),
64       link_clicked_(false),
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);
70
71   // Compensate for built-in vertical padding in the anchor view's image.
72   set_anchor_view_insets(gfx::Insets(5, 0, 5, 0));
73 }
74
75 // static
76 void ExtensionMessageBubbleView::MaybeShow(
77     Browser* browser,
78     ToolbarView* toolbar_view,
79     views::View* anchor_view) {
80 #if defined(OS_WIN)
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
85   // with the bubbles.
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);
96     return;
97   }
98
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())
105       return;
106
107     ExtensionService* service = extensions::ExtensionSystem::Get(
108         browser->profile())->extension_service();
109     extensions::ExtensionActionManager* extension_action_manager =
110         extensions::ExtensionActionManager::Get(browser->profile());
111
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);
119       if (!extension)
120         continue;
121       reference_view = container->GetBrowserActionView(
122           extension_action_manager->GetBrowserAction(*extension));
123       if (reference_view && reference_view->visible())
124         break;  // Found a good candidate.
125     }
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.
131     }
132     if (reference_view && reference_view->visible())
133       anchor_view = reference_view;  // Catch-all is the hotdog menu.
134
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);
140   }
141 #endif
142 }
143
144 void ExtensionMessageBubbleView::OnActionButtonClicked(
145     const base::Closure& callback) {
146   action_callback_ = callback;
147 }
148
149 void ExtensionMessageBubbleView::OnDismissButtonClicked(
150     const base::Closure& callback) {
151   dismiss_callback_ = callback;
152 }
153
154 void ExtensionMessageBubbleView::OnLinkClicked(
155     const base::Closure& callback) {
156   link_callback_ = callback;
157 }
158
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
167   // startup).
168   base::MessageLoop::current()->PostDelayedTask(
169       FROM_HERE,
170       base::Bind(&ExtensionMessageBubbleView::ShowBubble,
171                  weak_factory_.GetWeakPtr()),
172       base::TimeDelta::FromSeconds(kBubbleAppearanceWaitTime));
173 }
174
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();
180 }
181
182 ////////////////////////////////////////////////////////////////////////////////
183 // ExtensionMessageBubbleView - private.
184
185 ExtensionMessageBubbleView::~ExtensionMessageBubbleView() {
186 }
187
188 void ExtensionMessageBubbleView::ShowBubble() {
189   StartFade(true);
190 }
191
192 void ExtensionMessageBubbleView::Init() {
193   ui::ResourceBundle& rb = ui::ResourceBundle::GetSharedInstance();
194
195   views::GridLayout* layout = views::GridLayout::CreatePanel(this);
196   layout->SetInsets(kInsetTop, kInsetLeft,
197                     kInsetBottomRight, kInsetBottomRight);
198   SetLayoutManager(layout);
199
200   ExtensionMessageBubbleController::Delegate* delegate =
201       controller_->delegate();
202
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);
209
210   headline_ = new views::Label(delegate->GetTitle(),
211                                rb.GetFontList(ui::ResourceBundle::MediumFont));
212   layout->AddView(headline_);
213
214   layout->AddPaddingRow(0, kHeadlineRowPadding);
215
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);
222
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);
230
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);
239
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);
245
246     std::vector<base::string16> extension_list;
247     base::char16 bullet_point = 0x2022;
248
249     std::vector<base::string16> suspicious = controller_->GetExtensionList();
250     size_t i = 0;
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]);
255     }
256
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));
261     }
262
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);
267   }
268
269   base::string16 action_button = delegate->GetActionButtonLabel();
270
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);
282   }
283   layout->StartRowWithPadding(0, action_row_column_set_id,
284                               0, kMessageBubblePadding);
285
286   learn_more_ = new views::Link(delegate->GetLearnMoreLabel());
287   learn_more_->set_listener(this);
288   layout->AddView(learn_more_);
289
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_);
294   }
295
296   dismiss_button_ = new views::LabelButton(this,
297       delegate->GetDismissButtonLabel());
298   dismiss_button_->SetStyle(views::Button::STYLE_BUTTON);
299   layout->AddView(dismiss_button_);
300 }
301
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();
307   } else {
308     DCHECK_EQ(dismiss_button_, sender);
309   }
310   GetWidget()->Close();
311 }
312
313 void ExtensionMessageBubbleView::LinkClicked(views::Link* source,
314                                              int event_flags) {
315   DCHECK_EQ(learn_more_, source);
316   link_clicked_ = true;
317   link_callback_.Run();
318   GetWidget()->Close();
319 }
320
321 void ExtensionMessageBubbleView::GetAccessibleState(
322     ui::AccessibleViewState* state) {
323   state->role = ui::AccessibilityTypes::ROLE_ALERT;
324 }
325
326 void ExtensionMessageBubbleView::ViewHierarchyChanged(
327     const ViewHierarchyChangedDetails& details) {
328   if (details.is_add && details.child == this)
329     NotifyAccessibilityEvent(ui::AccessibilityTypes::EVENT_ALERT, true);
330 }
331
332 }  // namespace extensions