| // 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/basictypes.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.h" |
| #include "base/strings/string_split.h" |
| #include "base/strings/utf_string_conversions.h" |
| #include "chrome/browser/browser_process.h" |
| #include "chrome/common/time_format.h" |
| #include "googleurl/src/gurl.h" |
| #include "grit/generated_resources.h" |
| #include "net/base/escape.h" |
| #include "third_party/skia/include/core/SkBitmap.h" |
| #include "ui/gfx/codec/png_codec.h" |
| #include "ui/gfx/image/image.h" |
| #include "ui/gfx/image/image_skia.h" |
| #include "ui/webui/web_ui_util.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 | |
| 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, |
| ui::ScaleFactor* scale_factor, |
| IconLoader::IconSize* icon_size) { |
| typedef std::pair<std::string, std::string> KVPair; |
| std::vector<KVPair> parameters; |
| if (icon_size) |
| *icon_size = IconLoader::NORMAL; |
| if (scale_factor) |
| *scale_factor = ui::SCALE_FACTOR_100P; |
| base::SplitStringIntoKeyValuePairs(query, '=', '&', ¶meters); |
| for (std::vector<KVPair>::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() { |
| } |
| |
| FileIconSource::IconRequestDetails::~IconRequestDetails() { |
| } |
| |
| FileIconSource::FileIconSource() {} |
| |
| FileIconSource::~FileIconSource() {} |
| |
| void FileIconSource::FetchFileIcon( |
| const base::FilePath& path, |
| ui::ScaleFactor 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, |
| int render_process_id, |
| int render_view_id, |
| const content::URLDataSource::GotDataCallback& callback) { |
| std::string query; |
| base::FilePath file_path; |
| ui::ScaleFactor scale_factor; |
| IconLoader::IconSize icon_size; |
| 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); |
| } |
| } |