164 lines
		
	
	
	
		
			5.6 KiB
			
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			164 lines
		
	
	
	
		
			5.6 KiB
			
		
	
	
	
		
			C++
		
	
	
	
	
	
| // Copyright (c) 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/browser/ui/webui/fileicon_source.h"
 | |
| 
 | |
| #include "base/bind.h"
 | |
| #include "base/callback.h"
 | |
| #include "base/files/file_path.h"
 | |
| #include "base/memory/ref_counted_memory.h"
 | |
| #include "base/message_loop/message_loop.h"
 | |
| #include "base/strings/string_split.h"
 | |
| #include "base/strings/utf_string_conversions.h"
 | |
| #include "chrome/browser/browser_process.h"
 | |
| #include "net/base/escape.h"
 | |
| #include "third_party/skia/include/core/SkBitmap.h"
 | |
| #include "ui/base/webui/web_ui_util.h"
 | |
| #include "ui/gfx/codec/png_codec.h"
 | |
| #include "ui/gfx/image/image.h"
 | |
| #include "ui/gfx/image/image_skia.h"
 | |
| #include "url/gurl.h"
 | |
| 
 | |
| namespace {
 | |
| 
 | |
| typedef std::map<std::string, IconLoader::IconSize> QueryIconSizeMap;
 | |
| 
 | |
| // The path used in internal URLs to file icon data.
 | |
| const char kFileIconPath[] = "fileicon";
 | |
| 
 | |
| // URL parameter specifying icon size.
 | |
| const char kIconSize[] = "iconsize";
 | |
| 
 | |
| // URL parameter specifying scale factor.
 | |
| const char kScaleFactor[] = "scale";
 | |
| 
 | |
| // Assuming the url is of the form '/path?query', convert the path portion into
 | |
| // a FilePath and return the resulting |file_path| and |query|.  The path
 | |
| // portion may have been encoded using encodeURIComponent().
 | |
| void GetFilePathAndQuery(const std::string& url,
 | |
|                          base::FilePath* file_path,
 | |
|                          std::string* query) {
 | |
|   // We receive the url with chrome://fileicon/ stripped but GURL expects it.
 | |
|   const GURL gurl("chrome://fileicon/" + url);
 | |
|   std::string path = net::UnescapeURLComponent(
 | |
|       gurl.path().substr(1),
 | |
|       net::UnescapeRule::URL_SPECIAL_CHARS_EXCEPT_PATH_SEPARATORS |
 | |
|           net::UnescapeRule::PATH_SEPARATORS | net::UnescapeRule::SPACES);
 | |
| 
 | |
|   *file_path = base::FilePath::FromUTF8Unsafe(path);
 | |
|   *file_path = file_path->NormalizePathSeparators();
 | |
|   query->assign(gurl.query());
 | |
| }
 | |
| 
 | |
| IconLoader::IconSize SizeStringToIconSize(const std::string& size_string) {
 | |
|   if (size_string == "small") return IconLoader::SMALL;
 | |
|   if (size_string == "large") return IconLoader::LARGE;
 | |
|   // We default to NORMAL if we don't recognize the size_string. Including
 | |
|   // size_string=="normal".
 | |
|   return IconLoader::NORMAL;
 | |
| }
 | |
| 
 | |
| // Simple parser for data on the query.
 | |
| void ParseQueryParams(const std::string& query,
 | |
|                       float* scale_factor,
 | |
|                       IconLoader::IconSize* icon_size) {
 | |
|   base::StringPairs parameters;
 | |
|   if (icon_size)
 | |
|     *icon_size = IconLoader::NORMAL;
 | |
|   if (scale_factor)
 | |
|     *scale_factor = 1.0f;
 | |
|   base::SplitStringIntoKeyValuePairs(query, '=', '&', ¶meters);
 | |
|   for (base::StringPairs::const_iterator iter = parameters.begin();
 | |
|        iter != parameters.end(); ++iter) {
 | |
|     if (icon_size && iter->first == kIconSize)
 | |
|       *icon_size = SizeStringToIconSize(iter->second);
 | |
|     else if (scale_factor && iter->first == kScaleFactor)
 | |
|       webui::ParseScaleFactor(iter->second, scale_factor);
 | |
|   }
 | |
| }
 | |
| 
 | |
| }  // namespace
 | |
