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/extensions/external_install_ui.h"
6
7 #include <string>
8
9 #include "base/bind.h"
10 #include "base/lazy_instance.h"
11 #include "base/memory/ref_counted.h"
12 #include "base/memory/scoped_ptr.h"
13 #include "base/message_loop/message_loop.h"
14 #include "base/metrics/histogram.h"
15 #include "base/strings/utf_string_conversions.h"
16 #include "chrome/app/chrome_command_ids.h"
17 #include "chrome/browser/chrome_notification_types.h"
18 #include "chrome/browser/extensions/extension_install_prompt.h"
19 #include "chrome/browser/extensions/extension_install_ui.h"
20 #include "chrome/browser/extensions/extension_service.h"
21 #include "chrome/browser/extensions/extension_uninstall_dialog.h"
22 #include "chrome/browser/profiles/profile.h"
23 #include "chrome/browser/ui/browser.h"
24 #include "chrome/browser/ui/browser_finder.h"
25 #include "chrome/browser/ui/global_error/global_error.h"
26 #include "chrome/browser/ui/global_error/global_error_service.h"
27 #include "chrome/browser/ui/global_error/global_error_service_factory.h"
28 #include "chrome/browser/ui/host_desktop.h"
29 #include "chrome/common/extensions/extension_constants.h"
30 #include "chrome/common/extensions/manifest_url_handler.h"
31 #include "content/public/browser/notification_details.h"
32 #include "content/public/browser/notification_observer.h"
33 #include "content/public/browser/notification_registrar.h"
34 #include "content/public/browser/notification_source.h"
35 #include "extensions/common/extension.h"
36 #include "grit/chromium_strings.h"
37 #include "grit/generated_resources.h"
38 #include "grit/theme_resources.h"
39 #include "ui/base/l10n/l10n_util.h"
40 #include "ui/gfx/image/image.h"
41 #include "ui/gfx/image/image_skia_operations.h"
42 #include "ui/gfx/size.h"
43
44 namespace extensions {
45
46 namespace {
47
48 // Whether the external extension can use the streamlined bubble install flow.
UseBubbleInstall(const Extension * extension,bool is_new_profile)49 bool UseBubbleInstall(const Extension* extension, bool is_new_profile) {
50 return ManifestURL::UpdatesFromGallery(extension) && !is_new_profile;
51 }
52
53 } // namespace
54
55 static const int kMenuCommandId = IDC_EXTERNAL_EXTENSION_ALERT;
56
57 class ExternalInstallGlobalError;
58
59 // TODO(mpcomplete): Get rid of the refcounting on this class, or document
60 // why it's necessary. Will do after refactoring to merge back with
61 // ExtensionDisabledDialogDelegate.
62 class ExternalInstallDialogDelegate
63 : public ExtensionInstallPrompt::Delegate,
64 public base::RefCountedThreadSafe<ExternalInstallDialogDelegate> {
65 public:
66 ExternalInstallDialogDelegate(Browser* browser,
67 ExtensionService* service,
68 const Extension* extension,
69 bool use_global_error);
70
browser()71 Browser* browser() { return browser_; }
72
73 private:
74 friend class base::RefCountedThreadSafe<ExternalInstallDialogDelegate>;
75 friend class ExternalInstallGlobalError;
76
77 virtual ~ExternalInstallDialogDelegate();
78
79 // ExtensionInstallPrompt::Delegate:
80 virtual void InstallUIProceed() OVERRIDE;
81 virtual void InstallUIAbort(bool user_initiated) OVERRIDE;
82
83 // The UI for showing the install dialog when enabling.
84 scoped_ptr<ExtensionInstallPrompt> install_ui_;
85
86 Browser* browser_;
87 base::WeakPtr<ExtensionService> service_weak_;
88 const std::string extension_id_;
89 };
90
91 // Only shows a menu item, no bubble. Clicking the menu item shows
92 // an external install dialog.
93 class ExternalInstallMenuAlert : public GlobalErrorWithStandardBubble,
94 public content::NotificationObserver {
95 public:
96 ExternalInstallMenuAlert(ExtensionService* service,
97 const Extension* extension);
98 virtual ~ExternalInstallMenuAlert();
99
extension() const100 const Extension* extension() const { return extension_; }
101
102 // GlobalError implementation.
103 virtual Severity GetSeverity() OVERRIDE;
104 virtual bool HasMenuItem() OVERRIDE;
105 virtual int MenuItemCommandID() OVERRIDE;
106 virtual base::string16 MenuItemLabel() OVERRIDE;
107 virtual void ExecuteMenuItem(Browser* browser) OVERRIDE;
108 virtual bool HasBubbleView() OVERRIDE;
109 virtual base::string16 GetBubbleViewTitle() OVERRIDE;
110 virtual std::vector<base::string16> GetBubbleViewMessages() OVERRIDE;
111 virtual base::string16 GetBubbleViewAcceptButtonLabel() OVERRIDE;
112 virtual base::string16 GetBubbleViewCancelButtonLabel() OVERRIDE;
113 virtual void OnBubbleViewDidClose(Browser* browser) OVERRIDE;
114 virtual void BubbleViewAcceptButtonPressed(Browser* browser) OVERRIDE;
115 virtual void BubbleViewCancelButtonPressed(Browser* browser) OVERRIDE;
116
117 // content::NotificationObserver implementation.
118 virtual void Observe(int type,
119 const content::NotificationSource& source,
120 const content::NotificationDetails& details) OVERRIDE;
121
122 protected:
123 ExtensionService* service_;
124 const Extension* extension_;
125 content::NotificationRegistrar registrar_;
126 };
127
128 // Shows a menu item and a global error bubble, replacing the install dialog.
129 class ExternalInstallGlobalError : public ExternalInstallMenuAlert {
130 public:
131 ExternalInstallGlobalError(ExtensionService* service,
132 const Extension* extension,
133 ExternalInstallDialogDelegate* delegate,
134 const ExtensionInstallPrompt::Prompt& prompt);
135 virtual ~ExternalInstallGlobalError();
136
137 virtual void ExecuteMenuItem(Browser* browser) OVERRIDE;
138 virtual bool HasBubbleView() OVERRIDE;
139 virtual gfx::Image GetBubbleViewIcon() OVERRIDE;
140 virtual base::string16 GetBubbleViewTitle() OVERRIDE;
141 virtual std::vector<base::string16> GetBubbleViewMessages() OVERRIDE;
142 virtual base::string16 GetBubbleViewAcceptButtonLabel() OVERRIDE;
143 virtual base::string16 GetBubbleViewCancelButtonLabel() OVERRIDE;
144 virtual void OnBubbleViewDidClose(Browser* browser) OVERRIDE;
145 virtual void BubbleViewAcceptButtonPressed(Browser* browser) OVERRIDE;
146 virtual void BubbleViewCancelButtonPressed(Browser* browser) OVERRIDE;
147
148 protected:
149 // Ref-counted, but needs to be disposed of if we are dismissed without
150 // having been clicked (perhaps because the user enabled the extension
151 // manually).
152 ExternalInstallDialogDelegate* delegate_;
153 const ExtensionInstallPrompt::Prompt* prompt_;
154 };
155
CreateExternalInstallGlobalError(base::WeakPtr<ExtensionService> service,const std::string & extension_id,const ExtensionInstallPrompt::ShowParams & show_params,ExtensionInstallPrompt::Delegate * prompt_delegate,const ExtensionInstallPrompt::Prompt & prompt)156 static void CreateExternalInstallGlobalError(
157 base::WeakPtr<ExtensionService> service,
158 const std::string& extension_id,
159 const ExtensionInstallPrompt::ShowParams& show_params,
160 ExtensionInstallPrompt::Delegate* prompt_delegate,
161 const ExtensionInstallPrompt::Prompt& prompt) {
162 if (!service.get())
163 return;
164 const Extension* extension = service->GetInstalledExtension(extension_id);
165 if (!extension)
166 return;
167 GlobalErrorService* error_service =
168 GlobalErrorServiceFactory::GetForProfile(service->profile());
169 if (error_service->GetGlobalErrorByMenuItemCommandID(kMenuCommandId))
170 return;
171
172 ExternalInstallDialogDelegate* delegate =
173 static_cast<ExternalInstallDialogDelegate*>(prompt_delegate);
174 ExternalInstallGlobalError* error_bubble = new ExternalInstallGlobalError(
175 service.get(), extension, delegate, prompt);
176 error_service->AddGlobalError(error_bubble);
177 // Show bubble immediately if possible.
178 if (delegate->browser())
179 error_bubble->ShowBubbleView(delegate->browser());
180 }
181
ShowExternalInstallDialog(ExtensionService * service,Browser * browser,const Extension * extension)182 static void ShowExternalInstallDialog(
183 ExtensionService* service,
184 Browser* browser,
185 const Extension* extension) {
186 // This object manages its own lifetime.
187 new ExternalInstallDialogDelegate(browser, service, extension, false);
188 }
189
190 // ExternalInstallDialogDelegate --------------------------------------------
191
ExternalInstallDialogDelegate(Browser * browser,ExtensionService * service,const Extension * extension,bool use_global_error)192 ExternalInstallDialogDelegate::ExternalInstallDialogDelegate(
193 Browser* browser,
194 ExtensionService* service,
195 const Extension* extension,
196 bool use_global_error)
197 : browser_(browser),
198 service_weak_(service->AsWeakPtr()),
199 extension_id_(extension->id()) {
200 AddRef(); // Balanced in Proceed or Abort.
201
202 install_ui_.reset(
203 ExtensionInstallUI::CreateInstallPromptWithBrowser(browser));
204
205 const ExtensionInstallPrompt::ShowDialogCallback callback =
206 use_global_error ?
207 base::Bind(&CreateExternalInstallGlobalError,
208 service_weak_, extension_id_) :
209 ExtensionInstallPrompt::GetDefaultShowDialogCallback();
210 install_ui_->ConfirmExternalInstall(this, extension, callback);
211 }
212
~ExternalInstallDialogDelegate()213 ExternalInstallDialogDelegate::~ExternalInstallDialogDelegate() {
214 }
215
InstallUIProceed()216 void ExternalInstallDialogDelegate::InstallUIProceed() {
217 if (!service_weak_.get())
218 return;
219 const Extension* extension =
220 service_weak_->GetInstalledExtension(extension_id_);
221 if (!extension)
222 return;
223 service_weak_->GrantPermissionsAndEnableExtension(extension);
224 Release();
225 }
226
InstallUIAbort(bool user_initiated)227 void ExternalInstallDialogDelegate::InstallUIAbort(bool user_initiated) {
228 if (!service_weak_.get())
229 return;
230 const Extension* extension =
231 service_weak_->GetInstalledExtension(extension_id_);
232 if (!extension)
233 return;
234 service_weak_->UninstallExtension(extension_id_, false, NULL);
235 Release();
236 }
237
238 // ExternalInstallMenuAlert -------------------------------------------------
239
ExternalInstallMenuAlert(ExtensionService * service,const Extension * extension)240 ExternalInstallMenuAlert::ExternalInstallMenuAlert(
241 ExtensionService* service,
242 const Extension* extension)
243 : service_(service),
244 extension_(extension) {
245 registrar_.Add(this, chrome::NOTIFICATION_EXTENSION_LOADED,
246 content::Source<Profile>(service->profile()));
247 registrar_.Add(this, chrome::NOTIFICATION_EXTENSION_REMOVED,
248 content::Source<Profile>(service->profile()));
249 }
250
~ExternalInstallMenuAlert()251 ExternalInstallMenuAlert::~ExternalInstallMenuAlert() {
252 }
253
GetSeverity()254 GlobalError::Severity ExternalInstallMenuAlert::GetSeverity() {
255 return SEVERITY_LOW;
256 }
257
HasMenuItem()258 bool ExternalInstallMenuAlert::HasMenuItem() {
259 return true;
260 }
261
MenuItemCommandID()262 int ExternalInstallMenuAlert::MenuItemCommandID() {
263 return kMenuCommandId;
264 }
265
MenuItemLabel()266 base::string16 ExternalInstallMenuAlert::MenuItemLabel() {
267 int id = -1;
268 if (extension_->is_app())
269 id = IDS_EXTENSION_EXTERNAL_INSTALL_ALERT_APP;
270 else if (extension_->is_theme())
271 id = IDS_EXTENSION_EXTERNAL_INSTALL_ALERT_THEME;
272 else
273 id = IDS_EXTENSION_EXTERNAL_INSTALL_ALERT_EXTENSION;
274 return l10n_util::GetStringFUTF16(id, UTF8ToUTF16(extension_->name()));
275 }
276
ExecuteMenuItem(Browser * browser)277 void ExternalInstallMenuAlert::ExecuteMenuItem(Browser* browser) {
278 ShowExternalInstallDialog(service_, browser, extension_);
279 }
280
HasBubbleView()281 bool ExternalInstallMenuAlert::HasBubbleView() {
282 return false;
283 }
GetBubbleViewTitle()284 base::string16 ExternalInstallMenuAlert::GetBubbleViewTitle() {
285 return base::string16();
286 }
287
GetBubbleViewMessages()288 std::vector<base::string16> ExternalInstallMenuAlert::GetBubbleViewMessages() {
289 return std::vector<base::string16>();
290 }
291
GetBubbleViewAcceptButtonLabel()292 base::string16 ExternalInstallMenuAlert::GetBubbleViewAcceptButtonLabel() {
293 return base::string16();
294 }
295
GetBubbleViewCancelButtonLabel()296 base::string16 ExternalInstallMenuAlert::GetBubbleViewCancelButtonLabel() {
297 return base::string16();
298 }
299
OnBubbleViewDidClose(Browser * browser)300 void ExternalInstallMenuAlert::OnBubbleViewDidClose(Browser* browser) {
301 NOTREACHED();
302 }
303
BubbleViewAcceptButtonPressed(Browser * browser)304 void ExternalInstallMenuAlert::BubbleViewAcceptButtonPressed(
305 Browser* browser) {
306 NOTREACHED();
307 }
308
BubbleViewCancelButtonPressed(Browser * browser)309 void ExternalInstallMenuAlert::BubbleViewCancelButtonPressed(
310 Browser* browser) {
311 NOTREACHED();
312 }
313
Observe(int type,const content::NotificationSource & source,const content::NotificationDetails & details)314 void ExternalInstallMenuAlert::Observe(
315 int type,
316 const content::NotificationSource& source,
317 const content::NotificationDetails& details) {
318 // The error is invalidated if the extension has been loaded or removed.
319 DCHECK(type == chrome::NOTIFICATION_EXTENSION_LOADED ||
320 type == chrome::NOTIFICATION_EXTENSION_REMOVED);
321 const Extension* extension = content::Details<const Extension>(details).ptr();
322 if (extension != extension_)
323 return;
324 GlobalErrorService* error_service =
325 GlobalErrorServiceFactory::GetForProfile(service_->profile());
326 error_service->RemoveGlobalError(this);
327 service_->AcknowledgeExternalExtension(extension_->id());
328 delete this;
329 }
330
331 // ExternalInstallGlobalError -----------------------------------------------
332
ExternalInstallGlobalError(ExtensionService * service,const Extension * extension,ExternalInstallDialogDelegate * delegate,const ExtensionInstallPrompt::Prompt & prompt)333 ExternalInstallGlobalError::ExternalInstallGlobalError(
334 ExtensionService* service,
335 const Extension* extension,
336 ExternalInstallDialogDelegate* delegate,
337 const ExtensionInstallPrompt::Prompt& prompt)
338 : ExternalInstallMenuAlert(service, extension),
339 delegate_(delegate),
340 prompt_(&prompt) {
341 }
342
~ExternalInstallGlobalError()343 ExternalInstallGlobalError::~ExternalInstallGlobalError() {
344 if (delegate_)
345 delegate_->Release();
346 }
347
ExecuteMenuItem(Browser * browser)348 void ExternalInstallGlobalError::ExecuteMenuItem(Browser* browser) {
349 ShowBubbleView(browser);
350 }
351
HasBubbleView()352 bool ExternalInstallGlobalError::HasBubbleView() {
353 return true;
354 }
355
GetBubbleViewIcon()356 gfx::Image ExternalInstallGlobalError::GetBubbleViewIcon() {
357 if (prompt_->icon().IsEmpty())
358 return GlobalErrorWithStandardBubble::GetBubbleViewIcon();
359 // Scale icon to a reasonable size.
360 return gfx::Image(gfx::ImageSkiaOperations::CreateResizedImage(
361 *prompt_->icon().ToImageSkia(),
362 skia::ImageOperations::RESIZE_BEST,
363 gfx::Size(extension_misc::EXTENSION_ICON_SMALL,
364 extension_misc::EXTENSION_ICON_SMALL)));
365 }
366
GetBubbleViewTitle()367 base::string16 ExternalInstallGlobalError::GetBubbleViewTitle() {
368 return prompt_->GetDialogTitle();
369 }
370
371 std::vector<base::string16>
GetBubbleViewMessages()372 ExternalInstallGlobalError::GetBubbleViewMessages() {
373 std::vector<base::string16> messages;
374 messages.push_back(prompt_->GetHeading());
375 if (prompt_->GetPermissionCount()) {
376 messages.push_back(prompt_->GetPermissionsHeading());
377 for (size_t i = 0; i < prompt_->GetPermissionCount(); ++i) {
378 messages.push_back(l10n_util::GetStringFUTF16(
379 IDS_EXTENSION_PERMISSION_LINE,
380 prompt_->GetPermission(i)));
381 }
382 }
383 // TODO(yoz): OAuth issue advice?
384 return messages;
385 }
386
GetBubbleViewAcceptButtonLabel()387 base::string16 ExternalInstallGlobalError::GetBubbleViewAcceptButtonLabel() {
388 return prompt_->GetAcceptButtonLabel();
389 }
390
GetBubbleViewCancelButtonLabel()391 base::string16 ExternalInstallGlobalError::GetBubbleViewCancelButtonLabel() {
392 return prompt_->GetAbortButtonLabel();
393 }
394
OnBubbleViewDidClose(Browser * browser)395 void ExternalInstallGlobalError::OnBubbleViewDidClose(Browser* browser) {
396 }
397
BubbleViewAcceptButtonPressed(Browser * browser)398 void ExternalInstallGlobalError::BubbleViewAcceptButtonPressed(
399 Browser* browser) {
400 ExternalInstallDialogDelegate* delegate = delegate_;
401 delegate_ = NULL;
402 delegate->InstallUIProceed();
403 }
404
BubbleViewCancelButtonPressed(Browser * browser)405 void ExternalInstallGlobalError::BubbleViewCancelButtonPressed(
406 Browser* browser) {
407 ExternalInstallDialogDelegate* delegate = delegate_;
408 delegate_ = NULL;
409 delegate->InstallUIAbort(true);
410 }
411
412 // Public interface ---------------------------------------------------------
413
AddExternalInstallError(ExtensionService * service,const Extension * extension,bool is_new_profile)414 void AddExternalInstallError(ExtensionService* service,
415 const Extension* extension,
416 bool is_new_profile) {
417 GlobalErrorService* error_service =
418 GlobalErrorServiceFactory::GetForProfile(service->profile());
419 if (error_service->GetGlobalErrorByMenuItemCommandID(kMenuCommandId))
420 return;
421
422 if (UseBubbleInstall(extension, is_new_profile)) {
423 Browser* browser = NULL;
424 #if !defined(OS_ANDROID)
425 browser = chrome::FindTabbedBrowser(service->profile(),
426 true,
427 chrome::GetActiveDesktop());
428 #endif
429 new ExternalInstallDialogDelegate(browser, service, extension, true);
430 } else {
431 error_service->AddGlobalError(
432 new ExternalInstallMenuAlert(service, extension));
433 }
434 }
435
RemoveExternalInstallError(ExtensionService * service)436 void RemoveExternalInstallError(ExtensionService* service) {
437 GlobalErrorService* error_service =
438 GlobalErrorServiceFactory::GetForProfile(service->profile());
439 GlobalError* error = error_service->GetGlobalErrorByMenuItemCommandID(
440 kMenuCommandId);
441 if (error) {
442 error_service->RemoveGlobalError(error);
443 delete error;
444 }
445 }
446
HasExternalInstallError(ExtensionService * service)447 bool HasExternalInstallError(ExtensionService* service) {
448 GlobalErrorService* error_service =
449 GlobalErrorServiceFactory::GetForProfile(service->profile());
450 GlobalError* error = error_service->GetGlobalErrorByMenuItemCommandID(
451 kMenuCommandId);
452 return !!error;
453 }
454
HasExternalInstallBubble(ExtensionService * service)455 bool HasExternalInstallBubble(ExtensionService* service) {
456 GlobalErrorService* error_service =
457 GlobalErrorServiceFactory::GetForProfile(service->profile());
458 GlobalError* error = error_service->GetGlobalErrorByMenuItemCommandID(
459 kMenuCommandId);
460 return error && error->HasBubbleView();
461 }
462
463 } // namespace extensions
464