1 // Copyright 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/extensions/extension_installed_bubble.h"
10 #include "base/location.h"
11 #include "base/single_thread_task_runner.h"
12 #include "base/strings/utf_string_conversions.h"
13 #include "base/thread_task_runner_handle.h"
14 #include "base/time/time.h"
15 #include "chrome/browser/chrome_notification_types.h"
16 #include "chrome/browser/extensions/api/commands/command_service.h"
17 #include "chrome/browser/profiles/profile.h"
18 #include "chrome/browser/ui/browser.h"
19 #include "chrome/common/extensions/api/extension_action/action_info.h"
20 #include "chrome/common/extensions/api/omnibox/omnibox_handler.h"
21 #include "chrome/common/extensions/command.h"
22 #include "chrome/grit/generated_resources.h"
23 #include "content/public/browser/notification_details.h"
24 #include "content/public/browser/notification_source.h"
25 #include "extensions/browser/extension_registry.h"
26 #include "extensions/common/extension.h"
27 #include "ui/base/l10n/l10n_util.h"
29 using extensions::Extension
;
33 // How long to wait for browser action animations to complete before retrying.
34 const int kAnimationWaitMs
= 50;
35 // How often we retry when waiting for browser action animation to end.
36 const int kAnimationWaitRetries
= 10;
38 // Returns the keybinding for an extension command, or a null if none exists.
39 scoped_ptr
<extensions::Command
> GetCommand(
40 const std::string
& extension_id
,
42 ExtensionInstalledBubble::BubbleType type
) {
43 scoped_ptr
<extensions::Command
> result
;
44 extensions::Command command
;
45 extensions::CommandService
* command_service
=
46 extensions::CommandService::Get(profile
);
47 bool has_command
= false;
48 if (type
== ExtensionInstalledBubble::BROWSER_ACTION
) {
49 has_command
= command_service
->GetBrowserActionCommand(
50 extension_id
, extensions::CommandService::ACTIVE
, &command
, nullptr);
51 } else if (type
== ExtensionInstalledBubble::PAGE_ACTION
) {
52 has_command
= command_service
->GetPageActionCommand(
53 extension_id
, extensions::CommandService::ACTIVE
, &command
, nullptr);
56 result
.reset(new extensions::Command(command
));
62 ExtensionInstalledBubble::ExtensionInstalledBubble(Delegate
* delegate
,
63 const Extension
* extension
,
66 : delegate_(delegate
),
67 extension_(extension
),
70 extension_registry_observer_(this),
71 animation_wait_retries_(0),
73 if (!extensions::OmniboxInfo::GetKeyword(extension
).empty())
74 type_
= OMNIBOX_KEYWORD
;
75 else if (extensions::ActionInfo::GetBrowserActionInfo(extension
))
76 type_
= BROWSER_ACTION
;
77 else if (extensions::ActionInfo::GetPageActionInfo(extension
) &&
78 extensions::ActionInfo::IsVerboseInstallMessage(extension
))
83 // |extension| has been initialized but not loaded at this point. We need
84 // to wait on showing the Bubble until not only the EXTENSION_LOADED gets
85 // fired, but all of the EXTENSION_LOADED Observers have run. Only then can we
86 // be sure that a BrowserAction or PageAction has had views created which we
87 // can inspect for the purpose of previewing of pointing to them.
88 extension_registry_observer_
.Add(
89 extensions::ExtensionRegistry::Get(browser
->profile()));
91 registrar_
.Add(this, chrome::NOTIFICATION_BROWSER_CLOSING
,
92 content::Source
<Browser
>(browser
));
95 ExtensionInstalledBubble::~ExtensionInstalledBubble() {}
97 void ExtensionInstalledBubble::IgnoreBrowserClosing() {
98 registrar_
.Remove(this, chrome::NOTIFICATION_BROWSER_CLOSING
,
99 content::Source
<Browser
>(browser_
));
102 base::string16
ExtensionInstalledBubble::GetHowToUseDescription() const {
104 base::string16 extra
;
106 extra
= action_command_
->accelerator().GetShortcutText();
110 message_id
= extra
.empty() ? IDS_EXTENSION_INSTALLED_BROWSER_ACTION_INFO
:
111 IDS_EXTENSION_INSTALLED_BROWSER_ACTION_INFO_WITH_SHORTCUT
;
114 message_id
= extra
.empty() ? IDS_EXTENSION_INSTALLED_PAGE_ACTION_INFO
:
115 IDS_EXTENSION_INSTALLED_PAGE_ACTION_INFO_WITH_SHORTCUT
;
117 case OMNIBOX_KEYWORD
:
119 base::UTF8ToUTF16(extensions::OmniboxInfo::GetKeyword(extension_
));
120 message_id
= IDS_EXTENSION_INSTALLED_OMNIBOX_KEYWORD_INFO
;
127 return base::string16();
128 return extra
.empty() ? l10n_util::GetStringUTF16(message_id
) :
129 l10n_util::GetStringFUTF16(message_id
, extra
);
132 void ExtensionInstalledBubble::ShowInternal() {
133 if (delegate_
->MaybeShowNow())
135 if (animation_wait_retries_
++ < kAnimationWaitRetries
) {
136 base::ThreadTaskRunnerHandle::Get()->PostDelayedTask(
137 FROM_HERE
, base::Bind(&ExtensionInstalledBubble::ShowInternal
,
138 weak_factory_
.GetWeakPtr()),
139 base::TimeDelta::FromMilliseconds(kAnimationWaitMs
));
143 void ExtensionInstalledBubble::OnExtensionLoaded(
144 content::BrowserContext
* browser_context
,
145 const extensions::Extension
* extension
) {
146 if (extension
== extension_
) {
147 // Parse the extension command, if one exists.
148 action_command_
= GetCommand(extension_
->id(), browser_
->profile(), type_
);
150 animation_wait_retries_
= 0;
151 // PostTask to ourself to allow all EXTENSION_LOADED Observers to run.
152 base::ThreadTaskRunnerHandle::Get()->PostTask(
153 FROM_HERE
, base::Bind(&ExtensionInstalledBubble::ShowInternal
,
154 weak_factory_
.GetWeakPtr()));
158 void ExtensionInstalledBubble::OnExtensionUnloaded(
159 content::BrowserContext
* browser_context
,
160 const extensions::Extension
* extension
,
161 extensions::UnloadedExtensionInfo::Reason reason
) {
162 if (extension
== extension_
) {
163 // Extension is going away, make sure ShowInternal won't be called.
164 weak_factory_
.InvalidateWeakPtrs();
169 void ExtensionInstalledBubble::Observe(
171 const content::NotificationSource
& source
,
172 const content::NotificationDetails
& details
) {
173 DCHECK_EQ(type
, chrome::NOTIFICATION_BROWSER_CLOSING
)
174 << "Received unexpected notification";