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);
|
|
}
|
|
}
|