// Copyright (c) 2011 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.
#ifndef CHROME_BROWSER_GOOGLE_GOOGLE_URL_TRACKER_H_
#define CHROME_BROWSER_GOOGLE_GOOGLE_URL_TRACKER_H_
#pragma once
#include <string>
#include "base/gtest_prod_util.h"
#include "base/memory/scoped_ptr.h"
#include "chrome/browser/tab_contents/confirm_infobar_delegate.h"
#include "chrome/common/net/url_fetcher.h"
#include "content/common/notification_observer.h"
#include "content/common/notification_registrar.h"
#include "googleurl/src/gurl.h"
#include "net/base/network_change_notifier.h"
class NavigationController;
class PrefService;
class TabContents;
class TemplateURL;
// This object is responsible for checking the Google URL once per network
// change, and if necessary prompting the user to see if they want to change to
// using it. The current and last prompted values are saved to prefs.
//
// Most consumers should only call GoogleURL(), which is guaranteed to
// synchronously return a value at all times (even during startup or in unittest
// mode). Consumers who need to be notified when things change should listen to
// the notification service for NOTIFY_GOOGLE_URL_UPDATED, and call GoogleURL()
// again after receiving it, in order to get the updated value.
//
// To protect users' privacy and reduce server load, no updates will be
// performed (ever) unless at least one consumer registers interest by calling
// RequestServerCheck().
class GoogleURLTracker : public URLFetcher::Delegate,
public NotificationObserver,
public net::NetworkChangeNotifier::IPAddressObserver {
public:
// Only the main browser process loop should call this, when setting up
// g_browser_process->google_url_tracker_. No code other than the
// GoogleURLTracker itself should actually use
// g_browser_process->google_url_tracker() (which shouldn't be hard, since
// there aren't useful public functions on this object for consumers to access
// anyway).
GoogleURLTracker();
virtual ~GoogleURLTracker();
// Returns the current Google URL. This will return a valid URL even in
// unittest mode.
//
// This is the only function most code should ever call.
static GURL GoogleURL();
// Requests that the tracker perform a server check to update the Google URL
// as necessary. This will happen at most once per network change, not
// sooner than five seconds after startup (checks requested before that time
// will occur then; checks requested afterwards will occur immediately, if
// no other checks have been made during this run).
//
// In unittest mode, this function does nothing.
static void RequestServerCheck();
static void RegisterPrefs(PrefService* prefs);
// Notifies the tracker that the user has started a Google search.
// If prompting is necessary, we then listen for the subsequent
// NAV_ENTRY_PENDING notification to get the appropriate NavigationController.
// When the load commits, we'll show the infobar.
static void GoogleURLSearchCommitted();
static const char kDefaultGoogleHomepage[];
static const char kSearchDomainCheckURL[];
// Methods called from InfoBar delegate.
void AcceptGoogleURL(const GURL& google_url);
void CancelGoogleURL(const GURL& google_url);
void InfoBarClosed();
void RedoSearch();
private:
friend class GoogleURLTrackerTest;
typedef InfoBarDelegate* (*InfobarCreator)(TabContents*,
GoogleURLTracker*,
const GURL&);
// Registers consumer interest in getting an updated URL from the server.
// It will be notified as NotificationType::GOOGLE_URL_UPDATED, so the
// consumer should observe this notification before calling this.
void SetNeedToFetch();
// Begins the five-second startup sleep period, unless a test has cleared
// |queue_wakeup_task_|.
void QueueWakeupTask();
// Called when the five second startup sleep has finished. Runs any pending
// fetch.
void FinishSleep();
// Starts the fetch of the up-to-date Google URL if we actually want to fetch
// it and can currently do so.
void StartFetchIfDesirable();
// URLFetcher::Delegate
virtual void OnURLFetchComplete(const URLFetcher *source,
const GURL& url,
const net::URLRequestStatus& status,
int response_code,
const ResponseCookies& cookies,
const std::string& data);
// NotificationObserver
virtual void Observe(NotificationType type,
const NotificationSource& source,
const NotificationDetails& details);
// NetworkChangeNotifier::IPAddressObserver
virtual void OnIPAddressChanged();
void SearchCommitted();
void OnNavigationPending(const NotificationSource& source,
const GURL& pending_url);
void OnNavigationCommittedOrTabClosed(TabContents* tab_contents,
NotificationType::Type type);
void ShowGoogleURLInfoBarIfNecessary(TabContents* tab_contents);
NotificationRegistrar registrar_;
InfobarCreator infobar_creator_;
// TODO(ukai): GoogleURLTracker should track google domain (e.g. google.co.uk)
// rather than URL (e.g. http://www.google.co.uk/), so that user could
// configure to use https in search engine templates.
GURL google_url_;
GURL fetched_google_url_;
ScopedRunnableMethodFactory<GoogleURLTracker> runnable_method_factory_;
scoped_ptr<URLFetcher> fetcher_;
int fetcher_id_;
bool queue_wakeup_task_;
bool in_startup_sleep_; // True if we're in the five-second "no fetching"
// period that begins at browser start.
bool already_fetched_; // True if we've already fetched a URL once this run;
// we won't fetch again until after a restart.
bool need_to_fetch_; // True if a consumer actually wants us to fetch an
// updated URL. If this is never set, we won't
// bother to fetch anything.
// Consumers should observe
// NotificationType::GOOGLE_URL_UPDATED.
bool need_to_prompt_; // True if the last fetched Google URL is not
// matched with current user's default Google URL
// nor the last prompted Google URL.
NavigationController* controller_;
InfoBarDelegate* infobar_;
GURL search_url_;
DISALLOW_COPY_AND_ASSIGN(GoogleURLTracker);
};
// This infobar delegate is declared here (rather than in the .cc file) so test
// code can subclass it.
class GoogleURLTrackerInfoBarDelegate : public ConfirmInfoBarDelegate {
public:
GoogleURLTrackerInfoBarDelegate(TabContents* tab_contents,
GoogleURLTracker* google_url_tracker,
const GURL& new_google_url);
// ConfirmInfoBarDelegate:
virtual bool Accept();
virtual bool Cancel();
virtual void InfoBarClosed();
protected:
virtual ~GoogleURLTrackerInfoBarDelegate();
GoogleURLTracker* google_url_tracker_;
const GURL new_google_url_;
private:
// ConfirmInfoBarDelegate:
virtual string16 GetMessageText() const;
virtual string16 GetButtonLabel(InfoBarButton button) const;
DISALLOW_COPY_AND_ASSIGN(GoogleURLTrackerInfoBarDelegate);
};
#endif // CHROME_BROWSER_GOOGLE_GOOGLE_URL_TRACKER_H_