// Copyright 2012 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include "chrome/renderer/searchbox/searchbox.h" #include <string> #include "base/strings/string_number_conversions.h" #include "base/strings/string_util.h" #include "base/strings/utf_string_conversions.h" #include "chrome/common/chrome_switches.h" #include "chrome/common/favicon/favicon_types.h" #include "chrome/common/favicon/favicon_url_parser.h" #include "chrome/common/omnibox_focus_state.h" #include "chrome/common/render_messages.h" #include "chrome/common/url_constants.h" #include "chrome/renderer/searchbox/searchbox_extension.h" #include "content/public/renderer/render_view.h" #include "grit/renderer_resources.h" #include "net/base/escape.h" #include "third_party/WebKit/public/web/WebDocument.h" #include "third_party/WebKit/public/web/WebFrame.h" #include "third_party/WebKit/public/web/WebView.h" #include "ui/base/resource/resource_bundle.h" #include "url/gurl.h" namespace { // The size of the InstantMostVisitedItem cache. const size_t kMaxInstantMostVisitedItemCacheSize = 100; // Returns true if items stored in |old_item_id_pairs| and |new_items| are // equal. bool AreMostVisitedItemsEqual( const std::vector<InstantMostVisitedItemIDPair>& old_item_id_pairs, const std::vector<InstantMostVisitedItem>& new_items) { if (old_item_id_pairs.size() != new_items.size()) return false; for (size_t i = 0; i < new_items.size(); ++i) { if (new_items[i].url != old_item_id_pairs[i].second.url || new_items[i].title != old_item_id_pairs[i].second.title) { return false; } } return true; } } // namespace namespace internal { // for testing // Parses |path| and fills in |id| with the InstantRestrictedID obtained from // the |path|. |render_view_id| is the ID of the associated RenderView. // // |path| is a pair of |render_view_id| and |restricted_id|, and it is // contained in Instant Extended URLs. A valid |path| is in the form: // <render_view_id>/<restricted_id> // // If the |path| is valid, returns true and fills in |id| with restricted_id // value. If the |path| is invalid, returns false and |id| is not set. bool GetInstantRestrictedIDFromPath(int render_view_id, const std::string& path, InstantRestrictedID* id) { // Check that the path is of Most visited item ID form. std::vector<std::string> tokens; if (Tokenize(path, "/", &tokens) != 2) return false; int view_id = 0; if (!base::StringToInt(tokens[0], &view_id) || view_id != render_view_id) return false; return base::StringToInt(tokens[1], id); } bool GetRestrictedIDFromFaviconUrl(int render_view_id, const GURL& url, std::string* favicon_params, InstantRestrictedID* rid) { // Strip leading slash. std::string raw_path = url.path(); DCHECK_GT(raw_path.length(), (size_t) 0); DCHECK_EQ(raw_path[0], '/'); raw_path = raw_path.substr(1); chrome::ParsedFaviconPath parsed; if (!chrome::ParseFaviconPath(raw_path, chrome::FAVICON, &parsed)) return false; // The part of the URL which details the favicon parameters should be returned // so the favicon URL can be reconstructed, by replacing the restricted_id // with the actual URL from which the favicon is being requested. *favicon_params = raw_path.substr(0, parsed.path_index); // The part of the favicon URL which is supposed to contain the URL from // which the favicon is being requested (i.e., the page's URL) actually // contains a pair in the format "<view_id>/<restricted_id>". If the page's // URL is not in the expected format then the execution must be stopped, // returning |true|, indicating that the favicon URL should be translated // without the page's URL part, to prevent search providers from spoofing // the user's browsing history. For example, the following favicon URL // "chrome-search://favicon/http://www.secretsite.com" it is not in the // expected format "chrome-search://favicon/<view_id>/<restricted_id>" so // the pages's URL part ("http://www.secretsite.com") should be removed // entirely from the translated URL otherwise the search engine would know // if the user has visited that page (by verifying whether the favicon URL // returns an image for a particular page's URL); the translated URL in this // case would be "chrome-search://favicon/" which would simply return the // default favicon. std::string id_part = raw_path.substr(parsed.path_index); InstantRestrictedID id; if (!GetInstantRestrictedIDFromPath(render_view_id, id_part, &id)) return true; *rid = id; return true; } // Parses a thumbnail |url| and fills in |id| with the InstantRestrictedID // obtained from the |url|. |render_view_id| is the ID of the associated // RenderView. // // Valid |url| forms: // chrome-search://thumb/<view_id>/<restricted_id> // // If the |url| is valid, returns true and fills in |id| with restricted_id // value. If the |url| is invalid, returns false and |id| is not set. bool GetRestrictedIDFromThumbnailUrl(int render_view_id, const GURL& url, InstantRestrictedID* id) { // Strip leading slash. std::string path = url.path(); DCHECK_GT(path.length(), (size_t) 0); DCHECK_EQ(path[0], '/'); path = path.substr(1); return GetInstantRestrictedIDFromPath(render_view_id, path, id); } } // namespace internal SearchBox::SearchBox(content::RenderView* render_view) : content::RenderViewObserver(render_view), content::RenderViewObserverTracker<SearchBox>(render_view), app_launcher_enabled_(false), is_focused_(false), is_input_in_progress_(false), is_key_capture_enabled_(false), display_instant_results_(false), most_visited_items_cache_(kMaxInstantMostVisitedItemCacheSize), query_(), start_margin_(0), width_(0) { } SearchBox::~SearchBox() { } void SearchBox::LogEvent(NTPLoggingEventType event) { render_view()->Send(new ChromeViewHostMsg_LogEvent( render_view()->GetRoutingID(), render_view()->GetPageId(), event)); } void SearchBox::LogImpression(int position, const base::string16& provider) { render_view()->Send(new ChromeViewHostMsg_LogImpression( render_view()->GetRoutingID(), render_view()->GetPageId(), position, provider)); } void SearchBox::CheckIsUserSignedInToChromeAs(const base::string16& identity) { render_view()->Send(new ChromeViewHostMsg_ChromeIdentityCheck( render_view()->GetRoutingID(), render_view()->GetPageId(), identity)); } void SearchBox::DeleteMostVisitedItem( InstantRestrictedID most_visited_item_id) { render_view()->Send(new ChromeViewHostMsg_SearchBoxDeleteMostVisitedItem( render_view()->GetRoutingID(), render_view()->GetPageId(), GetURLForMostVisitedItem(most_visited_item_id))); } bool SearchBox::GenerateFaviconURLFromTransientURL(const GURL& transient_url, GURL* url) const { std::string favicon_params; InstantRestrictedID rid = -1; bool success = internal::GetRestrictedIDFromFaviconUrl( render_view()->GetRoutingID(), transient_url, &favicon_params, &rid); if (!success) return false; InstantMostVisitedItem item; std::string item_url; if (rid != -1 && GetMostVisitedItemWithID(rid, &item)) item_url = item.url.spec(); *url = GURL(base::StringPrintf("chrome-search://favicon/%s%s", favicon_params.c_str(), item_url.c_str())); return true; } bool SearchBox::GenerateThumbnailURLFromTransientURL(const GURL& transient_url, GURL* url) const { InstantRestrictedID rid = 0; if (!internal::GetRestrictedIDFromThumbnailUrl(render_view()->GetRoutingID(), transient_url, &rid)) { return false; } GURL most_visited_item_url(GetURLForMostVisitedItem(rid)); if (most_visited_item_url.is_empty()) return false; *url = GURL(base::StringPrintf("chrome-search://thumb/%s", most_visited_item_url.spec().c_str())); return true; } void SearchBox::GetMostVisitedItems( std::vector<InstantMostVisitedItemIDPair>* items) const { return most_visited_items_cache_.GetCurrentItems(items); } bool SearchBox::GetMostVisitedItemWithID( InstantRestrictedID most_visited_item_id, InstantMostVisitedItem* item) const { return most_visited_items_cache_.GetItemWithRestrictedID(most_visited_item_id, item); } const ThemeBackgroundInfo& SearchBox::GetThemeBackgroundInfo() { return theme_info_; } void SearchBox::Focus() { render_view()->Send(new ChromeViewHostMsg_FocusOmnibox( render_view()->GetRoutingID(), render_view()->GetPageId(), OMNIBOX_FOCUS_VISIBLE)); } void SearchBox::NavigateToURL(const GURL& url, WindowOpenDisposition disposition, bool is_most_visited_item_url) { render_view()->Send(new ChromeViewHostMsg_SearchBoxNavigate( render_view()->GetRoutingID(), render_view()->GetPageId(), url, disposition, is_most_visited_item_url)); } void SearchBox::Paste(const base::string16& text) { render_view()->Send(new ChromeViewHostMsg_PasteAndOpenDropdown( render_view()->GetRoutingID(), render_view()->GetPageId(), text)); } void SearchBox::SetVoiceSearchSupported(bool supported) { render_view()->Send(new ChromeViewHostMsg_SetVoiceSearchSupported( render_view()->GetRoutingID(), render_view()->GetPageId(), supported)); } void SearchBox::StartCapturingKeyStrokes() { render_view()->Send(new ChromeViewHostMsg_FocusOmnibox( render_view()->GetRoutingID(), render_view()->GetPageId(), OMNIBOX_FOCUS_INVISIBLE)); } void SearchBox::StopCapturingKeyStrokes() { render_view()->Send(new ChromeViewHostMsg_FocusOmnibox( render_view()->GetRoutingID(), render_view()->GetPageId(), OMNIBOX_FOCUS_NONE)); } void SearchBox::UndoAllMostVisitedDeletions() { render_view()->Send( new ChromeViewHostMsg_SearchBoxUndoAllMostVisitedDeletions( render_view()->GetRoutingID(), render_view()->GetPageId())); } void SearchBox::UndoMostVisitedDeletion( InstantRestrictedID most_visited_item_id) { render_view()->Send(new ChromeViewHostMsg_SearchBoxUndoMostVisitedDeletion( render_view()->GetRoutingID(), render_view()->GetPageId(), GetURLForMostVisitedItem(most_visited_item_id))); } bool SearchBox::OnMessageReceived(const IPC::Message& message) { bool handled = true; IPC_BEGIN_MESSAGE_MAP(SearchBox, message) IPC_MESSAGE_HANDLER(ChromeViewMsg_ChromeIdentityCheckResult, OnChromeIdentityCheckResult) IPC_MESSAGE_HANDLER(ChromeViewMsg_DetermineIfPageSupportsInstant, OnDetermineIfPageSupportsInstant) IPC_MESSAGE_HANDLER(ChromeViewMsg_SearchBoxFocusChanged, OnFocusChanged) IPC_MESSAGE_HANDLER(ChromeViewMsg_SearchBoxMarginChange, OnMarginChange) IPC_MESSAGE_HANDLER(ChromeViewMsg_SearchBoxMostVisitedItemsChanged, OnMostVisitedChanged) IPC_MESSAGE_HANDLER(ChromeViewMsg_SearchBoxPromoInformation, OnPromoInformationReceived) IPC_MESSAGE_HANDLER(ChromeViewMsg_SearchBoxSetDisplayInstantResults, OnSetDisplayInstantResults) IPC_MESSAGE_HANDLER(ChromeViewMsg_SearchBoxSetInputInProgress, OnSetInputInProgress) IPC_MESSAGE_HANDLER(ChromeViewMsg_SearchBoxSetSuggestionToPrefetch, OnSetSuggestionToPrefetch) IPC_MESSAGE_HANDLER(ChromeViewMsg_SearchBoxSubmit, OnSubmit) IPC_MESSAGE_HANDLER(ChromeViewMsg_SearchBoxThemeChanged, OnThemeChanged) IPC_MESSAGE_HANDLER(ChromeViewMsg_SearchBoxToggleVoiceSearch, OnToggleVoiceSearch) IPC_MESSAGE_UNHANDLED(handled = false) IPC_END_MESSAGE_MAP() return handled; } void SearchBox::OnChromeIdentityCheckResult(const base::string16& identity, bool identity_match) { if (render_view()->GetWebView() && render_view()->GetWebView()->mainFrame()) { extensions_v8::SearchBoxExtension::DispatchChromeIdentityCheckResult( render_view()->GetWebView()->mainFrame(), identity, identity_match); } } void SearchBox::OnDetermineIfPageSupportsInstant() { if (render_view()->GetWebView() && render_view()->GetWebView()->mainFrame()) { bool result = extensions_v8::SearchBoxExtension::PageSupportsInstant( render_view()->GetWebView()->mainFrame()); DVLOG(1) << render_view() << " PageSupportsInstant: " << result; render_view()->Send(new ChromeViewHostMsg_InstantSupportDetermined( render_view()->GetRoutingID(), render_view()->GetPageId(), result)); } } void SearchBox::OnFocusChanged(OmniboxFocusState new_focus_state, OmniboxFocusChangeReason reason) { bool key_capture_enabled = new_focus_state == OMNIBOX_FOCUS_INVISIBLE; if (key_capture_enabled != is_key_capture_enabled_) { // Tell the page if the key capture mode changed unless the focus state // changed because of TYPING. This is because in that case, the browser // hasn't really stopped capturing key strokes. // // (More practically, if we don't do this check, the page would receive // onkeycapturechange before the corresponding onchange, and the page would // have no way of telling whether the keycapturechange happened because of // some actual user action or just because they started typing.) if (reason != OMNIBOX_FOCUS_CHANGE_TYPING && render_view()->GetWebView() && render_view()->GetWebView()->mainFrame()) { is_key_capture_enabled_ = key_capture_enabled; DVLOG(1) << render_view() << " OnKeyCaptureChange"; extensions_v8::SearchBoxExtension::DispatchKeyCaptureChange( render_view()->GetWebView()->mainFrame()); } } bool is_focused = new_focus_state == OMNIBOX_FOCUS_VISIBLE; if (is_focused != is_focused_) { is_focused_ = is_focused; DVLOG(1) << render_view() << " OnFocusChange"; if (render_view()->GetWebView() && render_view()->GetWebView()->mainFrame()) { extensions_v8::SearchBoxExtension::DispatchFocusChange( render_view()->GetWebView()->mainFrame()); } } } void SearchBox::OnMarginChange(int margin, int width) { start_margin_ = margin; width_ = width; if (render_view()->GetWebView() && render_view()->GetWebView()->mainFrame()) { extensions_v8::SearchBoxExtension::DispatchMarginChange( render_view()->GetWebView()->mainFrame()); } } void SearchBox::OnMostVisitedChanged( const std::vector<InstantMostVisitedItem>& items) { std::vector<InstantMostVisitedItemIDPair> last_known_items; GetMostVisitedItems(&last_known_items); if (AreMostVisitedItemsEqual(last_known_items, items)) return; // Do not send duplicate onmostvisitedchange events. most_visited_items_cache_.AddItems(items); if (render_view()->GetWebView() && render_view()->GetWebView()->mainFrame()) { extensions_v8::SearchBoxExtension::DispatchMostVisitedChanged( render_view()->GetWebView()->mainFrame()); } } void SearchBox::OnPromoInformationReceived(bool is_app_launcher_enabled) { app_launcher_enabled_ = is_app_launcher_enabled; } void SearchBox::OnSetDisplayInstantResults(bool display_instant_results) { display_instant_results_ = display_instant_results; } void SearchBox::OnSetInputInProgress(bool is_input_in_progress) { if (is_input_in_progress_ != is_input_in_progress) { is_input_in_progress_ = is_input_in_progress; DVLOG(1) << render_view() << " OnSetInputInProgress"; if (render_view()->GetWebView() && render_view()->GetWebView()->mainFrame()) { if (is_input_in_progress_) { extensions_v8::SearchBoxExtension::DispatchInputStart( render_view()->GetWebView()->mainFrame()); } else { extensions_v8::SearchBoxExtension::DispatchInputCancel( render_view()->GetWebView()->mainFrame()); } } } } void SearchBox::OnSetSuggestionToPrefetch(const InstantSuggestion& suggestion) { suggestion_ = suggestion; if (render_view()->GetWebView() && render_view()->GetWebView()->mainFrame()) { DVLOG(1) << render_view() << " OnSetSuggestionToPrefetch"; extensions_v8::SearchBoxExtension::DispatchSuggestionChange( render_view()->GetWebView()->mainFrame()); } } void SearchBox::OnSubmit(const base::string16& query) { query_ = query; if (render_view()->GetWebView() && render_view()->GetWebView()->mainFrame()) { DVLOG(1) << render_view() << " OnSubmit"; extensions_v8::SearchBoxExtension::DispatchSubmit( render_view()->GetWebView()->mainFrame()); } if (!query.empty()) Reset(); } void SearchBox::OnThemeChanged(const ThemeBackgroundInfo& theme_info) { // Do not send duplicate notifications. if (theme_info_ == theme_info) return; theme_info_ = theme_info; if (render_view()->GetWebView() && render_view()->GetWebView()->mainFrame()) { extensions_v8::SearchBoxExtension::DispatchThemeChange( render_view()->GetWebView()->mainFrame()); } } void SearchBox::OnToggleVoiceSearch() { if (render_view()->GetWebView() && render_view()->GetWebView()->mainFrame()) { extensions_v8::SearchBoxExtension::DispatchToggleVoiceSearch( render_view()->GetWebView()->mainFrame()); } } GURL SearchBox::GetURLForMostVisitedItem(InstantRestrictedID item_id) const { InstantMostVisitedItem item; return GetMostVisitedItemWithID(item_id, &item) ? item.url : GURL(); } void SearchBox::Reset() { query_.clear(); suggestion_ = InstantSuggestion(); start_margin_ = 0; width_ = 0; is_focused_ = false; is_key_capture_enabled_ = false; theme_info_ = ThemeBackgroundInfo(); }