| // 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. |
| |
| #include "chrome/browser/history/top_sites.h" |
| |
| #include <algorithm> |
| #include <set> |
| |
| #include "base/command_line.h" |
| #include "base/logging.h" |
| #include "base/md5.h" |
| #include "base/string_util.h" |
| #include "base/utf_string_conversions.h" |
| #include "base/values.h" |
| #include "chrome/browser/history/history_backend.h" |
| #include "chrome/browser/history/history_notifications.h" |
| #include "chrome/browser/history/page_usage_data.h" |
| #include "chrome/browser/history/top_sites_backend.h" |
| #include "chrome/browser/history/top_sites_cache.h" |
| #include "chrome/browser/prefs/pref_service.h" |
| #include "chrome/browser/prefs/scoped_user_pref_update.h" |
| #include "chrome/browser/profiles/profile.h" |
| #include "chrome/browser/ui/webui/most_visited_handler.h" |
| #include "chrome/common/chrome_switches.h" |
| #include "chrome/common/pref_names.h" |
| #include "chrome/common/thumbnail_score.h" |
| #include "content/browser/browser_thread.h" |
| #include "content/browser/tab_contents/navigation_controller.h" |
| #include "content/browser/tab_contents/navigation_entry.h" |
| #include "content/common/notification_service.h" |
| #include "grit/chromium_strings.h" |
| #include "grit/generated_resources.h" |
| #include "grit/locale_settings.h" |
| #include "third_party/skia/include/core/SkBitmap.h" |
| #include "ui/base/l10n/l10n_util.h" |
| #include "ui/gfx/codec/jpeg_codec.h" |
| |
| namespace history { |
| |
| // How many top sites to store in the cache. |
| static const size_t kTopSitesNumber = 20; |
| |
| // Max number of temporary images we'll cache. See comment above |
| // temp_images_ for details. |
| static const size_t kMaxTempTopImages = 8; |
| |
| static const size_t kTopSitesShown = 8; |
| static const int kDaysOfHistory = 90; |
| // Time from startup to first HistoryService query. |
| static const int64 kUpdateIntervalSecs = 15; |
| // Intervals between requests to HistoryService. |
| static const int64 kMinUpdateIntervalMinutes = 1; |
| static const int64 kMaxUpdateIntervalMinutes = 60; |
| |
| // IDs of the sites we force into top sites. |
| static const int kPrepopulatePageIDs[] = |
| { IDS_CHROME_WELCOME_URL, IDS_THEMES_GALLERY_URL }; |
| |
| // Favicons of the sites we force into top sites. |
| static const char kPrepopulateFaviconURLs[][54] = |
| { "chrome://theme/IDR_NEWTAB_CHROME_WELCOME_PAGE_FAVICON", |
| "chrome://theme/IDR_NEWTAB_THEMES_GALLERY_FAVICON" }; |
| |
| static const int kPrepopulateTitleIDs[] = |
| { IDS_NEW_TAB_CHROME_WELCOME_PAGE_TITLE, |
| IDS_NEW_TAB_THEMES_GALLERY_PAGE_TITLE }; |
| |
| namespace { |
| |
| // HistoryDBTask used during migration of thumbnails from history to top sites. |
| // When run on the history thread it collects the top sites and the |
| // corresponding thumbnails. When run back on the ui thread it calls into |
| // TopSites::FinishHistoryMigration. |
| class LoadThumbnailsFromHistoryTask : public HistoryDBTask { |
| public: |
| LoadThumbnailsFromHistoryTask(TopSites* top_sites, |
| int result_count) |
| : top_sites_(top_sites), |
| result_count_(result_count) { |
| // l10n_util isn't thread safe, so cache for use on the db thread. |
| ignore_urls_.insert(l10n_util::GetStringUTF8(IDS_CHROME_WELCOME_URL)); |
| ignore_urls_.insert(l10n_util::GetStringUTF8(IDS_THEMES_GALLERY_URL)); |
| } |
| |
| virtual bool RunOnDBThread(history::HistoryBackend* backend, |
| history::HistoryDatabase* db) { |
| // Get the most visited urls. |
| backend->QueryMostVisitedURLsImpl(result_count_, |
| kDaysOfHistory, |
| &data_.most_visited); |
| |
| // And fetch the thumbnails. |
| for (size_t i = 0; i < data_.most_visited.size(); ++i) { |
| const GURL& url = data_.most_visited[i].url; |
| if (ShouldFetchThumbnailFor(url)) { |
| scoped_refptr<RefCountedBytes> data; |
| backend->GetPageThumbnailDirectly(url, &data); |
| data_.url_to_thumbnail_map[url] = data; |
| } |
| } |
| return true; |
| } |
| |
| virtual void DoneRunOnMainThread() { |
| top_sites_->FinishHistoryMigration(data_); |
| } |
| |
| private: |
| bool ShouldFetchThumbnailFor(const GURL& url) { |
| return ignore_urls_.find(url.spec()) == ignore_urls_.end(); |
| } |
| |
| // Set of URLs we don't load thumbnails for. This is created on the UI thread |
| // and used on the history thread. |
| std::set<std::string> ignore_urls_; |
| |
| scoped_refptr<TopSites> top_sites_; |
| |
| // Number of results to request from history. |
| const int result_count_; |
| |
| ThumbnailMigration data_; |
| |
| DISALLOW_COPY_AND_ASSIGN(LoadThumbnailsFromHistoryTask); |
| }; |
| |
| } // namespace |
| |
| TopSites::TopSites(Profile* profile) |
| : backend_(NULL), |
| cache_(new TopSitesCache()), |
| thread_safe_cache_(new TopSitesCache()), |
| profile_(profile), |
| last_num_urls_changed_(0), |
| blacklist_(NULL), |
| pinned_urls_(NULL), |
| history_state_(HISTORY_LOADING), |
| top_sites_state_(TOP_SITES_LOADING), |
| loaded_(false) { |
| if (!profile_) |
| return; |
| |
| if (NotificationService::current()) { |
| registrar_.Add(this, NotificationType::HISTORY_URLS_DELETED, |
| Source<Profile>(profile_)); |
| registrar_.Add(this, NotificationType::NAV_ENTRY_COMMITTED, |
| NotificationService::AllSources()); |
| } |
| |
| // We create update objects here to be sure that dictionaries are created |
| // in the user preferences. |
| DictionaryPrefUpdate(profile_->GetPrefs(), |
| prefs::kNTPMostVisitedURLsBlacklist).Get(); |
| DictionaryPrefUpdate(profile_->GetPrefs(), |
| prefs::kNTPMostVisitedPinnedURLs).Get(); |
| |
| // Now the dictionaries are guaranteed to exist and we can cache pointers |
| // to them. |
| blacklist_ = |
| profile_->GetPrefs()->GetDictionary(prefs::kNTPMostVisitedURLsBlacklist); |
| pinned_urls_ = |
| profile_->GetPrefs()->GetDictionary(prefs::kNTPMostVisitedPinnedURLs); |
| } |
| |
| void TopSites::Init(const FilePath& db_name) { |
| // Create the backend here, rather than in the constructor, so that |
| // unit tests that do not need the backend can run without a problem. |
| backend_ = new TopSitesBackend; |
| backend_->Init(db_name); |
| backend_->GetMostVisitedThumbnails( |
| &cancelable_consumer_, |
| NewCallback(this, &TopSites::OnGotMostVisitedThumbnails)); |
| |
| // History may have already finished loading by the time we're created. |
| HistoryService* history = profile_->GetHistoryServiceWithoutCreating(); |
| if (history && history->backend_loaded()) { |
| if (history->needs_top_sites_migration()) |
| MigrateFromHistory(); |
| else |
| history_state_ = HISTORY_LOADED; |
| } |
| } |
| |
| bool TopSites::SetPageThumbnail(const GURL& url, |
| const SkBitmap& thumbnail, |
| const ThumbnailScore& score) { |
| DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); |
| |
| if (!loaded_) { |
| // TODO(sky): I need to cache these and apply them after the load |
| // completes. |
| return false; |
| } |
| |
| bool add_temp_thumbnail = false; |
| if (!IsKnownURL(url)) { |
| if (!IsFull()) { |
| add_temp_thumbnail = true; |
| } else { |
| return false; // This URL is not known to us. |
| } |
| } |
| |
| if (!HistoryService::CanAddURL(url)) |
| return false; // It's not a real webpage. |
| |
| scoped_refptr<RefCountedBytes> thumbnail_data; |
| if (!EncodeBitmap(thumbnail, &thumbnail_data)) |
| return false; |
| |
| if (add_temp_thumbnail) { |
| // Always remove the existing entry and then add it back. That way if we end |
| // up with too many temp thumbnails we'll prune the oldest first. |
| RemoveTemporaryThumbnailByURL(url); |
| AddTemporaryThumbnail(url, thumbnail_data, score); |
| return true; |
| } |
| |
| return SetPageThumbnailEncoded(url, thumbnail_data, score); |
| } |
| |
| void TopSites::GetMostVisitedURLs(CancelableRequestConsumer* consumer, |
| GetTopSitesCallback* callback) { |
| // WARNING: this may be invoked on any thread. |
| scoped_refptr<CancelableRequest<GetTopSitesCallback> > request( |
| new CancelableRequest<GetTopSitesCallback>(callback)); |
| // This ensures cancellation of requests when either the consumer or the |
| // provider is deleted. Deletion of requests is also guaranteed. |
| AddRequest(request, consumer); |
| MostVisitedURLList filtered_urls; |
| { |
| base::AutoLock lock(lock_); |
| if (!loaded_) { |
| // A request came in before we finished loading. Put the request in |
| // pending_callbacks_ and we'll notify it when we finish loading. |
| pending_callbacks_.insert(request); |
| return; |
| } |
| |
| filtered_urls = thread_safe_cache_->top_sites(); |
| } |
| request->ForwardResult(GetTopSitesCallback::TupleType(filtered_urls)); |
| } |
| |
| bool TopSites::GetPageThumbnail(const GURL& url, |
| scoped_refptr<RefCountedBytes>* bytes) { |
| // WARNING: this may be invoked on any thread. |
| base::AutoLock lock(lock_); |
| return thread_safe_cache_->GetPageThumbnail(url, bytes); |
| } |
| |
| bool TopSites::GetPageThumbnailScore(const GURL& url, |
| ThumbnailScore* score) { |
| // WARNING: this may be invoked on any thread. |
| base::AutoLock lock(lock_); |
| return thread_safe_cache_->GetPageThumbnailScore(url, score); |
| } |
| |
| bool TopSites::GetTemporaryPageThumbnailScore(const GURL& url, |
| ThumbnailScore* score) { |
| for (TempImages::iterator i = temp_images_.begin(); i != temp_images_.end(); |
| ++i) { |
| if (i->first == url) { |
| *score = i->second.thumbnail_score; |
| return true; |
| } |
| } |
| return false; |
| } |
| |
| |
| // Returns the index of |url| in |urls|, or -1 if not found. |
| static int IndexOf(const MostVisitedURLList& urls, const GURL& url) { |
| for (size_t i = 0; i < urls.size(); i++) { |
| if (urls[i].url == url) |
| return i; |
| } |
| return -1; |
| } |
| |
| void TopSites::MigrateFromHistory() { |
| DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); |
| DCHECK_EQ(history_state_, HISTORY_LOADING); |
| |
| history_state_ = HISTORY_MIGRATING; |
| profile_->GetHistoryService(Profile::EXPLICIT_ACCESS)->ScheduleDBTask( |
| new LoadThumbnailsFromHistoryTask( |
| this, |
| num_results_to_request_from_history()), |
| &cancelable_consumer_); |
| MigratePinnedURLs(); |
| } |
| |
| void TopSites::FinishHistoryMigration(const ThumbnailMigration& data) { |
| DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); |
| DCHECK_EQ(history_state_, HISTORY_MIGRATING); |
| |
| history_state_ = HISTORY_LOADED; |
| |
| SetTopSites(data.most_visited); |
| |
| for (size_t i = 0; i < data.most_visited.size(); ++i) { |
| URLToThumbnailMap::const_iterator image_i = |
| data.url_to_thumbnail_map.find(data.most_visited[i].url); |
| if (image_i != data.url_to_thumbnail_map.end()) { |
| SetPageThumbnailEncoded(data.most_visited[i].url, |
| image_i->second, |
| ThumbnailScore()); |
| } |
| } |
| |
| MoveStateToLoaded(); |
| |
| ResetThreadSafeImageCache(); |
| |
| // We've scheduled all the thumbnails and top sites to be written to the top |
| // sites db, but it hasn't happened yet. Schedule a request on the db thread |
| // that notifies us when done. When done we'll know everything was written and |
| // we can tell history to finish its part of migration. |
| backend_->DoEmptyRequest( |
| &cancelable_consumer_, |
| NewCallback(this, &TopSites::OnHistoryMigrationWrittenToDisk)); |
| } |
| |
| void TopSites::HistoryLoaded() { |
| DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); |
| DCHECK_NE(history_state_, HISTORY_LOADED); |
| |
| if (history_state_ != HISTORY_MIGRATING) { |
| // No migration from history is needed. |
| history_state_ = HISTORY_LOADED; |
| if (top_sites_state_ == TOP_SITES_LOADED_WAITING_FOR_HISTORY) { |
| // TopSites thought it needed migration, but it really didn't. This |
| // typically happens the first time a profile is run with Top Sites |
| // enabled |
| SetTopSites(MostVisitedURLList()); |
| MoveStateToLoaded(); |
| } |
| } |
| } |
| |
| bool TopSites::HasBlacklistedItems() const { |
| return !blacklist_->empty(); |
| } |
| |
| void TopSites::AddBlacklistedURL(const GURL& url) { |
| DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); |
| |
| RemovePinnedURL(url); |
| Value* dummy = Value::CreateNullValue(); |
| { |
| DictionaryPrefUpdate update(profile_->GetPrefs(), |
| prefs::kNTPMostVisitedURLsBlacklist); |
| DictionaryValue* blacklist = update.Get(); |
| blacklist->SetWithoutPathExpansion(GetURLHash(url), dummy); |
| } |
| |
| ResetThreadSafeCache(); |
| } |
| |
| void TopSites::RemoveBlacklistedURL(const GURL& url) { |
| DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); |
| { |
| DictionaryPrefUpdate update(profile_->GetPrefs(), |
| prefs::kNTPMostVisitedURLsBlacklist); |
| DictionaryValue* blacklist = update.Get(); |
| blacklist->RemoveWithoutPathExpansion(GetURLHash(url), NULL); |
| } |
| ResetThreadSafeCache(); |
| } |
| |
| bool TopSites::IsBlacklisted(const GURL& url) { |
| DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); |
| return blacklist_->HasKey(GetURLHash(url)); |
| } |
| |
| void TopSites::ClearBlacklistedURLs() { |
| DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); |
| { |
| DictionaryPrefUpdate update(profile_->GetPrefs(), |
| prefs::kNTPMostVisitedURLsBlacklist); |
| DictionaryValue* blacklist = update.Get(); |
| blacklist->Clear(); |
| } |
| ResetThreadSafeCache(); |
| } |
| |
| void TopSites::AddPinnedURL(const GURL& url, size_t pinned_index) { |
| DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); |
| |
| GURL old; |
| if (GetPinnedURLAtIndex(pinned_index, &old)) |
| RemovePinnedURL(old); |
| |
| if (IsURLPinned(url)) |
| RemovePinnedURL(url); |
| |
| Value* index = Value::CreateIntegerValue(pinned_index); |
| |
| { |
| DictionaryPrefUpdate update(profile_->GetPrefs(), |
| prefs::kNTPMostVisitedPinnedURLs); |
| DictionaryValue* pinned_urls = update.Get(); |
| pinned_urls->SetWithoutPathExpansion(GetURLString(url), index); |
| } |
| |
| ResetThreadSafeCache(); |
| } |
| |
| bool TopSites::IsURLPinned(const GURL& url) { |
| int tmp; |
| return pinned_urls_->GetIntegerWithoutPathExpansion(GetURLString(url), &tmp); |
| } |
| |
| void TopSites::RemovePinnedURL(const GURL& url) { |
| DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); |
| |
| { |
| DictionaryPrefUpdate update(profile_->GetPrefs(), |
| prefs::kNTPMostVisitedPinnedURLs); |
| DictionaryValue* pinned_urls = update.Get(); |
| pinned_urls->RemoveWithoutPathExpansion(GetURLString(url), NULL); |
| } |
| |
| ResetThreadSafeCache(); |
| } |
| |
| bool TopSites::GetPinnedURLAtIndex(size_t index, GURL* url) { |
| for (DictionaryValue::key_iterator it = pinned_urls_->begin_keys(); |
| it != pinned_urls_->end_keys(); ++it) { |
| int current_index; |
| if (pinned_urls_->GetIntegerWithoutPathExpansion(*it, ¤t_index)) { |
| if (static_cast<size_t>(current_index) == index) { |
| *url = GURL(*it); |
| return true; |
| } |
| } |
| } |
| return false; |
| } |
| |
| void TopSites::Shutdown() { |
| profile_ = NULL; |
| // Cancel all requests so that the service doesn't callback to us after we've |
| // invoked Shutdown (this could happen if we have a pending request and |
| // Shutdown is invoked). |
| cancelable_consumer_.CancelAllRequests(); |
| backend_->Shutdown(); |
| } |
| |
| // static |
| void TopSites::DiffMostVisited(const MostVisitedURLList& old_list, |
| const MostVisitedURLList& new_list, |
| TopSitesDelta* delta) { |
| // Add all the old URLs for quick lookup. This maps URLs to the corresponding |
| // index in the input. |
| std::map<GURL, size_t> all_old_urls; |
| for (size_t i = 0; i < old_list.size(); i++) |
| all_old_urls[old_list[i].url] = i; |
| |
| // Check all the URLs in the new set to see which ones are new or just moved. |
| // When we find a match in the old set, we'll reset its index to our special |
| // marker. This allows us to quickly identify the deleted ones in a later |
| // pass. |
| const size_t kAlreadyFoundMarker = static_cast<size_t>(-1); |
| for (size_t i = 0; i < new_list.size(); i++) { |
| std::map<GURL, size_t>::iterator found = all_old_urls.find(new_list[i].url); |
| if (found == all_old_urls.end()) { |
| MostVisitedURLWithRank added; |
| added.url = new_list[i]; |
| added.rank = i; |
| delta->added.push_back(added); |
| } else { |
| if (found->second != i) { |
| MostVisitedURLWithRank moved; |
| moved.url = new_list[i]; |
| moved.rank = i; |
| delta->moved.push_back(moved); |
| } |
| found->second = kAlreadyFoundMarker; |
| } |
| } |
| |
| // Any member without the special marker in the all_old_urls list means that |
| // there wasn't a "new" URL that mapped to it, so it was deleted. |
| for (std::map<GURL, size_t>::const_iterator i = all_old_urls.begin(); |
| i != all_old_urls.end(); ++i) { |
| if (i->second != kAlreadyFoundMarker) |
| delta->deleted.push_back(old_list[i->second]); |
| } |
| } |
| |
| CancelableRequestProvider::Handle TopSites::StartQueryForMostVisited() { |
| DCHECK(loaded_); |
| if (!profile_) |
| return 0; |
| |
| HistoryService* hs = profile_->GetHistoryService(Profile::EXPLICIT_ACCESS); |
| // |hs| may be null during unit tests. |
| if (hs) { |
| return hs->QueryMostVisitedURLs( |
| num_results_to_request_from_history(), |
| kDaysOfHistory, |
| &cancelable_consumer_, |
| NewCallback(this, &TopSites::OnTopSitesAvailableFromHistory)); |
| } |
| return 0; |
| } |
| |
| bool TopSites::IsKnownURL(const GURL& url) { |
| return loaded_ && cache_->IsKnownURL(url); |
| } |
| |
| bool TopSites::IsFull() { |
| return loaded_ && cache_->top_sites().size() >= kTopSitesNumber; |
| } |
| |
| TopSites::~TopSites() { |
| } |
| |
| bool TopSites::SetPageThumbnailNoDB(const GURL& url, |
| const RefCountedBytes* thumbnail_data, |
| const ThumbnailScore& score) { |
| // This should only be invoked when we know about the url. |
| DCHECK(cache_->IsKnownURL(url)); |
| |
| const MostVisitedURL& most_visited = |
| cache_->top_sites()[cache_->GetURLIndex(url)]; |
| Images* image = cache_->GetImage(url); |
| |
| // When comparing the thumbnail scores, we need to take into account the |
| // redirect hops, which are not generated when the thumbnail is because the |
| // redirects weren't known. We fill that in here since we know the redirects. |
| ThumbnailScore new_score_with_redirects(score); |
| new_score_with_redirects.redirect_hops_from_dest = |
| GetRedirectDistanceForURL(most_visited, url); |
| |
| if (!ShouldReplaceThumbnailWith(image->thumbnail_score, |
| new_score_with_redirects) && |
| image->thumbnail.get()) |
| return false; // The one we already have is better. |
| |
| image->thumbnail = const_cast<RefCountedBytes*>(thumbnail_data); |
| image->thumbnail_score = new_score_with_redirects; |
| |
| ResetThreadSafeImageCache(); |
| return true; |
| } |
| |
| bool TopSites::SetPageThumbnailEncoded(const GURL& url, |
| const RefCountedBytes* thumbnail, |
| const ThumbnailScore& score) { |
| if (!SetPageThumbnailNoDB(url, thumbnail, score)) |
| return false; |
| |
| // Update the database. |
| if (!cache_->IsKnownURL(url)) |
| return false; |
| |
| size_t index = cache_->GetURLIndex(url); |
| const MostVisitedURL& most_visited = cache_->top_sites()[index]; |
| backend_->SetPageThumbnail(most_visited, |
| index, |
| *(cache_->GetImage(most_visited.url))); |
| return true; |
| } |
| |
| // static |
| bool TopSites::EncodeBitmap(const SkBitmap& bitmap, |
| scoped_refptr<RefCountedBytes>* bytes) { |
| *bytes = new RefCountedBytes(); |
| SkAutoLockPixels bitmap_lock(bitmap); |
| std::vector<unsigned char> data; |
| if (!gfx::JPEGCodec::Encode( |
| reinterpret_cast<unsigned char*>(bitmap.getAddr32(0, 0)), |
| gfx::JPEGCodec::FORMAT_BGRA, bitmap.width(), |
| bitmap.height(), |
| static_cast<int>(bitmap.rowBytes()), 90, |
| &data)) { |
| return false; |
| } |
| // As we're going to cache this data, make sure the vector is only as big as |
| // it needs to be. |
| (*bytes)->data = data; |
| return true; |
| } |
| |
| void TopSites::RemoveTemporaryThumbnailByURL(const GURL& url) { |
| for (TempImages::iterator i = temp_images_.begin(); i != temp_images_.end(); |
| ++i) { |
| if (i->first == url) { |
| temp_images_.erase(i); |
| return; |
| } |
| } |
| } |
| |
| void TopSites::AddTemporaryThumbnail(const GURL& url, |
| const RefCountedBytes* thumbnail, |
| const ThumbnailScore& score) { |
| if (temp_images_.size() == kMaxTempTopImages) |
| temp_images_.erase(temp_images_.begin()); |
| |
| TempImage image; |
| image.first = url; |
| image.second.thumbnail = const_cast<RefCountedBytes*>(thumbnail); |
| image.second.thumbnail_score = score; |
| temp_images_.push_back(image); |
| } |
| |
| void TopSites::TimerFired() { |
| StartQueryForMostVisited(); |
| } |
| |
| // static |
| int TopSites::GetRedirectDistanceForURL(const MostVisitedURL& most_visited, |
| const GURL& url) { |
| for (size_t i = 0; i < most_visited.redirects.size(); i++) { |
| if (most_visited.redirects[i] == url) |
| return static_cast<int>(most_visited.redirects.size() - i - 1); |
| } |
| NOTREACHED() << "URL should always be found."; |
| return 0; |
| } |
| |
| // static |
| MostVisitedURLList TopSites::GetPrepopulatePages() { |
| MostVisitedURLList urls; |
| urls.resize(arraysize(kPrepopulatePageIDs)); |
| for (size_t i = 0; i < arraysize(kPrepopulatePageIDs); ++i) { |
| MostVisitedURL& url = urls[i]; |
| url.url = GURL(l10n_util::GetStringUTF8(kPrepopulatePageIDs[i])); |
| url.redirects.push_back(url.url); |
| url.favicon_url = GURL(kPrepopulateFaviconURLs[i]); |
| url.title = l10n_util::GetStringUTF16(kPrepopulateTitleIDs[i]); |
| } |
| return urls; |
| } |
| |
| // static |
| bool TopSites::AddPrepopulatedPages(MostVisitedURLList* urls) { |
| bool added = false; |
| MostVisitedURLList prepopulate_urls = GetPrepopulatePages(); |
| for (size_t i = 0; i < prepopulate_urls.size(); ++i) { |
| if (urls->size() < kTopSitesNumber && |
| IndexOf(*urls, prepopulate_urls[i].url) == -1) { |
| urls->push_back(prepopulate_urls[i]); |
| added = true; |
| } |
| } |
| return added; |
| } |
| |
| void TopSites::MigratePinnedURLs() { |
| DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); |
| |
| std::map<GURL, size_t> tmp_map; |
| for (DictionaryValue::key_iterator it = pinned_urls_->begin_keys(); |
| it != pinned_urls_->end_keys(); ++it) { |
| Value* value; |
| if (!pinned_urls_->GetWithoutPathExpansion(*it, &value)) |
| continue; |
| |
| if (value->IsType(DictionaryValue::TYPE_DICTIONARY)) { |
| DictionaryValue* dict = static_cast<DictionaryValue*>(value); |
| std::string url_string; |
| int index; |
| if (dict->GetString("url", &url_string) && |
| dict->GetInteger("index", &index)) |
| tmp_map[GURL(url_string)] = index; |
| } |
| } |
| |
| { |
| DictionaryPrefUpdate update(profile_->GetPrefs(), |
| prefs::kNTPMostVisitedPinnedURLs); |
| DictionaryValue* pinned_urls = update.Get(); |
| pinned_urls->Clear(); |
| } |
| |
| for (std::map<GURL, size_t>::iterator it = tmp_map.begin(); |
| it != tmp_map.end(); ++it) |
| AddPinnedURL(it->first, it->second); |
| } |
| |
| void TopSites::ApplyBlacklistAndPinnedURLs(const MostVisitedURLList& urls, |
| MostVisitedURLList* out) { |
| MostVisitedURLList urls_copy; |
| for (size_t i = 0; i < urls.size(); i++) { |
| if (!IsBlacklisted(urls[i].url)) |
| urls_copy.push_back(urls[i]); |
| } |
| |
| for (size_t pinned_index = 0; pinned_index < kTopSitesShown; pinned_index++) { |
| GURL url; |
| bool found = GetPinnedURLAtIndex(pinned_index, &url); |
| if (!found) |
| continue; |
| |
| DCHECK(!url.is_empty()); |
| int cur_index = IndexOf(urls_copy, url); |
| MostVisitedURL tmp; |
| if (cur_index < 0) { |
| // Pinned URL not in urls. |
| tmp.url = url; |
| } else { |
| tmp = urls_copy[cur_index]; |
| urls_copy.erase(urls_copy.begin() + cur_index); |
| } |
| if (pinned_index > out->size()) |
| out->resize(pinned_index); // Add empty URLs as fillers. |
| out->insert(out->begin() + pinned_index, tmp); |
| } |
| |
| // Add non-pinned URLs in the empty spots. |
| size_t current_url = 0; // Index into the remaining URLs in urls_copy. |
| for (size_t i = 0; i < kTopSitesShown && current_url < urls_copy.size(); |
| i++) { |
| if (i == out->size()) { |
| out->push_back(urls_copy[current_url]); |
| current_url++; |
| } else if (i < out->size()) { |
| if ((*out)[i].url.is_empty()) { |
| // Replace the filler |
| (*out)[i] = urls_copy[current_url]; |
| current_url++; |
| } |
| } else { |
| NOTREACHED(); |
| } |
| } |
| } |
| |
| std::string TopSites::GetURLString(const GURL& url) { |
| return cache_->GetCanonicalURL(url).spec(); |
| } |
| |
| std::string TopSites::GetURLHash(const GURL& url) { |
| // We don't use canonical URLs here to be able to blacklist only one of |
| // the two 'duplicate' sites, e.g. 'gmail.com' and 'mail.google.com'. |
| return MD5String(url.spec()); |
| } |
| |
| base::TimeDelta TopSites::GetUpdateDelay() { |
| if (cache_->top_sites().size() <= arraysize(kPrepopulateTitleIDs)) |
| return base::TimeDelta::FromSeconds(30); |
| |
| int64 range = kMaxUpdateIntervalMinutes - kMinUpdateIntervalMinutes; |
| int64 minutes = kMaxUpdateIntervalMinutes - |
| last_num_urls_changed_ * range / cache_->top_sites().size(); |
| return base::TimeDelta::FromMinutes(minutes); |
| } |
| |
| // static |
| void TopSites::ProcessPendingCallbacks( |
| const PendingCallbackSet& pending_callbacks, |
| const MostVisitedURLList& urls) { |
| PendingCallbackSet::const_iterator i; |
| for (i = pending_callbacks.begin(); |
| i != pending_callbacks.end(); ++i) { |
| scoped_refptr<CancelableRequest<GetTopSitesCallback> > request = *i; |
| if (!request->canceled()) |
| request->ForwardResult(GetTopSitesCallback::TupleType(urls)); |
| } |
| } |
| |
| void TopSites::Observe(NotificationType type, |
| const NotificationSource& source, |
| const NotificationDetails& details) { |
| if (!loaded_) |
| return; |
| |
| if (type == NotificationType::HISTORY_URLS_DELETED) { |
| Details<history::URLsDeletedDetails> deleted_details(details); |
| if (deleted_details->all_history) { |
| SetTopSites(MostVisitedURLList()); |
| backend_->ResetDatabase(); |
| } else { |
| std::set<size_t> indices_to_delete; // Indices into top_sites_. |
| for (std::set<GURL>::iterator i = deleted_details->urls.begin(); |
| i != deleted_details->urls.end(); ++i) { |
| if (cache_->IsKnownURL(*i)) |
| indices_to_delete.insert(cache_->GetURLIndex(*i)); |
| } |
| |
| if (indices_to_delete.empty()) |
| return; |
| |
| MostVisitedURLList new_top_sites(cache_->top_sites()); |
| for (std::set<size_t>::reverse_iterator i = indices_to_delete.rbegin(); |
| i != indices_to_delete.rend(); i++) { |
| size_t index = *i; |
| RemovePinnedURL(new_top_sites[index].url); |
| new_top_sites.erase(new_top_sites.begin() + index); |
| } |
| SetTopSites(new_top_sites); |
| } |
| StartQueryForMostVisited(); |
| } else if (type == NotificationType::NAV_ENTRY_COMMITTED) { |
| if (!IsFull()) { |
| NavigationController::LoadCommittedDetails* load_details = |
| Details<NavigationController::LoadCommittedDetails>(details).ptr(); |
| if (!load_details) |
| return; |
| const GURL& url = load_details->entry->url(); |
| if (!cache_->IsKnownURL(url) && HistoryService::CanAddURL(url)) { |
| // To avoid slamming history we throttle requests when the url updates. |
| // To do otherwise negatively impacts perf tests. |
| RestartQueryForTopSitesTimer(GetUpdateDelay()); |
| } |
| } |
| } |
| } |
| |
| void TopSites::SetTopSites(const MostVisitedURLList& new_top_sites) { |
| DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); |
| |
| MostVisitedURLList top_sites(new_top_sites); |
| AddPrepopulatedPages(&top_sites); |
| |
| TopSitesDelta delta; |
| DiffMostVisited(cache_->top_sites(), top_sites, &delta); |
| if (!delta.deleted.empty() || !delta.added.empty() || !delta.moved.empty()) |
| backend_->UpdateTopSites(delta); |
| |
| last_num_urls_changed_ = delta.added.size() + delta.moved.size(); |
| |
| // We always do the following steps (setting top sites in cache, and resetting |
| // thread safe cache ...) as this method is invoked during startup at which |
| // point the caches haven't been updated yet. |
| cache_->SetTopSites(top_sites); |
| |
| // See if we have any tmp thumbnails for the new sites. |
| if (!temp_images_.empty()) { |
| for (size_t i = 0; i < top_sites.size(); ++i) { |
| const MostVisitedURL& mv = top_sites[i]; |
| GURL canonical_url = cache_->GetCanonicalURL(mv.url); |
| // At the time we get the thumbnail redirects aren't known, so we have to |
| // iterate through all the images. |
| for (TempImages::iterator it = temp_images_.begin(); |
| it != temp_images_.end(); ++it) { |
| if (canonical_url == cache_->GetCanonicalURL(it->first)) { |
| SetPageThumbnailEncoded(mv.url, |
| it->second.thumbnail, |
| it->second.thumbnail_score); |
| temp_images_.erase(it); |
| break; |
| } |
| } |
| } |
| } |
| |
| if (top_sites.size() >= kTopSitesNumber) |
| temp_images_.clear(); |
| |
| ResetThreadSafeCache(); |
| ResetThreadSafeImageCache(); |
| |
| // Restart the timer that queries history for top sites. This is done to |
| // ensure we stay in sync with history. |
| RestartQueryForTopSitesTimer(GetUpdateDelay()); |
| } |
| |
| int TopSites::num_results_to_request_from_history() const { |
| DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); |
| |
| return kTopSitesNumber + blacklist_->size(); |
| } |
| |
| void TopSites::MoveStateToLoaded() { |
| DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); |
| |
| MostVisitedURLList filtered_urls; |
| PendingCallbackSet pending_callbacks; |
| { |
| base::AutoLock lock(lock_); |
| |
| if (loaded_) |
| return; // Don't do anything if we're already loaded. |
| loaded_ = true; |
| |
| // Now that we're loaded we can service the queued up callbacks. Copy them |
| // here and service them outside the lock. |
| if (!pending_callbacks_.empty()) { |
| filtered_urls = thread_safe_cache_->top_sites(); |
| pending_callbacks.swap(pending_callbacks_); |
| } |
| } |
| |
| ProcessPendingCallbacks(pending_callbacks, filtered_urls); |
| |
| NotificationService::current()->Notify(NotificationType::TOP_SITES_LOADED, |
| Source<Profile>(profile_), |
| Details<TopSites>(this)); |
| } |
| |
| void TopSites::ResetThreadSafeCache() { |
| base::AutoLock lock(lock_); |
| MostVisitedURLList cached; |
| ApplyBlacklistAndPinnedURLs(cache_->top_sites(), &cached); |
| thread_safe_cache_->SetTopSites(cached); |
| } |
| |
| void TopSites::ResetThreadSafeImageCache() { |
| base::AutoLock lock(lock_); |
| thread_safe_cache_->SetThumbnails(cache_->images()); |
| thread_safe_cache_->RemoveUnreferencedThumbnails(); |
| } |
| |
| void TopSites::RestartQueryForTopSitesTimer(base::TimeDelta delta) { |
| if (timer_.IsRunning() && ((timer_start_time_ + timer_.GetCurrentDelay()) < |
| (base::TimeTicks::Now() + delta))) { |
| return; |
| } |
| |
| timer_start_time_ = base::TimeTicks::Now(); |
| timer_.Stop(); |
| timer_.Start(delta, this, &TopSites::TimerFired); |
| } |
| |
| void TopSites::OnHistoryMigrationWrittenToDisk(TopSitesBackend::Handle handle) { |
| DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); |
| |
| if (!profile_) |
| return; |
| |
| HistoryService* history = |
| profile_->GetHistoryService(Profile::EXPLICIT_ACCESS); |
| if (history) |
| history->OnTopSitesReady(); |
| } |
| |
| void TopSites::OnGotMostVisitedThumbnails( |
| CancelableRequestProvider::Handle handle, |
| scoped_refptr<MostVisitedThumbnails> data, |
| bool may_need_history_migration) { |
| DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); |
| DCHECK_EQ(top_sites_state_, TOP_SITES_LOADING); |
| |
| if (!may_need_history_migration) { |
| top_sites_state_ = TOP_SITES_LOADED; |
| |
| // Set the top sites directly in the cache so that SetTopSites diffs |
| // correctly. |
| cache_->SetTopSites(data->most_visited); |
| SetTopSites(data->most_visited); |
| cache_->SetThumbnails(data->url_to_images_map); |
| |
| ResetThreadSafeImageCache(); |
| |
| MoveStateToLoaded(); |
| |
| // Start a timer that refreshes top sites from history. |
| RestartQueryForTopSitesTimer( |
| base::TimeDelta::FromSeconds(kUpdateIntervalSecs)); |
| } else { |
| // The top sites file didn't exist or is the wrong version. We need to wait |
| // for history to finish loading to know if we really needed to migrate. |
| if (history_state_ == HISTORY_LOADED) { |
| top_sites_state_ = TOP_SITES_LOADED; |
| SetTopSites(MostVisitedURLList()); |
| MoveStateToLoaded(); |
| } else { |
| top_sites_state_ = TOP_SITES_LOADED_WAITING_FOR_HISTORY; |
| // Ask for history just in case it hasn't been loaded yet. When history |
| // finishes loading we'll do migration and/or move to loaded. |
| profile_->GetHistoryService(Profile::EXPLICIT_ACCESS); |
| } |
| } |
| } |
| |
| void TopSites::OnTopSitesAvailableFromHistory( |
| CancelableRequestProvider::Handle handle, |
| MostVisitedURLList pages) { |
| SetTopSites(pages); |
| |
| // Used only in testing. |
| NotificationService::current()->Notify( |
| NotificationType::TOP_SITES_UPDATED, |
| Source<TopSites>(this), |
| Details<CancelableRequestProvider::Handle>(&handle)); |
| } |
| |
| } // namespace history |