1 // Copyright 2014 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.
5 #include "chrome/browser/thumbnails/thumbnail_list_source.h"
9 #include "base/base64.h"
10 #include "base/bind.h"
11 #include "base/logging.h"
12 #include "base/memory/ref_counted_memory.h"
13 #include "base/strings/string_util.h"
14 #include "chrome/browser/history/top_sites.h"
15 #include "chrome/browser/profiles/profile.h"
16 #include "chrome/browser/search/instant_io_context.h"
17 #include "chrome/browser/thumbnails/thumbnail_service.h"
18 #include "chrome/browser/thumbnails/thumbnail_service_factory.h"
19 #include "chrome/common/url_constants.h"
20 #include "net/base/escape.h"
21 #include "net/url_request/url_request.h"
25 const char kHtmlHeader[] =
26 "<!DOCTYPE html>\n<html>\n<head>\n<title>TopSites Thumbnails</title>\n"
27 "<meta charset=\"utf-8\">\n"
28 "<style type=\"text/css\">\nimg.thumb {border: 1px solid black;}\n"
29 "li {white-space: nowrap;}\n</style>\n";
30 const char kHtmlBody[] = "</head>\n<body>\n";
31 const char kHtmlFooter[] = "</body>\n</html>\n";
33 // If |want_thumbnails| == true, then renders elements in |mvurl_list| that have
34 // thumbnails, with their thumbnails. Otherwise renders elements in |mvurl_list|
35 // that have no thumbnails.
36 void RenderMostVisitedURLList(
37 const history::MostVisitedURLList& mvurl_list,
38 const std::vector<std::string>& base64_encoded_pngs,
40 std::vector<std::string>* out) {
41 DCHECK_EQ(mvurl_list.size(), base64_encoded_pngs.size());
42 bool doing_forced_urls = true;
43 out->push_back("<div><b>Forced URLs:</b></div>\n"
45 for (size_t i = 0; i < mvurl_list.size(); ++i) {
46 const history::MostVisitedURL& mvurl = mvurl_list[i];
47 if (doing_forced_urls && mvurl.last_forced_time.is_null()) {
48 out->push_back("</ul></div>\n"
49 "<div><b>Non-forced URLs:</b></div>\n"
51 doing_forced_urls = false;
53 bool has_thumbnail = !base64_encoded_pngs[i].empty();
54 if (has_thumbnail == want_thumbnails) {
55 out->push_back("<li>\n");
56 out->push_back(net::EscapeForHTML(mvurl.url.spec()) + "\n");
57 if (want_thumbnails) {
58 out->push_back("<div><img class=\"thumb\" "
59 "src=\"data:image/png;base64," +
60 base64_encoded_pngs[i] + "\"/></div>\n");
62 if (!mvurl.redirects.empty()) {
63 out->push_back("<ul>\n");
64 history::RedirectList::const_iterator jt;
65 for (jt = mvurl.redirects.begin();
66 jt != mvurl.redirects.end(); ++jt) {
67 out->push_back("<li>" + net::EscapeForHTML(jt->spec()) + "</li>\n");
69 out->push_back("</ul>\n");
71 out->push_back("</li>\n");
74 out->push_back("</ul></div>\n");
79 ThumbnailListSource::ThumbnailListSource(Profile* profile)
80 : thumbnail_service_(ThumbnailServiceFactory::GetForProfile(profile)),
82 weak_ptr_factory_(this) {
85 ThumbnailListSource::~ThumbnailListSource() {
88 std::string ThumbnailListSource::GetSource() const {
89 return chrome::kChromeUIThumbnailListHost;
92 void ThumbnailListSource::StartDataRequest(
93 const std::string& path,
94 int render_process_id,
96 const content::URLDataSource::GotDataCallback& callback) {
97 if (!profile_->GetTopSites()) {
102 profile_->GetTopSites()->GetMostVisitedURLs(
103 base::Bind(&ThumbnailListSource::OnMostVisitedURLsAvailable,
104 weak_ptr_factory_.GetWeakPtr(),
108 std::string ThumbnailListSource::GetMimeType(const std::string& path) const {
112 base::MessageLoop* ThumbnailListSource::MessageLoopForRequestPath(
113 const std::string& path) const {
114 // TopSites can be accessed from the IO thread.
115 return thumbnail_service_.get() ?
116 NULL : content::URLDataSource::MessageLoopForRequestPath(path);
119 bool ThumbnailListSource::ShouldServiceRequest(
120 const net::URLRequest* request) const {
121 if (request->url().SchemeIs(chrome::kChromeSearchScheme))
122 return InstantIOContext::ShouldServiceRequest(request);
123 return URLDataSource::ShouldServiceRequest(request);
126 bool ThumbnailListSource::ShouldReplaceExistingSource() const {
130 void ThumbnailListSource::OnMostVisitedURLsAvailable(
131 const content::URLDataSource::GotDataCallback& callback,
132 const history::MostVisitedURLList& mvurl_list) {
133 const size_t num_mv = mvurl_list.size();
134 size_t num_mv_with_thumb = 0;
136 // Encode all available thumbnails and store into |base64_encoded_pngs|.
137 std::vector<std::string> base64_encoded_pngs(num_mv);
138 for (size_t i = 0; i < num_mv; ++i) {
139 scoped_refptr<base::RefCountedMemory> data;
140 if (thumbnail_service_->GetPageThumbnail(mvurl_list[i].url, false, &data)) {
141 base::Base64Encode(std::string(data->front_as<char>(), data->size()),
142 &base64_encoded_pngs[i]);
147 // Render HTML to embed URLs and thumbnails.
148 std::vector<std::string> out;
149 out.push_back(kHtmlHeader);
150 out.push_back(kHtmlBody);
151 if (num_mv_with_thumb > 0) {
152 out.push_back("<h2>TopSites URLs with Thumbnails</h2>\n");
153 RenderMostVisitedURLList(mvurl_list, base64_encoded_pngs, true, &out);
155 if (num_mv_with_thumb < num_mv) {
156 out.push_back("<h2>TopSites URLs without Thumbnails</h2>\n");
157 RenderMostVisitedURLList(mvurl_list, base64_encoded_pngs, false, &out);
159 out.push_back(kHtmlFooter);
161 std::string out_html = JoinString(out, "");
162 callback.Run(base::RefCountedString::TakeString(&out_html));