| // 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/icon_loader.h" |
| |
| #include <map> |
| #include <string> |
| #include <utility> |
| |
| #include "base/bind.h" |
| #include "base/files/file_path.h" |
| #include "base/lazy_instance.h" |
| #include "base/memory/ref_counted_memory.h" |
| #include "base/message_loop/message_loop.h" |
| #include "base/strings/string_util.h" |
| #include "chrome/browser/icon_loader.h" |
| #include "grit/theme_resources.h" |
| #include "third_party/skia/include/core/SkBitmap.h" |
| #include "ui/base/layout.h" |
| #include "ui/base/resource/resource_bundle.h" |
| #include "ui/gfx/canvas.h" |
| #include "ui/gfx/codec/png_codec.h" |
| #include "ui/gfx/image/image.h" |
| #include "ui/gfx/image/image_skia.h" |
| #include "ui/gfx/image/image_skia_operations.h" |
| |
| namespace { |
| |
| // Used with GenerateImageWithSize() to indicate that the image shouldn't be |
| // resized. |
| const int kDoNotResize = -1; |
| |
| struct IdrBySize { |
| int idr_small; |
| int idr_normal; |
| int idr_large; |
| }; |
| |
| // Performs mapping of <file extension, icon size> to icon resource IDs. |
| class IconMapper { |
| public: |
| IconMapper(); |
| |
| // Lookup icon resource ID for a given filename |extension| and |
| // |icon_size|. Defaults to generic icons if there are no icons for the given |
| // extension. |
| int Lookup(const std::string& extension, IconLoader::IconSize icon_size); |
| |
| private: |
| typedef std::map<std::string, IdrBySize> ExtensionIconMap; |
| |
| ExtensionIconMap extension_icon_map_; |
| }; |
| |
| const IdrBySize kAudioIdrs = { |
| IDR_FILETYPE_AUDIO, |
| IDR_FILETYPE_LARGE_AUDIO, |
| IDR_FILETYPE_LARGE_AUDIO |
| }; |
| const IdrBySize kGenericIdrs = { |
| IDR_FILETYPE_GENERIC, |
| IDR_FILETYPE_LARGE_GENERIC, |
| IDR_FILETYPE_LARGE_GENERIC |
| }; |
| const IdrBySize kImageIdrs = { |
| IDR_FILETYPE_IMAGE, |
| IDR_FILETYPE_IMAGE, |
| IDR_FILETYPE_IMAGE |
| }; |
| #if defined(USE_PROPRIETARY_CODECS) |
| const IdrBySize kPdfIdrs = { |
| IDR_FILETYPE_PDF, |
| IDR_FILETYPE_PDF, |
| IDR_FILETYPE_PDF |
| }; |
| #endif |
| const IdrBySize kVideoIdrs = { |
| IDR_FILETYPE_VIDEO, |
| IDR_FILETYPE_LARGE_VIDEO, |
| IDR_FILETYPE_LARGE_VIDEO |
| }; |
| |
| IconMapper::IconMapper() { |
| // The code below should match translation in |
| // ui/file_manager/file_manager/js/file_manager.js |
| // ui/file_manager/file_manager/css/file_manager.css |
| // 'audio': /\.(mp3|m4a|oga|ogg|wav)$/i, |
| // 'html': /\.(html?)$/i, |
| // 'image': /\.(bmp|gif|jpe?g|ico|png|webp)$/i, |
| // 'pdf' : /\.(pdf)$/i, |
| // 'text': /\.(pod|rst|txt|log)$/i, |
| // 'video': /\.(mov|mp4|m4v|mpe?g4?|ogm|ogv|ogx|webm)$/i |
| |
| const ExtensionIconMap::value_type kExtensionIdrBySizeData[] = { |
| #if defined(USE_PROPRIETARY_CODECS) |
| std::make_pair(".m4a", kAudioIdrs), |
| std::make_pair(".mp3", kAudioIdrs), |
| std::make_pair(".pdf", kPdfIdrs), |
| std::make_pair(".3gp", kVideoIdrs), |
| std::make_pair(".avi", kVideoIdrs), |
| std::make_pair(".m4v", kVideoIdrs), |
| std::make_pair(".mov", kVideoIdrs), |
| std::make_pair(".mp4", kVideoIdrs), |
| std::make_pair(".mpeg", kVideoIdrs), |
| std::make_pair(".mpg", kVideoIdrs), |
| std::make_pair(".mpeg4", kVideoIdrs), |
| std::make_pair(".mpg4", kVideoIdrs), |
| #endif |
| std::make_pair(".flac", kAudioIdrs), |
| std::make_pair(".oga", kAudioIdrs), |
| std::make_pair(".ogg", kAudioIdrs), |
| std::make_pair(".wav", kAudioIdrs), |
| std::make_pair(".bmp", kImageIdrs), |
| std::make_pair(".gif", kImageIdrs), |
| std::make_pair(".ico", kImageIdrs), |
| std::make_pair(".jpeg", kImageIdrs), |
| std::make_pair(".jpg", kImageIdrs), |
| std::make_pair(".png", kImageIdrs), |
| std::make_pair(".webp", kImageIdrs), |
| std::make_pair(".ogm", kVideoIdrs), |
| std::make_pair(".ogv", kVideoIdrs), |
| std::make_pair(".ogx", kVideoIdrs), |
| std::make_pair(".webm", kVideoIdrs), |
| }; |
| |
| const size_t kESize = arraysize(kExtensionIdrBySizeData); |
| ExtensionIconMap source(&kExtensionIdrBySizeData[0], |
| &kExtensionIdrBySizeData[kESize]); |
| extension_icon_map_.swap(source); |
| } |
| |
| int IconMapper::Lookup(const std::string& extension, |
| IconLoader::IconSize icon_size) { |
| DCHECK(icon_size == IconLoader::SMALL || |
| icon_size == IconLoader::NORMAL || |
| icon_size == IconLoader::LARGE); |
| ExtensionIconMap::const_iterator it = extension_icon_map_.find(extension); |
| const IdrBySize& idrbysize = |
| ((it == extension_icon_map_.end()) ? kGenericIdrs : it->second); |
| int idr = -1; |
| switch (icon_size) { |
| case IconLoader::SMALL: idr = idrbysize.idr_small; break; |
| case IconLoader::NORMAL: idr = idrbysize.idr_normal; break; |
| case IconLoader::LARGE: idr = idrbysize.idr_large; break; |
| case IconLoader::ALL: |
| default: |
| NOTREACHED(); |
| } |
| return idr; |
| } |
| |
| // Returns a copy of |source| that is |dip_size| in width and height. If |
| // |dip_size| is |kDoNotResize|, returns an unmodified copy of |source|. |
| // |source| must be a square image (width == height). |
| gfx::ImageSkia ResizeImage(const gfx::ImageSkia& source, int dip_size) { |
| DCHECK(!source.isNull()); |
| DCHECK(source.width() == source.height()); |
| |
| if (dip_size == kDoNotResize || source.width() == dip_size) |
| return source; |
| |
| return gfx::ImageSkiaOperations::CreateResizedImage(source, |
| skia::ImageOperations::RESIZE_BEST, gfx::Size(dip_size, dip_size)); |
| } |
| |
| int IconSizeToDIPSize(IconLoader::IconSize size) { |
| switch (size) { |
| case IconLoader::SMALL: return 16; |
| case IconLoader::NORMAL: return 32; |
| case IconLoader::LARGE: // fallthrough |
| // On ChromeOS, we consider LARGE to mean "the largest image we have." |
| // Since we have already chosen the largest applicable image resource, we |
| // return the image as-is. |
| case IconLoader::ALL: // fallthrough |
| default: |
| return kDoNotResize; |
| } |
| } |
| |
| } // namespace |
| |
| // static |
| IconGroupID IconLoader::ReadGroupIDFromFilepath( |
| const base::FilePath& filepath) { |
| return base::ToLowerASCII(filepath.Extension()); |
| } |
| |
| // static |
| bool IconLoader::IsIconMutableFromFilepath(const base::FilePath&) { |
| return false; |
| } |
| |
| // static |
| content::BrowserThread::ID IconLoader::ReadIconThreadID() { |
| return content::BrowserThread::FILE; |
| } |
| |
| void IconLoader::ReadIcon() { |
| static base::LazyInstance<IconMapper>::Leaky icon_mapper = |
| LAZY_INSTANCE_INITIALIZER; |
| int idr = icon_mapper.Get().Lookup(group_, icon_size_); |
| ResourceBundle& rb = ResourceBundle::GetSharedInstance(); |
| gfx::ImageSkia image_skia(ResizeImage(*(rb.GetImageNamed(idr)).ToImageSkia(), |
| IconSizeToDIPSize(icon_size_))); |
| image_skia.MakeThreadSafe(); |
| image_.reset(new gfx::Image(image_skia)); |
| target_task_runner_->PostTask( |
| FROM_HERE, base::Bind(&IconLoader::NotifyDelegate, this)); |
| } |