1 // Copyright (c) 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 #ifndef CHROME_BROWSER_FAVICON_FAVICON_HANDLER_H_ 6 #define CHROME_BROWSER_FAVICON_FAVICON_HANDLER_H_ 7 8 #include <map> 9 10 #include "base/basictypes.h" 11 #include "base/callback_forward.h" 12 #include "base/memory/ref_counted.h" 13 #include "chrome/browser/favicon/favicon_service.h" 14 #include "chrome/browser/favicon/favicon_tab_helper.h" 15 #include "chrome/common/cancelable_task_tracker.h" 16 #include "chrome/common/ref_counted_util.h" 17 #include "content/public/common/favicon_url.h" 18 #include "ui/gfx/favicon_size.h" 19 #include "ui/gfx/image/image.h" 20 #include "url/gurl.h" 21 22 class FaviconHandlerDelegate; 23 class Profile; 24 class SkBitmap; 25 26 namespace base { 27 class RefCountedMemory; 28 } 29 30 namespace content { 31 class NavigationEntry; 32 } 33 34 // FaviconHandler works with FaviconTabHelper to fetch the specific type of 35 // favicon. 36 // 37 // FetchFavicon requests the favicon from the favicon service which in turn 38 // requests the favicon from the history database. At this point 39 // we only know the URL of the page, and not necessarily the url of the 40 // favicon. To ensure we handle reloading stale favicons as well as 41 // reloading a favicon on page reload we always request the favicon from 42 // history regardless of whether the NavigationEntry has a favicon. 43 // 44 // After the navigation two types of events are delivered (which is 45 // first depends upon who is faster): notification from the history 46 // db on our request for the favicon (OnFaviconDataForInitialURL), 47 // or a message from the renderer giving us the URL of the favicon for 48 // the page (SetFaviconURL). 49 // . If the history db has a valid up to date favicon for the page, we update 50 // the NavigationEntry and use the favicon. 51 // . When we receive the favicon url if it matches that of the NavigationEntry 52 // and the NavigationEntry's favicon is set, we do nothing (everything is 53 // ok). 54 // . On the other hand if the database does not know the favicon for url, or 55 // the favicon is out date, or the URL from the renderer does not match that 56 // NavigationEntry we proceed to DownloadFaviconOrAskHistory. Before we 57 // invoke DownloadFaviconOrAskHistory we wait until we've received both 58 // the favicon url and the callback from history. We wait to ensure we 59 // truly know both the favicon url and the state of the database. 60 // 61 // DownloadFaviconOrAskHistory does the following: 62 // . If we have a valid favicon, but it is expired we ask the renderer to 63 // download the favicon. 64 // . Otherwise we ask the history database to update the mapping from 65 // page url to favicon url and call us back with the favicon. Remember, it is 66 // possible for the db to already have the favicon, just not the mapping 67 // between page to favicon url. The callback for this is OnFaviconData. 68 // 69 // OnFaviconData either updates the favicon of the NavigationEntry (if the 70 // db knew about the favicon), or requests the renderer to download the 71 // favicon. 72 // 73 // When the renderer downloads favicons, it considers the entire list of 74 // favicon candidates and chooses the one that best matches the preferred size 75 // (or the first one if there is no preferred size). Once the matching favicon 76 // has been determined, SetFavicon is called which updates the favicon of the 77 // NavigationEntry and notifies the database to save the favicon. 78 79 class FaviconHandler { 80 public: 81 enum Type { 82 FAVICON, 83 TOUCH, 84 }; 85 86 FaviconHandler(Profile* profile, 87 FaviconHandlerDelegate* delegate, 88 Type icon_type); 89 virtual ~FaviconHandler(); 90 91 // Initiates loading the favicon for the specified url. 92 void FetchFavicon(const GURL& url); 93 94 // Message Handler. Must be public, because also called from 95 // PrerenderContents. Collects the |image_urls| list. 96 void OnUpdateFaviconURL(int32 page_id, 97 const std::vector<content::FaviconURL>& candidates); 98 99 // Processes the current image_irls_ entry, requesting the image from the 100 // history / download service. 101 void ProcessCurrentUrl(); 102 103 // Message handler for ImageHostMsg_DidDownloadImage. Called when the image 104 // at |image_url| has been downloaded. 105 // |bitmaps| is a list of all the frames of the image at |image_url|. 106 // |original_bitmap_sizes| are the sizes of |bitmaps| before they were resized 107 // to the maximum bitmap size passed to DownloadFavicon(). 108 void OnDidDownloadFavicon( 109 int id, 110 const GURL& image_url, 111 const std::vector<SkBitmap>& bitmaps, 112 const std::vector<gfx::Size>& original_bitmap_sizes); 113 114 // For testing. image_urls()115 const std::deque<content::FaviconURL>& image_urls() const { 116 return image_urls_; 117 } 118 119 protected: 120 // These virtual methods make FaviconHandler testable and are overridden by 121 // TestFaviconHandler. 122 123 // Return the NavigationEntry for the active entry, or NULL if the active 124 // entries URL does not match that of the URL last passed to FetchFavicon. 125 virtual content::NavigationEntry* GetEntry(); 126 127 // Asks the render to download favicon, returns the request id. 128 virtual int DownloadFavicon(const GURL& image_url, int max_bitmap_size); 129 130 // Ask the favicon from history 131 virtual void UpdateFaviconMappingAndFetch( 132 const GURL& page_url, 133 const GURL& icon_url, 134 chrome::IconType icon_type, 135 const FaviconService::FaviconResultsCallback& callback, 136 CancelableTaskTracker* tracker); 137 138 virtual void GetFavicon( 139 const GURL& icon_url, 140 chrome::IconType icon_type, 141 const FaviconService::FaviconResultsCallback& callback, 142 CancelableTaskTracker* tracker); 143 144 virtual void GetFaviconForURL( 145 const GURL& page_url, 146 int icon_types, 147 const FaviconService::FaviconResultsCallback& callback, 148 CancelableTaskTracker* tracker); 149 150 virtual void SetHistoryFavicons(const GURL& page_url, 151 const GURL& icon_url, 152 chrome::IconType icon_type, 153 const gfx::Image& image); 154 155 virtual FaviconService* GetFaviconService(); 156 157 // Returns true if the favicon should be saved. 158 virtual bool ShouldSaveFavicon(const GURL& url); 159 160 // Notifies the delegate that the favicon for the active entry was updated. 161 // |icon_url_changed| is true if a favicon with a different icon URL has been 162 // selected since the previous call to NotifyFaviconUpdated(). 163 virtual void NotifyFaviconUpdated(bool icon_url_changed); 164 165 private: 166 friend class TestFaviconHandler; // For testing 167 168 struct DownloadRequest { 169 DownloadRequest(); 170 ~DownloadRequest(); 171 172 DownloadRequest(const GURL& url, 173 const GURL& image_url, 174 chrome::IconType icon_type); 175 176 GURL url; 177 GURL image_url; 178 chrome::IconType icon_type; 179 }; 180 181 struct FaviconCandidate { 182 FaviconCandidate(); 183 ~FaviconCandidate(); 184 185 FaviconCandidate(const GURL& url, 186 const GURL& image_url, 187 const gfx::Image& image, 188 float score, 189 chrome::IconType icon_type); 190 191 GURL url; 192 GURL image_url; 193 gfx::Image image; 194 float score; 195 chrome::IconType icon_type; 196 }; 197 198 // See description above class for details. 199 void OnFaviconDataForInitialURL( 200 const std::vector<chrome::FaviconBitmapResult>& favicon_bitmap_results); 201 202 // If the favicon has expired, asks the renderer to download the favicon. 203 // Otherwise asks history to update the mapping between page url and icon 204 // url with a callback to OnFaviconData when done. 205 void DownloadFaviconOrAskHistory(const GURL& page_url, 206 const GURL& icon_url, 207 chrome::IconType icon_type); 208 209 // See description above class for details. 210 void OnFaviconData( 211 const std::vector<chrome::FaviconBitmapResult>& favicon_bitmap_results); 212 213 // Schedules a download for the specified entry. This adds the request to 214 // download_requests_. 215 int ScheduleDownload(const GURL& url, 216 const GURL& image_url, 217 chrome::IconType icon_type); 218 219 // Updates |favicon_candidate_| and returns true if it is an exact match. 220 bool UpdateFaviconCandidate(const GURL& url, 221 const GURL& image_url, 222 const gfx::Image& image, 223 float score, 224 chrome::IconType icon_type); 225 226 // Sets the image data for the favicon. 227 void SetFavicon(const GURL& url, 228 const GURL& icon_url, 229 const gfx::Image& image, 230 chrome::IconType icon_type); 231 232 // Sets the favicon's data on the NavigationEntry. 233 // If the WebContents has a delegate, it is invalidated (INVALIDATE_TYPE_TAB). 234 void UpdateFavicon(content::NavigationEntry* entry, 235 const std::vector<chrome::FaviconBitmapResult>& favicon_bitmap_results); 236 void UpdateFavicon(content::NavigationEntry* entry, 237 const GURL& icon_url, 238 const gfx::Image& image); 239 240 // Return the current candidate if any. current_candidate()241 content::FaviconURL* current_candidate() { 242 return (image_urls_.size() > 0) ? &image_urls_[0] : NULL; 243 } 244 245 // Returns the preferred_icon_size according icon_types_, 0 means no 246 // preference. preferred_icon_size()247 int preferred_icon_size() { 248 #if defined(OS_ANDROID) 249 return 0; 250 #else 251 return icon_types_ == chrome::FAVICON ? gfx::kFaviconSize : 0; 252 #endif 253 } 254 255 // Used for FaviconService requests. 256 CancelableTaskTracker cancelable_task_tracker_; 257 258 // URL of the page we're requesting the favicon for. 259 GURL url_; 260 261 // Whether we got the initial response for the favicon back from the renderer. 262 bool got_favicon_from_history_; 263 264 // Whether the favicon is out of date or the favicon data in 265 // |history_results_| is known to be incomplete. If true, it means history 266 // knows about the favicon, but we need to download the favicon because the 267 // icon has expired or the data in the database is incomplete. 268 bool favicon_expired_or_incomplete_; 269 270 // Requests to the renderer to download favicons. 271 typedef std::map<int, DownloadRequest> DownloadRequests; 272 DownloadRequests download_requests_; 273 274 // The combination of the supported icon types. 275 const int icon_types_; 276 277 // The prioritized favicon candidates from the page back from the renderer. 278 std::deque<content::FaviconURL> image_urls_; 279 280 // The FaviconBitmapResults from history. 281 std::vector<chrome::FaviconBitmapResult> history_results_; 282 283 // The Profile associated with this handler. 284 Profile* profile_; 285 286 // This handler's delegate. 287 FaviconHandlerDelegate* delegate_; // weak 288 289 // Current favicon candidate. 290 FaviconCandidate favicon_candidate_; 291 292 DISALLOW_COPY_AND_ASSIGN(FaviconHandler); 293 }; 294 295 #endif // CHROME_BROWSER_FAVICON_FAVICON_HANDLER_H_ 296