| 
 | |
| FileIconSource::IconRequestDetails::IconRequestDetails() : scale_factor(1.0f) {
 | |
| }
 | |
| 
 | |
| FileIconSource::IconRequestDetails::IconRequestDetails(
 | |
|     const IconRequestDetails& other) = default;
 | |
| 
 | |
| FileIconSource::IconRequestDetails::~IconRequestDetails() {
 | |
| }
 | |
| 
 | |
| FileIconSource::FileIconSource() {}
 | |
| 
 | |
| FileIconSource::~FileIconSource() {}
 | |
| 
 | |
| void FileIconSource::FetchFileIcon(
 | |
|     const base::FilePath& path,
 | |
|     float scale_factor,
 | |
|     IconLoader::IconSize icon_size,
 | |
|     const content::URLDataSource::GotDataCallback& callback) {
 | |
|   IconManager* im = g_browser_process->icon_manager();
 | |
|   gfx::Image* icon = im->LookupIconFromFilepath(path, icon_size);
 | |
| 
 | |
|   if (icon) {
 | |
|     scoped_refptr<base::RefCountedBytes> icon_data(new base::RefCountedBytes);
 | |
|     gfx::PNGCodec::EncodeBGRASkBitmap(
 | |
|         icon->ToImageSkia()->GetRepresentation(scale_factor).sk_bitmap(),
 | |
|         false,
 | |
|         &icon_data->data());
 | |
| 
 | |
|     callback.Run(icon_data.get());
 | |
|   } else {
 | |
|     // Attach the ChromeURLDataManager request ID to the history request.
 | |
|     IconRequestDetails details;
 | |
|     details.callback = callback;
 | |
|     details.scale_factor = scale_factor;
 | |
| 
 | |
|     // Icon was not in cache, go fetch it slowly.
 | |
|     im->LoadIcon(path,
 | |
|                  icon_size,
 | |
|                  base::Bind(&FileIconSource::OnFileIconDataAvailable,
 | |
|                             base::Unretained(this), details),
 | |
|                  &cancelable_task_tracker_);
 | |
|   }
 | |
| }
 | |
| 
 | |
| std::string FileIconSource::GetSource() const {
 | |
|   return kFileIconPath;
 | |
| }
 | |
| 
 | |
| // void FileIconSource::StartDataRequest(
 | |
| //     const std::string& url_path,
 | |
| //     const content::ResourceRequestInfo::WebContentsGetter& wc_getter,
 | |
| //     const content::URLDataSource::GotDataCallback& callback) {
 | |
| //   std::string query;
 | |
| //   base::FilePath file_path;
 | |
| //   IconLoader::IconSize icon_size;
 | |
| //   float scale_factor = 1.0f;
 | |
| //   GetFilePathAndQuery(url_path, &file_path, &query);
 | |
| //   ParseQueryParams(query, &scale_factor, &icon_size);
 | |
| //   FetchFileIcon(file_path, scale_factor, icon_size, callback);
 | |
| // }
 | |
| 
 | |
| std::string FileIconSource::GetMimeType(const std::string&) const {
 | |
|   // Rely on image decoder inferring the correct type.
 | |
|   return std::string();
 | |
| }
 | |
| 
 | |
| void FileIconSource::OnFileIconDataAvailable(const IconRequestDetails& details,
 | |
|                                              gfx::Image* icon) {
 | |
|   if (icon) {
 | |
|     scoped_refptr<base::RefCountedBytes> icon_data(new base::RefCountedBytes);
 | |
|     gfx::PNGCodec::EncodeBGRASkBitmap(
 | |
|         icon->ToImageSkia()->GetRepresentation(
 | |
|             details.scale_factor).sk_bitmap(),
 | |
|         false,
 | |
|         &icon_data->data());
 | |
| 
 | |
|     details.callback.Run(icon_data.get());
 | |
|   } else {
 | |
|     // TODO(glen): send a dummy icon.
 | |
|     details.callback.Run(NULL);
 | |
|   }
 | |
| }
 | 
