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/autocomplete/extension_app_provider.h" 6 7 #include <algorithm> 8 #include <cmath> 9 10 #include "base/strings/string16.h" 11 #include "base/strings/utf_string_conversions.h" 12 #include "chrome/browser/chrome_notification_types.h" 13 #include "chrome/browser/extensions/extension_service.h" 14 #include "chrome/browser/extensions/extension_system_factory.h" 15 #include "chrome/browser/extensions/extension_util.h" 16 #include "chrome/browser/history/history_service.h" 17 #include "chrome/browser/history/history_service_factory.h" 18 #include "chrome/browser/history/url_database.h" 19 #include "chrome/browser/profiles/profile.h" 20 #include "chrome/browser/ui/extensions/application_launch.h" 21 #include "chrome/browser/ui/webui/ntp/core_app_launcher_handler.h" 22 #include "chrome/common/extensions/manifest_handlers/app_launch_info.h" 23 #include "content/public/browser/notification_source.h" 24 #include "extensions/common/extension.h" 25 #include "ui/base/l10n/l10n_util.h" 26 27 ExtensionAppProvider::ExtensionAppProvider( 28 AutocompleteProviderListener* listener, 29 Profile* profile) 30 : AutocompleteProvider(listener, profile, 31 AutocompleteProvider::TYPE_EXTENSION_APP) { 32 // Notifications of extensions loading and unloading always come from the 33 // non-incognito profile, but we need to see them regardless, as the incognito 34 // windows can be affected. 35 registrar_.Add(this, chrome::NOTIFICATION_EXTENSION_LOADED, 36 content::Source<Profile>(profile_->GetOriginalProfile())); 37 registrar_.Add(this, chrome::NOTIFICATION_EXTENSION_UNINSTALLED, 38 content::Source<Profile>(profile_->GetOriginalProfile())); 39 RefreshAppList(); 40 } 41 42 // static. 43 void ExtensionAppProvider::LaunchAppFromOmnibox( 44 const AutocompleteMatch& match, 45 Profile* profile, 46 WindowOpenDisposition disposition) { 47 ExtensionService* service = 48 extensions::ExtensionSystemFactory::GetForProfile(profile)-> 49 extension_service(); 50 const extensions::Extension* extension = 51 service->GetInstalledApp(match.destination_url); 52 // While the Omnibox popup is open, the extension can be updated, changing 53 // its URL and leaving us with no extension being found. In this case, we 54 // ignore the request. 55 if (!extension) 56 return; 57 58 CoreAppLauncherHandler::RecordAppLaunchType( 59 extension_misc::APP_LAUNCH_OMNIBOX_APP, 60 extension->GetType()); 61 62 OpenApplication(AppLaunchParams(profile, extension, disposition)); 63 } 64 65 void ExtensionAppProvider::AddExtensionAppForTesting( 66 const ExtensionApp& extension_app) { 67 extension_apps_.push_back(extension_app); 68 } 69 70 AutocompleteMatch ExtensionAppProvider::CreateAutocompleteMatch( 71 const AutocompleteInput& input, 72 const ExtensionApp& app, 73 size_t name_match_index, 74 size_t url_match_index) { 75 // TODO(finnur): Figure out what type to return here, might want to have 76 // the extension icon/a generic icon show up in the Omnibox. 77 AutocompleteMatch match(this, 0, false, 78 AutocompleteMatchType::EXTENSION_APP); 79 match.fill_into_edit = 80 app.should_match_against_launch_url ? app.launch_url : input.text(); 81 match.destination_url = GURL(app.launch_url); 82 match.allowed_to_be_default_match = true; 83 match.contents = AutocompleteMatch::SanitizeString(app.name); 84 AutocompleteMatch::ClassifyLocationInString(name_match_index, 85 input.text().length(), app.name.length(), ACMatchClassification::NONE, 86 &match.contents_class); 87 if (app.should_match_against_launch_url) { 88 match.description = app.launch_url; 89 AutocompleteMatch::ClassifyLocationInString(url_match_index, 90 input.text().length(), app.launch_url.length(), 91 ACMatchClassification::URL, &match.description_class); 92 } 93 match.relevance = CalculateRelevance( 94 input.type(), 95 input.text().length(), 96 name_match_index != base::string16::npos ? 97 app.name.length() : app.launch_url.length(), 98 match.destination_url); 99 return match; 100 } 101 102 void ExtensionAppProvider::Start(const AutocompleteInput& input, 103 bool minimal_changes) { 104 matches_.clear(); 105 106 if ((input.type() == AutocompleteInput::INVALID) || 107 (input.type() == AutocompleteInput::FORCED_QUERY)) 108 return; 109 110 if (input.text().empty()) 111 return; 112 113 for (ExtensionApps::const_iterator app = extension_apps_.begin(); 114 app != extension_apps_.end(); ++app) { 115 // See if the input matches this extension application. 116 const base::string16& name = app->name; 117 base::string16::const_iterator name_iter = 118 std::search(name.begin(), name.end(), 119 input.text().begin(), input.text().end(), 120 base::CaseInsensitiveCompare<char16>()); 121 bool matches_name = name_iter != name.end(); 122 size_t name_match_index = matches_name ? 123 static_cast<size_t>(name_iter - name.begin()) : base::string16::npos; 124 125 bool matches_url = false; 126 size_t url_match_index = base::string16::npos; 127 if (app->should_match_against_launch_url) { 128 const base::string16& url = app->launch_url; 129 base::string16::const_iterator url_iter = 130 std::search(url.begin(), url.end(), 131 input.text().begin(), input.text().end(), 132 base::CaseInsensitiveCompare<char16>()); 133 matches_url = url_iter != url.end() && 134 input.type() != AutocompleteInput::FORCED_QUERY; 135 url_match_index = matches_url ? 136 static_cast<size_t>(url_iter - url.begin()) : base::string16::npos; 137 } 138 139 if (matches_name || matches_url) { 140 // We have a match, might be a partial match. 141 matches_.push_back(CreateAutocompleteMatch( 142 input, *app, name_match_index, url_match_index)); 143 } 144 } 145 } 146 147 ExtensionAppProvider::~ExtensionAppProvider() { 148 } 149 150 void ExtensionAppProvider::RefreshAppList() { 151 ExtensionService* extension_service = 152 extensions::ExtensionSystemFactory::GetForProfile(profile_)-> 153 extension_service(); 154 if (!extension_service) 155 return; // During testing, there is no extension service. 156 const ExtensionSet* extensions = extension_service->extensions(); 157 extension_apps_.clear(); 158 for (ExtensionSet::const_iterator iter = extensions->begin(); 159 iter != extensions->end(); ++iter) { 160 const extensions::Extension* app = iter->get(); 161 if (!app->ShouldDisplayInAppLauncher()) 162 continue; 163 // Note: Apps that appear in the NTP only are not added here since this 164 // provider is currently only used in the app launcher. 165 166 if (profile_->IsOffTheRecord() && 167 !extension_util::CanLoadInIncognito(app, extension_service)) 168 continue; 169 170 GURL launch_url = app->is_platform_app() ? 171 app->url() : extensions::AppLaunchInfo::GetFullLaunchURL(app); 172 DCHECK(launch_url.is_valid()); 173 174 ExtensionApp extension_app = { 175 UTF8ToUTF16(app->name()), 176 UTF8ToUTF16(launch_url.spec()), 177 // Only hosted apps have recognizable URLs that users might type in, 178 // packaged apps and hosted apps use chrome-extension:// URLs that are 179 // normally not shown to users. 180 app->is_hosted_app() 181 }; 182 extension_apps_.push_back(extension_app); 183 } 184 } 185 186 void ExtensionAppProvider::Observe(int type, 187 const content::NotificationSource& source, 188 const content::NotificationDetails& details) { 189 RefreshAppList(); 190 } 191 192 int ExtensionAppProvider::CalculateRelevance(AutocompleteInput::Type type, 193 int input_length, 194 int target_length, 195 const GURL& url) { 196 // If you update the algorithm here, please remember to update the tables in 197 // autocomplete.h also. 198 const int kMaxRelevance = 1425; 199 200 if (input_length == target_length) 201 return kMaxRelevance; 202 203 // We give a boost proportionally based on how much of the input matches the 204 // app name, up to a maximum close to 200 (we can be close to, but we'll never 205 // reach 200 because the 100% match is taken care of above). 206 double fraction_boost = static_cast<double>(200) * 207 input_length / target_length; 208 209 // We also give a boost relative to how often the user has previously typed 210 // the Extension App URL/selected the Extension App suggestion from this 211 // provider (boost is between 200-400). 212 double type_count_boost = 0; 213 HistoryService* const history_service = 214 HistoryServiceFactory::GetForProfile(profile_, Profile::EXPLICIT_ACCESS); 215 history::URLDatabase* url_db = history_service ? 216 history_service->InMemoryDatabase() : NULL; 217 if (url_db) { 218 history::URLRow info; 219 url_db->GetRowForURL(url, &info); 220 type_count_boost = 221 400 * (1.0 - (std::pow(static_cast<double>(2), -info.typed_count()))); 222 } 223 int relevance = 575 + static_cast<int>(type_count_boost) + 224 static_cast<int>(fraction_boost); 225 DCHECK_LE(relevance, kMaxRelevance); 226 return relevance; 227 } 228