Update To 11.40.268.0
[platform/framework/web/crosswalk.git] / src / chrome / browser / ui / browser_instant_controller.cc
1 // Copyright 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/ui/browser_instant_controller.h"
6
7 #include "base/bind.h"
8 #include "chrome/browser/infobars/infobar_service.h"
9 #include "chrome/browser/profiles/profile.h"
10 #include "chrome/browser/search/instant_service.h"
11 #include "chrome/browser/search/instant_service_factory.h"
12 #include "chrome/browser/search/search.h"
13 #include "chrome/browser/ui/browser.h"
14 #include "chrome/browser/ui/browser_window.h"
15 #include "chrome/browser/ui/location_bar/location_bar.h"
16 #include "chrome/browser/ui/omnibox/omnibox_popup_model.h"
17 #include "chrome/browser/ui/omnibox/omnibox_view.h"
18 #include "chrome/browser/ui/search/instant_search_prerenderer.h"
19 #include "chrome/browser/ui/search/search_model.h"
20 #include "chrome/browser/ui/search/search_tab_helper.h"
21 #include "chrome/browser/ui/tabs/tab_strip_model.h"
22 #include "chrome/common/instant_types.h"
23 #include "chrome/common/url_constants.h"
24 #include "content/public/browser/render_process_host.h"
25 #include "content/public/browser/user_metrics.h"
26 #include "content/public/browser/web_contents.h"
27
28
29 // Helpers --------------------------------------------------------------------
30
31 namespace {
32
33 InstantSearchPrerenderer* GetInstantSearchPrerenderer(Profile* profile) {
34   DCHECK(profile);
35   InstantService* instant_service =
36       InstantServiceFactory::GetForProfile(profile);
37   return instant_service ? instant_service->instant_search_prerenderer() : NULL;
38 }
39
40 }  // namespace
41
42
43 // BrowserInstantController ---------------------------------------------------
44
45 BrowserInstantController::BrowserInstantController(Browser* browser)
46     : browser_(browser),
47       instant_(this) {
48   browser_->search_model()->AddObserver(this);
49
50   InstantService* instant_service =
51       InstantServiceFactory::GetForProfile(profile());
52   instant_service->AddObserver(this);
53 }
54
55 BrowserInstantController::~BrowserInstantController() {
56   browser_->search_model()->RemoveObserver(this);
57
58   InstantService* instant_service =
59       InstantServiceFactory::GetForProfile(profile());
60   instant_service->RemoveObserver(this);
61 }
62
63 bool BrowserInstantController::OpenInstant(WindowOpenDisposition disposition,
64                                            const GURL& url) {
65   // Unsupported dispositions.
66   if (disposition == NEW_BACKGROUND_TAB || disposition == NEW_WINDOW ||
67       disposition == NEW_FOREGROUND_TAB)
68     return false;
69
70   // The omnibox currently doesn't use other dispositions, so we don't attempt
71   // to handle them. If you hit this DCHECK file a bug and I'll (sky) add
72   // support for the new disposition.
73   DCHECK(disposition == CURRENT_TAB) << disposition;
74
75   const base::string16& search_terms =
76       chrome::ExtractSearchTermsFromURL(profile(), url);
77   EmbeddedSearchRequestParams request_params(url);
78   if (search_terms.empty())
79     return false;
80
81   InstantSearchPrerenderer* prerenderer =
82       GetInstantSearchPrerenderer(profile());
83   if (prerenderer) {
84     if (prerenderer->CanCommitQuery(GetActiveWebContents(), search_terms)) {
85       // Submit query to render the prefetched results. Browser will swap the
86       // prerendered contents with the active tab contents.
87       prerenderer->Commit(search_terms, request_params);
88       return false;
89     } else {
90       prerenderer->Cancel();
91     }
92   }
93
94   // If we will not be replacing search terms from this URL, don't send to
95   // InstantController.
96   if (!chrome::IsQueryExtractionAllowedForURL(profile(), url))
97     return false;
98   return instant_.SubmitQuery(search_terms, request_params);
99 }
100
101 Profile* BrowserInstantController::profile() const {
102   return browser_->profile();
103 }
104
105 content::WebContents* BrowserInstantController::GetActiveWebContents() const {
106   return browser_->tab_strip_model()->GetActiveWebContents();
107 }
108
109 void BrowserInstantController::ActiveTabChanged() {
110   instant_.ActiveTabChanged();
111 }
112
113 void BrowserInstantController::TabDeactivated(content::WebContents* contents) {
114   instant_.TabDeactivated(contents);
115
116   InstantSearchPrerenderer* prerenderer =
117       GetInstantSearchPrerenderer(profile());
118   if (prerenderer)
119     prerenderer->Cancel();
120 }
121
122 void BrowserInstantController::ModelChanged(
123     const SearchModel::State& old_state,
124     const SearchModel::State& new_state) {
125   if (old_state.mode != new_state.mode) {
126     const SearchMode& new_mode = new_state.mode;
127
128     // Record some actions corresponding to the mode change. Note that to get
129     // the full story, it's necessary to look at other UMA actions as well,
130     // such as tab switches.
131     if (new_mode.is_search_results())
132       content::RecordAction(base::UserMetricsAction("InstantExtended.ShowSRP"));
133     else if (new_mode.is_ntp())
134       content::RecordAction(base::UserMetricsAction("InstantExtended.ShowNTP"));
135
136     instant_.SearchModeChanged(old_state.mode, new_mode);
137   }
138
139   if (old_state.instant_support != new_state.instant_support)
140     instant_.InstantSupportChanged(new_state.instant_support);
141 }
142
143 void BrowserInstantController::DefaultSearchProviderChanged() {
144   InstantService* instant_service =
145       InstantServiceFactory::GetForProfile(profile());
146   if (!instant_service)
147     return;
148
149   TabStripModel* tab_model = browser_->tab_strip_model();
150   int count = tab_model->count();
151   for (int index = 0; index < count; ++index) {
152     content::WebContents* contents = tab_model->GetWebContentsAt(index);
153     if (!contents)
154       continue;
155
156     // Send new search URLs to the renderer.
157     content::RenderProcessHost* rph = contents->GetRenderProcessHost();
158     instant_service->SendSearchURLsToRenderer(rph);
159
160     // Reload the contents to ensure that it gets assigned to a non-priviledged
161     // renderer.
162     if (!instant_service->IsInstantProcess(rph->GetID()))
163       continue;
164
165     contents->GetController().Reload(false);
166
167     // As the reload was not triggered by the user we don't want to close any
168     // infobars. We have to tell the InfoBarService after the reload, otherwise
169     // it would ignore this call when
170     // WebContentsObserver::DidStartNavigationToPendingEntry is invoked.
171     InfoBarService::FromWebContents(contents)->set_ignore_next_reload();
172   }
173 }