blob: 76f85cca74f46d09e1b886da46401696636b931a [file] [log] [blame]
[email protected]e41982a72012-11-20 07:16:511// Copyright 2012 The Chromium Authors. All rights reserved.
[email protected]7acfaf92012-07-11 15:51:592// 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
[email protected]0a46856e2013-04-24 00:33:027#include "base/bind.h"
[email protected]7acfaf92012-07-11 15:51:598#include "chrome/browser/extensions/extension_service.h"
[email protected]26015ac2013-02-27 02:50:139#include "chrome/browser/extensions/extension_web_ui.h"
[email protected]7acfaf92012-07-11 15:51:5910#include "chrome/browser/profiles/profile.h"
[email protected]0a46856e2013-04-24 00:33:0211#include "chrome/browser/search/instant_service.h"
12#include "chrome/browser/search/instant_service_factory.h"
[email protected]a7b8e43d2013-03-18 18:52:4313#include "chrome/browser/search/search.h"
[email protected]7acfaf92012-07-11 15:51:5914#include "chrome/browser/ui/browser.h"
[email protected]7acfaf92012-07-11 15:51:5915#include "chrome/browser/ui/browser_window.h"
16#include "chrome/browser/ui/omnibox/location_bar.h"
[email protected]6cf51b62013-08-10 13:49:2217#include "chrome/browser/ui/omnibox/omnibox_popup_model.h"
[email protected]c18cb672012-12-05 04:42:1218#include "chrome/browser/ui/omnibox/omnibox_view.h"
[email protected]4ff347e2013-07-22 19:39:0019#include "chrome/browser/ui/search/instant_ntp.h"
[email protected]4066a695d2013-06-20 14:08:5420#include "chrome/browser/ui/search/search_model.h"
[email protected]9d3d11702012-11-08 01:01:1221#include "chrome/browser/ui/search/search_tab_helper.h"
[email protected]e41982a72012-11-20 07:16:5122#include "chrome/browser/ui/tabs/tab_strip_model.h"
[email protected]7acfaf92012-07-11 15:51:5923#include "chrome/browser/ui/webui/ntp/app_launcher_handler.h"
[email protected]0c9406632013-02-08 01:13:3324#include "chrome/common/url_constants.h"
[email protected]0a46856e2013-04-24 00:33:0225#include "content/public/browser/render_process_host.h"
[email protected]233f0f962013-02-27 21:14:1926#include "content/public/browser/user_metrics.h"
[email protected]0a46856e2013-04-24 00:33:0227#include "content/public/browser/web_contents.h"
[email protected]6cf51b62013-08-10 13:49:2228#include "content/public/browser/web_contents_view.h"
[email protected]a6827652012-11-20 23:41:0829
[email protected]233f0f962013-02-27 21:14:1930using content::UserMetricsAction;
31
[email protected]7acfaf92012-07-11 15:51:5932////////////////////////////////////////////////////////////////////////////////
33// BrowserInstantController, public:
34
35BrowserInstantController::BrowserInstantController(Browser* browser)
[email protected]8a236702012-09-28 13:30:5736 : browser_(browser),
[email protected]a780c7b22013-08-02 18:36:5937 instant_(this),
[email protected]6af41782013-06-22 13:49:1138 instant_unload_handler_(browser) {
[email protected]0b10c9ff2012-10-09 17:31:5539 browser_->search_model()->AddObserver(this);
[email protected]4ff347e2013-07-22 19:39:0040
41 InstantService* instant_service =
42 InstantServiceFactory::GetForProfile(profile());
43 instant_service->OnBrowserInstantControllerCreated();
[email protected]c8a118e2013-09-24 21:33:4044 instant_service->AddObserver(this);
[email protected]7acfaf92012-07-11 15:51:5945}
46
47BrowserInstantController::~BrowserInstantController() {
[email protected]0b10c9ff2012-10-09 17:31:5548 browser_->search_model()->RemoveObserver(this);
[email protected]4ff347e2013-07-22 19:39:0049
50 InstantService* instant_service =
51 InstantServiceFactory::GetForProfile(profile());
[email protected]c8a118e2013-09-24 21:33:4052 instant_service->RemoveObserver(this);
[email protected]4ff347e2013-07-22 19:39:0053 instant_service->OnBrowserInstantControllerDestroyed();
[email protected]7acfaf92012-07-11 15:51:5954}
55
[email protected]0c9406632013-02-08 01:13:3356bool BrowserInstantController::MaybeSwapInInstantNTPContents(
57 const GURL& url,
58 content::WebContents* source_contents,
59 content::WebContents** target_contents) {
60 if (url != GURL(chrome::kChromeUINewTabURL))
61 return false;
62
[email protected]26015ac2013-02-27 02:50:1363 GURL extension_url(url);
64 if (ExtensionWebUI::HandleChromeURLOverride(&extension_url, profile())) {
65 // If there is an extension overriding the NTP do not use the Instant NTP.
66 return false;
67 }
68
[email protected]4ff347e2013-07-22 19:39:0069 InstantService* instant_service =
70 InstantServiceFactory::GetForProfile(profile());
71 scoped_ptr<content::WebContents> instant_ntp =
72 instant_service->ReleaseNTPContents();
[email protected]0c9406632013-02-08 01:13:3373 if (!instant_ntp)
74 return false;
75
76 *target_contents = instant_ntp.get();
[email protected]0c9406632013-02-08 01:13:3377 if (source_contents) {
[email protected]474f8512013-05-31 22:31:1678 // If the Instant NTP hasn't yet committed an entry, we can't call
79 // CopyStateFromAndPrune. Instead, load the Local NTP URL directly in the
80 // source contents.
81 // TODO(sreeram): Always using the local URL is wrong in the case of the
82 // first tab in a window where we might want to use the remote URL. Fix.
83 if (!instant_ntp->GetController().CanPruneAllButVisible()) {
84 source_contents->GetController().LoadURL(chrome::GetLocalInstantURL(
85 profile()), content::Referrer(), content::PAGE_TRANSITION_GENERATED,
86 std::string());
87 *target_contents = source_contents;
88 } else {
89 instant_ntp->GetController().CopyStateFromAndPrune(
90 &source_contents->GetController());
91 ReplaceWebContentsAt(
92 browser_->tab_strip_model()->GetIndexOfWebContents(source_contents),
93 instant_ntp.Pass());
94 }
[email protected]0c9406632013-02-08 01:13:3395 } else {
[email protected]474f8512013-05-31 22:31:1696 // If the Instant NTP hasn't yet committed an entry, we can't call
97 // PruneAllButVisible. In that case, there shouldn't be any entries to
98 // prune anyway.
99 if (instant_ntp->GetController().CanPruneAllButVisible())
100 instant_ntp->GetController().PruneAllButVisible();
101 else
102 CHECK(!instant_ntp->GetController().GetLastCommittedEntry());
103
[email protected]0c9406632013-02-08 01:13:33104 // If |source_contents| is NULL, then the caller is responsible for
105 // inserting instant_ntp into the tabstrip and will take ownership.
106 ignore_result(instant_ntp.release());
107 }
[email protected]0c9406632013-02-08 01:13:33108 return true;
109}
110
[email protected]413558cb2013-06-10 16:44:45111bool BrowserInstantController::OpenInstant(WindowOpenDisposition disposition,
112 const GURL& url) {
[email protected]e41982a72012-11-20 07:16:51113 // Unsupported dispositions.
[email protected]413558cb2013-06-10 16:44:45114 if (disposition == NEW_BACKGROUND_TAB || disposition == NEW_WINDOW ||
115 disposition == NEW_FOREGROUND_TAB)
[email protected]7acfaf92012-07-11 15:51:59116 return false;
[email protected]7acfaf92012-07-11 15:51:59117
[email protected]7acfaf92012-07-11 15:51:59118 // The omnibox currently doesn't use other dispositions, so we don't attempt
[email protected]c72226c82012-10-01 21:02:32119 // to handle them. If you hit this DCHECK file a bug and I'll (sky) add
[email protected]7acfaf92012-07-11 15:51:59120 // support for the new disposition.
[email protected]413558cb2013-06-10 16:44:45121 DCHECK(disposition == CURRENT_TAB) << disposition;
[email protected]c72226c82012-10-01 21:02:32122
[email protected]413558cb2013-06-10 16:44:45123 // If we will not be replacing search terms from this URL, don't send to
124 // InstantController.
125 const string16& search_terms =
126 chrome::GetSearchTermsFromURL(browser_->profile(), url);
127 if (search_terms.empty())
128 return false;
129
130 return instant_.SubmitQuery(search_terms);
[email protected]7acfaf92012-07-11 15:51:59131}
132
[email protected]0c9406632013-02-08 01:13:33133Profile* BrowserInstantController::profile() const {
134 return browser_->profile();
135}
136
[email protected]0c9406632013-02-08 01:13:33137void BrowserInstantController::ReplaceWebContentsAt(
138 int index,
139 scoped_ptr<content::WebContents> new_contents) {
140 DCHECK_NE(TabStripModel::kNoTab, index);
[email protected]d572bfd2013-02-14 06:14:20141 scoped_ptr<content::WebContents> old_contents(browser_->tab_strip_model()->
142 ReplaceWebContentsAt(index, new_contents.release()));
143 instant_unload_handler_.RunUnloadListenersOrDestroy(old_contents.Pass(),
144 index);
[email protected]0c9406632013-02-08 01:13:33145}
146
[email protected]6cf51b62013-08-10 13:49:22147void BrowserInstantController::FocusOmnibox(OmniboxFocusState state) {
[email protected]c18cb672012-12-05 04:42:12148 OmniboxView* omnibox_view = browser_->window()->GetLocationBar()->
149 GetLocationEntry();
[email protected]6cf51b62013-08-10 13:49:22150
151 // Do not add a default case in the switch block for the following reasons:
152 // (1) Explicitly handle the new states. If new states are added in the
153 // OmniboxFocusState, the compiler will warn the developer to handle the new
154 // states.
155 // (2) An attacker may control the renderer and sends the browser process a
156 // malformed IPC. This function responds to the invalid |state| values by
157 // doing nothing instead of crashing the browser process (intentional no-op).
158 switch (state) {
159 case OMNIBOX_FOCUS_VISIBLE:
160 omnibox_view->SetFocus();
161 omnibox_view->model()->SetCaretVisibility(true);
162 break;
163 case OMNIBOX_FOCUS_INVISIBLE:
164 omnibox_view->SetFocus();
165 omnibox_view->model()->SetCaretVisibility(false);
166 // If the user clicked on the fakebox, any text already in the omnibox
167 // should get cleared when they start typing. Selecting all the existing
168 // text is a convenient way to accomplish this. It also gives a slight
169 // visual cue to users who really understand selection state about what
170 // will happen if they start typing.
171 omnibox_view->SelectAll(false);
172 break;
173 case OMNIBOX_FOCUS_NONE:
174 // Remove focus only if the popup is closed. This will prevent someone
175 // from changing the omnibox value and closing the popup without user
176 // interaction.
177 if (!omnibox_view->model()->popup_model()->IsOpen()) {
178 content::WebContents* contents = GetActiveWebContents();
179 if (contents)
180 contents->GetView()->Focus();
181 }
182 break;
[email protected]0d0b4a42013-06-14 00:46:26183 }
[email protected]c18cb672012-12-05 04:42:12184}
185
[email protected]cd533bf2012-12-04 19:14:59186content::WebContents* BrowserInstantController::GetActiveWebContents() const {
187 return browser_->tab_strip_model()->GetActiveWebContents();
[email protected]7acfaf92012-07-11 15:51:59188}
189
[email protected]e41982a72012-11-20 07:16:51190void BrowserInstantController::ActiveTabChanged() {
191 instant_.ActiveTabChanged();
192}
193
[email protected]3d6a8952012-12-14 03:18:07194void BrowserInstantController::TabDeactivated(content::WebContents* contents) {
195 instant_.TabDeactivated(contents);
196}
197
[email protected]3c3acca2013-02-26 03:07:07198void BrowserInstantController::OpenURL(
[email protected]e3033eb2012-12-13 23:46:08199 const GURL& url,
[email protected]3c3acca2013-02-26 03:07:07200 content::PageTransition transition,
201 WindowOpenDisposition disposition) {
[email protected]e3033eb2012-12-13 23:46:08202 browser_->OpenURL(content::OpenURLParams(url,
203 content::Referrer(),
[email protected]3c3acca2013-02-26 03:07:07204 disposition,
[email protected]e3033eb2012-12-13 23:46:08205 transition,
206 false));
207}
208
[email protected]6cf51b62013-08-10 13:49:22209void BrowserInstantController::PasteIntoOmnibox(const string16& text) {
210 OmniboxView* omnibox_view = browser_->window()->GetLocationBar()->
211 GetLocationEntry();
212 // The first case is for right click to paste, where the text is retrieved
213 // from the clipboard already sanitized. The second case is needed to handle
214 // drag-and-drop value and it has to be sanitazed before setting it into the
215 // omnibox.
216 string16 text_to_paste = text.empty() ?
217 omnibox_view->GetClipboardText() :
218 omnibox_view->SanitizeTextForPaste(text);
219
220 if (!text_to_paste.empty()) {
221 if (!omnibox_view->model()->has_focus())
222 omnibox_view->SetFocus();
223 omnibox_view->OnBeforePossibleChange();
224 omnibox_view->model()->on_paste();
225 omnibox_view->SetUserText(text_to_paste);
226 omnibox_view->OnAfterPossibleChange();
227 }
228}
229
[email protected]fcde79a2013-02-28 02:25:09230void BrowserInstantController::SetOmniboxBounds(const gfx::Rect& bounds) {
231 instant_.SetOmniboxBounds(bounds);
[email protected]ec4aad542012-12-14 01:11:04232}
233
[email protected]987fad782013-08-28 06:23:18234void BrowserInstantController::SetSuggestionToPrefetch(
235 const InstantSuggestion& suggestion) {
236 instant_.SetSuggestionToPrefetch(suggestion);
237}
238
[email protected]3473ae02013-06-07 00:28:08239void BrowserInstantController::ToggleVoiceSearch() {
240 instant_.ToggleVoiceSearch();
241}
242
[email protected]7acfaf92012-07-11 15:51:59243////////////////////////////////////////////////////////////////////////////////
[email protected]165fe422013-03-27 06:34:03244// BrowserInstantController, SearchModelObserver implementation:
[email protected]0b10c9ff2012-10-09 17:31:55245
[email protected]5ee671f2013-03-19 11:23:05246void BrowserInstantController::ModelChanged(
[email protected]165fe422013-03-27 06:34:03247 const SearchModel::State& old_state,
248 const SearchModel::State& new_state) {
[email protected]4066a695d2013-06-20 14:08:54249 if (old_state.mode != new_state.mode) {
250 const SearchMode& new_mode = new_state.mode;
[email protected]5ee671f2013-03-19 11:23:05251
[email protected]a780c7b22013-08-02 18:36:59252 // Record some actions corresponding to the mode change. Note that to get
253 // the full story, it's necessary to look at other UMA actions as well,
254 // such as tab switches.
255 if (new_mode.is_search_results())
256 content::RecordAction(UserMetricsAction("InstantExtended.ShowSRP"));
257 else if (new_mode.is_ntp())
258 content::RecordAction(UserMetricsAction("InstantExtended.ShowNTP"));
[email protected]5ee671f2013-03-19 11:23:05259
[email protected]4066a695d2013-06-20 14:08:54260 instant_.SearchModeChanged(old_state.mode, new_mode);
[email protected]c19ba1042013-03-11 17:17:13261 }
262
[email protected]4066a695d2013-06-20 14:08:54263 if (old_state.instant_support != new_state.instant_support)
264 instant_.InstantSupportChanged(new_state.instant_support);
[email protected]7acfaf92012-07-11 15:51:59265}
266
[email protected]c8a118e2013-09-24 21:33:40267////////////////////////////////////////////////////////////////////////////////
268// BrowserInstantController, InstantServiceObserver implementation:
[email protected]0a46856e2013-04-24 00:33:02269
[email protected]c8a118e2013-09-24 21:33:40270void BrowserInstantController::DefaultSearchProviderChanged() {
271 ReloadTabsInInstantProcess();
272}
[email protected]0a46856e2013-04-24 00:33:02273
[email protected]c8a118e2013-09-24 21:33:40274void BrowserInstantController::GoogleURLUpdated() {
275 ReloadTabsInInstantProcess();
276}
277
278void BrowserInstantController::ReloadTabsInInstantProcess() {
[email protected]0a46856e2013-04-24 00:33:02279 InstantService* instant_service =
[email protected]c8a118e2013-09-24 21:33:40280 InstantServiceFactory::GetForProfile(profile());
[email protected]0a46856e2013-04-24 00:33:02281 if (!instant_service)
282 return;
283
284 TabStripModel* tab_model = browser_->tab_strip_model();
285 int count = tab_model->count();
286 for (int index = 0; index < count; ++index) {
287 content::WebContents* contents = tab_model->GetWebContentsAt(index);
288 if (!contents)
289 continue;
290
[email protected]2309e912013-10-01 01:33:30291 // Send new search URLs to the renderer.
292 content::RenderProcessHost* rph = contents->GetRenderProcessHost();
293 instant_service->SendSearchURLsToRenderer(rph);
[email protected]0a46856e2013-04-24 00:33:02294
295 // Reload the contents to ensure that it gets assigned to a non-priviledged
296 // renderer.
[email protected]2309e912013-10-01 01:33:30297 if (!instant_service->IsInstantProcess(rph->GetID()))
298 continue;
[email protected]0a46856e2013-04-24 00:33:02299 contents->GetController().Reload(false);
300 }
[email protected]0a46856e2013-04-24 00:33:02301}