2016-10-29 12:35:50 +00:00
|
|
|
// Copyright (c) 2011 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/icon_manager.h"
|
|
|
|
|
|
|
|
#include <memory>
|
|
|
|
#include <tuple>
|
|
|
|
|
|
|
|
#include "base/bind.h"
|
|
|
|
#include "base/task_runner.h"
|
|
|
|
#include "third_party/skia/include/core/SkBitmap.h"
|
|
|
|
#include "third_party/skia/include/core/SkCanvas.h"
|
|
|
|
|
2017-02-16 19:19:19 +00:00
|
|
|
namespace {
|
2016-10-29 12:35:50 +00:00
|
|
|
|
2017-02-16 19:19:19 +00:00
|
|
|
void RunCallbackIfNotCanceled(
|
|
|
|
const base::CancelableTaskTracker::IsCanceledCallback& is_canceled,
|
|
|
|
const IconManager::IconRequestCallback& callback,
|
|
|
|
gfx::Image* image) {
|
|
|
|
if (is_canceled.Run())
|
|
|
|
return;
|
|
|
|
callback.Run(image);
|
2016-10-29 12:35:50 +00:00
|
|
|
}
|
|
|
|
|
2017-02-16 19:19:19 +00:00
|
|
|
} // namespace
|
|
|
|
|
|
|
|
IconManager::IconManager() : weak_factory_(this) {}
|
2016-11-02 18:57:16 +00:00
|
|
|
|
2018-04-18 01:57:05 +00:00
|
|
|
IconManager::~IconManager() {}
|
2016-10-29 12:35:50 +00:00
|
|
|
|
2017-02-16 19:19:19 +00:00
|
|
|
gfx::Image* IconManager::LookupIconFromFilepath(const base::FilePath& file_path,
|
2016-10-29 12:35:50 +00:00
|
|
|
IconLoader::IconSize size) {
|
2017-02-16 19:19:19 +00:00
|
|
|
auto group_it = group_cache_.find(file_path);
|
|
|
|
if (group_it == group_cache_.end())
|
|
|
|
return nullptr;
|
2016-10-29 12:35:50 +00:00
|
|
|
|
2017-02-16 19:19:19 +00:00
|
|
|
CacheKey key(group_it->second, size);
|
|
|
|
auto icon_it = icon_cache_.find(key);
|
|
|
|
if (icon_it == icon_cache_.end())
|
|
|
|
return nullptr;
|
2016-10-29 12:35:50 +00:00
|
|
|
|
2017-02-16 19:19:19 +00:00
|
|
|
return icon_it->second.get();
|
2016-10-29 12:35:50 +00:00
|
|
|
}
|
|
|
|
|
2017-02-16 19:19:19 +00:00
|
|
|
base::CancelableTaskTracker::TaskId IconManager::LoadIcon(
|
|
|
|
const base::FilePath& file_path,
|
|
|
|
IconLoader::IconSize size,
|
|
|
|
const IconRequestCallback& callback,
|
|
|
|
base::CancelableTaskTracker* tracker) {
|
|
|
|
base::CancelableTaskTracker::IsCanceledCallback is_canceled;
|
|
|
|
base::CancelableTaskTracker::TaskId id =
|
|
|
|
tracker->NewTrackedTaskId(&is_canceled);
|
2018-04-18 01:57:05 +00:00
|
|
|
IconRequestCallback callback_runner =
|
|
|
|
base::Bind(&RunCallbackIfNotCanceled, is_canceled, callback);
|
2017-02-16 19:19:19 +00:00
|
|
|
|
|
|
|
IconLoader* loader = IconLoader::Create(
|
|
|
|
file_path, size,
|
|
|
|
base::Bind(&IconManager::OnIconLoaded, weak_factory_.GetWeakPtr(),
|
|
|
|
callback_runner, file_path, size));
|
2016-10-29 12:35:50 +00:00
|
|
|
loader->Start();
|
|
|
|
|
2017-02-16 19:19:19 +00:00
|
|
|
return id;
|
2016-10-29 12:35:50 +00:00
|
|
|
}
|
|
|
|
|
2017-02-16 19:19:19 +00:00
|
|
|
void IconManager::OnIconLoaded(IconRequestCallback callback,
|
|
|
|
base::FilePath file_path,
|
|
|
|
IconLoader::IconSize size,
|
|
|
|
std::unique_ptr<gfx::Image> result,
|
|
|
|
const IconLoader::IconGroup& group) {
|
|
|
|
// Cache the bitmap. Watch out: |result| may be null, which indicates a
|
|
|
|
// failure. We assume that if we have an entry in |icon_cache_| it must not be
|
|
|
|
// null.
|
|
|
|
CacheKey key(group, size);
|
|
|
|
if (result) {
|
|
|
|
callback.Run(result.get());
|
|
|
|
icon_cache_[key] = std::move(result);
|
|
|
|
} else {
|
|
|
|
callback.Run(nullptr);
|
|
|
|
icon_cache_.erase(key);
|
2016-10-29 12:35:50 +00:00
|
|
|
}
|
|
|
|
|
2017-02-16 19:19:19 +00:00
|
|
|
group_cache_[file_path] = group;
|
2016-10-29 12:35:50 +00:00
|
|
|
}
|
|
|
|
|
2017-02-16 19:19:19 +00:00
|
|
|
IconManager::CacheKey::CacheKey(const IconLoader::IconGroup& group,
|
2016-10-29 12:35:50 +00:00
|
|
|
IconLoader::IconSize size)
|
2016-11-02 18:57:16 +00:00
|
|
|
: group(group), size(size) {}
|
2016-10-29 12:35:50 +00:00
|
|
|
|
2018-04-18 01:57:05 +00:00
|
|
|
bool IconManager::CacheKey::operator<(const CacheKey& other) const {
|
2016-10-29 12:35:50 +00:00
|
|
|
return std::tie(group, size) < std::tie(other.group, other.size);
|
|
|
|
}
|