Home | History | Annotate | Download | only in autocomplete
      1 // Copyright (c) 2011 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/string16.h"
     11 #include "base/utf_string_conversions.h"
     12 #include "chrome/browser/extensions/extension_service.h"
     13 #include "chrome/browser/history/history.h"
     14 #include "chrome/browser/history/url_database.h"
     15 #include "chrome/browser/profiles/profile.h"
     16 #include "content/common/notification_service.h"
     17 #include "ui/base/l10n/l10n_util.h"
     18 
     19 ExtensionAppProvider::ExtensionAppProvider(ACProviderListener* listener,
     20                                            Profile* profile)
     21     : AutocompleteProvider(listener, profile, "ExtensionApps") {
     22   RegisterForNotifications();
     23   RefreshAppList();
     24 }
     25 
     26 void ExtensionAppProvider::AddExtensionAppForTesting(
     27     const std::string& app_name,
     28     const std::string url) {
     29   extension_apps_.push_back(std::make_pair(app_name, url));
     30 }
     31 
     32 void ExtensionAppProvider::Start(const AutocompleteInput& input,
     33                                  bool minimal_changes) {
     34   matches_.clear();
     35 
     36   if (input.type() == AutocompleteInput::INVALID)
     37     return;
     38 
     39   if (!input.text().empty()) {
     40     std::string input_utf8 = UTF16ToUTF8(input.text());
     41     for (ExtensionApps::const_iterator app = extension_apps_.begin();
     42          app != extension_apps_.end(); ++app) {
     43       // See if the input matches this extension application.
     44       const std::string& name = app->first;
     45       const std::string& url = app->second;
     46       std::string::const_iterator name_iter =
     47           std::search(name.begin(),
     48                       name.end(),
     49                       input_utf8.begin(),
     50                       input_utf8.end(),
     51                       base::CaseInsensitiveCompare<char>());
     52       std::string::const_iterator url_iter =
     53           std::search(url.begin(),
     54                       url.end(),
     55                       input_utf8.begin(),
     56                       input_utf8.end(),
     57                       base::CaseInsensitiveCompare<char>());
     58 
     59       bool matches_name = name_iter != name.end();
     60       bool matches_url = url_iter != url.end() &&
     61                          input.type() != AutocompleteInput::FORCED_QUERY;
     62       if (matches_name || matches_url) {
     63         // We have a match, might be a partial match.
     64         // TODO(finnur): Figure out what type to return here, might want to have
     65         // the extension icon/a generic icon show up in the Omnibox.
     66         AutocompleteMatch match(this, 0, false,
     67                                 AutocompleteMatch::EXTENSION_APP);
     68         match.fill_into_edit = UTF8ToUTF16(url);
     69         match.destination_url = GURL(url);
     70         match.inline_autocomplete_offset = string16::npos;
     71         match.contents = UTF8ToUTF16(name);
     72         HighlightMatch(input, &match.contents_class, name_iter, name);
     73         match.description = UTF8ToUTF16(url);
     74         HighlightMatch(input, &match.description_class, url_iter, url);
     75         match.relevance = CalculateRelevance(input.type(),
     76                                              input.text().length(),
     77                                              matches_name ?
     78                                                  name.length() : url.length(),
     79                                              GURL(url));
     80         matches_.push_back(match);
     81       }
     82     }
     83   }
     84 }
     85 
     86 ExtensionAppProvider::~ExtensionAppProvider() {
     87 }
     88 
     89 void ExtensionAppProvider::RefreshAppList() {
     90   ExtensionService* extension_service = profile_->GetExtensionService();
     91   if (!extension_service)
     92     return;  // During testing, there is no extension service.
     93   const ExtensionList* extensions = extension_service->extensions();
     94   extension_apps_.clear();
     95   for (ExtensionList::const_iterator app = extensions->begin();
     96        app != extensions->end(); ++app) {
     97     if ((*app)->is_app() && (*app)->GetFullLaunchURL().is_valid()) {
     98       extension_apps_.push_back(
     99           std::make_pair((*app)->name(),
    100                          (*app)->GetFullLaunchURL().spec()));
    101     }
    102   }
    103 }
    104 
    105 void ExtensionAppProvider::RegisterForNotifications() {
    106   registrar_.Add(this, NotificationType::EXTENSION_LOADED,
    107                  NotificationService::AllSources());
    108   registrar_.Add(this, NotificationType::EXTENSION_UNINSTALLED,
    109                  NotificationService::AllSources());
    110 }
    111 
    112 void ExtensionAppProvider::Observe(NotificationType type,
    113                                    const NotificationSource& source,
    114                                    const NotificationDetails& details) {
    115   RefreshAppList();
    116 }
    117 
    118 void ExtensionAppProvider::HighlightMatch(const AutocompleteInput& input,
    119                                           ACMatchClassifications* match_class,
    120                                           std::string::const_iterator iter,
    121                                           const std::string& match_string) {
    122   size_t pos = iter - match_string.begin();
    123   bool match_found = iter != match_string.end();
    124   if (!match_found || pos > 0) {
    125     match_class->push_back(
    126         ACMatchClassification(0, ACMatchClassification::DIM));
    127   }
    128   if (match_found) {
    129     match_class->push_back(
    130         ACMatchClassification(pos, ACMatchClassification::MATCH));
    131     if (pos + input.text().length() < match_string.length()) {
    132       match_class->push_back(ACMatchClassification(pos + input.text().length(),
    133                                 ACMatchClassification::DIM));
    134     }
    135   }
    136 }
    137 
    138 int ExtensionAppProvider::CalculateRelevance(AutocompleteInput::Type type,
    139                                              int input_length,
    140                                              int target_length,
    141                                              const GURL& url) {
    142   // If you update the algorithm here, please remember to update the tables in
    143   // autocomplete.h also.
    144   const int kMaxRelevance = 1425;
    145 
    146   if (input_length == target_length)
    147     return kMaxRelevance;
    148 
    149   // We give a boost proportionally based on how much of the input matches the
    150   // app name, up to a maximum close to 200 (we can be close to, but we'll never
    151   // reach 200 because the 100% match is taken care of above).
    152   double fraction_boost = static_cast<double>(200) *
    153                           input_length / target_length;
    154 
    155   // We also give a boost relative to how often the user has previously typed
    156   // the Extension App URL/selected the Extension App suggestion from this
    157   // provider (boost is between 200-400).
    158   double type_count_boost = 0;
    159   HistoryService* const history_service =
    160       profile_->GetHistoryService(Profile::EXPLICIT_ACCESS);
    161   history::URLDatabase* url_db = history_service ?
    162       history_service->InMemoryDatabase() : NULL;
    163   if (url_db) {
    164     history::URLRow info;
    165     url_db->GetRowForURL(url, &info);
    166     type_count_boost =
    167         400 * (1.0 - (std::pow(static_cast<double>(2), -info.typed_count())));
    168   }
    169   int relevance = 575 + static_cast<int>(type_count_boost) +
    170                         static_cast<int>(fraction_boost);
    171   DCHECK_LE(relevance, kMaxRelevance);
    172   return relevance;
    173 }
    174