blob: e47af46a5087de08cf05a69412b2acba180a6429 [file] [log] [blame]
[email protected]25cc7502012-01-31 19:33:551// Copyright (c) 2012 The Chromium Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
5#include "chrome/browser/icon_loader.h"
6
7#include <map>
8#include <string>
9#include <utility>
10
11#include "base/bind.h"
12#include "base/lazy_instance.h"
13#include "base/message_loop.h"
14#include "chrome/browser/icon_loader.h"
15#include "grit/component_extension_resources.h"
16#include "skia/ext/image_operations.h"
17#include "third_party/skia/include/core/SkBitmap.h"
18#include "ui/base/resource/resource_bundle.h"
19#include "ui/gfx/canvas_skia.h"
20#include "ui/gfx/codec/png_codec.h"
21#include "ui/gfx/image/image.h"
22#include "webkit/glue/image_decoder.h"
23
24namespace {
25
26// Used with GenerateBitmapWithSize() to indicate that the image shouldn't be
27// resized.
28const int kDoNotResize = -1;
29
30struct IdrBySize {
31 int idr_small;
32 int idr_normal;
33 int idr_large;
34};
35
36// Performs mapping of <file extension, icon size> to icon resource IDs.
37class IconMapper {
38 public:
39 IconMapper();
40
41 // Lookup icon resource ID for a given filename |extension| and
42 // |icon_size|. Defaults to generic icons if there are no icons for the given
43 // extension.
44 int Lookup(const std::string& extension, IconLoader::IconSize icon_size);
45
46 private:
47 typedef std::map<std::string, IdrBySize> ExtensionIconMap;
48
49 ExtensionIconMap extension_icon_map_;
50};
51
52const IdrBySize kAudioIdrs = {
53 IDR_FILE_MANAGER_IMG_FILETYPE_AUDIO,
54 IDR_FILE_MANAGER_IMG_FILETYPE_LARGE_AUDIO,
55 IDR_FILE_MANAGER_IMG_FILETYPE_LARGE_AUDIO
56};
57const IdrBySize kGenericIdrs = {
58 IDR_FILE_MANAGER_IMG_FILETYPE_GENERIC,
59 IDR_FILE_MANAGER_IMG_FILETYPE_LARGE_GENERIC,
60 IDR_FILE_MANAGER_IMG_FILETYPE_LARGE_GENERIC
61};
62const IdrBySize kHtmlIdrs = {
63 IDR_FILE_MANAGER_IMG_FILETYPE_HTML,
64 IDR_FILE_MANAGER_IMG_FILETYPE_HTML,
65 IDR_FILE_MANAGER_IMG_FILETYPE_HTML
66};
67const IdrBySize kImageIdrs = {
68 IDR_FILE_MANAGER_IMG_FILETYPE_IMAGE,
69 IDR_FILE_MANAGER_IMG_FILETYPE_IMAGE,
70 IDR_FILE_MANAGER_IMG_FILETYPE_IMAGE
71};
72const IdrBySize kPdfIdrs = {
73 IDR_FILE_MANAGER_IMG_FILETYPE_PDF,
74 IDR_FILE_MANAGER_IMG_FILETYPE_PDF,
75 IDR_FILE_MANAGER_IMG_FILETYPE_PDF
76};
77const IdrBySize kTextIdrs = {
78 IDR_FILE_MANAGER_IMG_FILETYPE_TEXT,
79 IDR_FILE_MANAGER_IMG_FILETYPE_TEXT,
80 IDR_FILE_MANAGER_IMG_FILETYPE_TEXT
81};
82const IdrBySize kVideoIdrs = {
83 IDR_FILE_MANAGER_IMG_FILETYPE_VIDEO,
84 IDR_FILE_MANAGER_IMG_FILETYPE_LARGE_VIDEO,
85 IDR_FILE_MANAGER_IMG_FILETYPE_LARGE_VIDEO
86};
87
88IconMapper::IconMapper() {
89 // The code below should match translation in
90 // chrome/browser/resources/file_manager/js/file_manager.js
91 // chrome/browser/resources/file_manager/css/file_manager.css
92 // 'audio': /\.(mp3|m4a|oga|ogg|wav)$/i,
93 // 'html': /\.(html?)$/i,
94 // 'image': /\.(bmp|gif|jpe?g|ico|png|webp)$/i,
95 // 'pdf' : /\.(pdf)$/i,
96 // 'text': /\.(pod|rst|txt|log)$/i,
97 // 'video': /\.(mov|mp4|m4v|mpe?g4?|ogm|ogv|ogx|webm)$/i
98
99 const ExtensionIconMap::value_type kExtensionIdrBySizeData[] = {
100#if defined(GOOGLE_CHROME_BUILD) || defined(USE_PROPRIETARY_CODECS)
101 std::make_pair(".m4a", kAudioIdrs),
102 std::make_pair(".mp3", kAudioIdrs),
103 std::make_pair(".pdf", kPdfIdrs),
104 std::make_pair(".3gp", kVideoIdrs),
105 std::make_pair(".avi", kVideoIdrs),
106 std::make_pair(".m4v", kVideoIdrs),
107 std::make_pair(".mov", kVideoIdrs),
108 std::make_pair(".mp4", kVideoIdrs),
109 std::make_pair(".mpeg", kVideoIdrs),
110 std::make_pair(".mpg", kVideoIdrs),
111 std::make_pair(".mpeg4", kVideoIdrs),
112 std::make_pair(".mpg4", kVideoIdrs),
113#endif
114 std::make_pair(".flac", kAudioIdrs),
115 std::make_pair(".oga", kAudioIdrs),
116 std::make_pair(".ogg", kAudioIdrs),
117 std::make_pair(".wav", kAudioIdrs),
118 std::make_pair(".htm", kHtmlIdrs),
119 std::make_pair(".html", kHtmlIdrs),
120 std::make_pair(".bmp", kImageIdrs),
121 std::make_pair(".gif", kImageIdrs),
122 std::make_pair(".ico", kImageIdrs),
123 std::make_pair(".jpeg", kImageIdrs),
124 std::make_pair(".jpg", kImageIdrs),
125 std::make_pair(".png", kImageIdrs),
126 std::make_pair(".webp", kImageIdrs),
127 std::make_pair(".log", kTextIdrs),
128 std::make_pair(".pod", kTextIdrs),
129 std::make_pair(".rst", kTextIdrs),
130 std::make_pair(".txt", kTextIdrs),
131 std::make_pair(".ogm", kVideoIdrs),
132 std::make_pair(".ogv", kVideoIdrs),
133 std::make_pair(".ogx", kVideoIdrs),
134 std::make_pair(".webm", kVideoIdrs),
135 };
136
137 const size_t kESize = arraysize(kExtensionIdrBySizeData);
138 ExtensionIconMap source(&kExtensionIdrBySizeData[0],
139 &kExtensionIdrBySizeData[kESize]);
140 extension_icon_map_.swap(source);
141}
142
143int IconMapper::Lookup(const std::string& extension,
144 IconLoader::IconSize icon_size) {
145 DCHECK(icon_size == IconLoader::SMALL ||
146 icon_size == IconLoader::NORMAL ||
147 icon_size == IconLoader::LARGE);
148 ExtensionIconMap::const_iterator it = extension_icon_map_.find(extension);
149 const IdrBySize& idrbysize =
150 ((it == extension_icon_map_.end()) ? kGenericIdrs : it->second);
151 int idr = -1;
152 switch (icon_size) {
153 case IconLoader::SMALL: idr = idrbysize.idr_small; break;
154 case IconLoader::NORMAL: idr = idrbysize.idr_normal; break;
155 case IconLoader::LARGE: idr = idrbysize.idr_large; break;
156 case IconLoader::ALL:
157 default:
158 NOTREACHED();
159 }
160 return idr;
161}
162
163// Returns a copy of |source| that is |pixel_size| in width and height. If
164// |pixel_size| is |kDoNotResize|, returns an unmodified copy of |source|.
165// |source| must be a square image (width == height).
166SkBitmap* GenerateBitmapWithSize(SkBitmap* source, int pixel_size) {
167 DCHECK(source);
168 DCHECK(source->width() == source->height());
169
170 if (pixel_size == kDoNotResize || source->width() == pixel_size)
171 return new SkBitmap(*source);
172
173 return new SkBitmap(skia::ImageOperations::Resize(
174 *source, skia::ImageOperations::RESIZE_BEST, pixel_size, pixel_size));
175}
176
177int IconSizeToPixelSize(IconLoader::IconSize size) {
178 switch (size) {
179 case IconLoader::SMALL: return 16;
180 case IconLoader::NORMAL: return 32;
181 case IconLoader::LARGE: // fallthrough
182 // On ChromeOS, we consider LARGE to mean "the largest image we have."
183 // Since we have already chosen the largest applicable image resource, we
184 // return the image as-is.
185 case IconLoader::ALL: // fallthrough
186 default:
187 return kDoNotResize;
188 }
189}
190
191} // namespace
192
193void IconLoader::ReadIcon() {
194 static base::LazyInstance<IconMapper>::Leaky icon_mapper =
195 LAZY_INSTANCE_INITIALIZER;
196 int idr = icon_mapper.Get().Lookup(group_, icon_size_);
197 ResourceBundle& rb = ResourceBundle::GetSharedInstance();
198 scoped_refptr<RefCountedStaticMemory> bytes(rb.LoadDataResourceBytes(idr));
199 DCHECK(bytes.get());
200 SkBitmap bitmap;
201 if (!gfx::PNGCodec::Decode(bytes->front(), bytes->size(), &bitmap))
202 NOTREACHED();
203 image_.reset(new gfx::Image(
204 GenerateBitmapWithSize(&bitmap, IconSizeToPixelSize(icon_size_))));
205 target_message_loop_->PostTask(
206 FROM_HERE, base::Bind(&IconLoader::NotifyDelegate, this));
207}