1 // 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.
5 #ifndef UI_BASE_RESOURCE_RESOURCE_BUNDLE_H_
6 #define UI_BASE_RESOURCE_RESOURCE_BUNDLE_H_
8 #include "build/build_config.h"
13 #include "base/basictypes.h"
14 #include "base/files/file.h"
15 #include "base/files/file_path.h"
16 #include "base/gtest_prod_util.h"
17 #include "base/memory/scoped_ptr.h"
18 #include "base/memory/scoped_vector.h"
19 #include "base/strings/string16.h"
20 #include "base/strings/string_piece.h"
21 #include "ui/base/layout.h"
22 #include "ui/base/ui_base_export.h"
23 #include "ui/gfx/font_list.h"
24 #include "ui/gfx/image/image.h"
25 #include "ui/gfx/native_widget_types.h"
31 class RefCountedStaticMemory
;
39 // ResourceBundle is a central facility to load images and other resources,
40 // such as theme graphics. Every resource is loaded only once.
41 class UI_BASE_EXPORT ResourceBundle
{
43 // An enumeration of the various font styles used throughout Chrome.
44 // The following holds true for the font sizes:
45 // Small <= SmallBold <= Base <= Bold <= Medium <= MediumBold <= Large.
47 // NOTE: depending upon the locale, using one of the *BoldFont below
48 // may *not* actually result in a bold font.
60 // Images are flipped in RTL locales.
62 // Images are never flipped.
66 // Delegate class that allows interception of pack file loading and resource
67 // requests. The methods of this class may be called on multiple threads.
70 // Called before a resource pack file is loaded. Return the full path for
71 // the pack file to continue loading or an empty value to cancel loading.
72 // |pack_path| will contain the complete default path for the pack file if
73 // known or just the pack file name otherwise.
74 virtual base::FilePath
GetPathForResourcePack(
75 const base::FilePath
& pack_path
,
76 ScaleFactor scale_factor
) = 0;
78 // Called before a locale pack file is loaded. Return the full path for
79 // the pack file to continue loading or an empty value to cancel loading.
80 // |pack_path| will contain the complete default path for the pack file if
81 // known or just the pack file name otherwise.
82 virtual base::FilePath
GetPathForLocalePack(
83 const base::FilePath
& pack_path
,
84 const std::string
& locale
) = 0;
86 // Return an image resource or an empty value to attempt retrieval of the
88 virtual gfx::Image
GetImageNamed(int resource_id
) = 0;
90 // Return an image resource or an empty value to attempt retrieval of the
92 virtual gfx::Image
GetNativeImageNamed(int resource_id
, ImageRTL rtl
) = 0;
94 // Return a static memory resource or NULL to attempt retrieval of the
96 virtual base::RefCountedStaticMemory
* LoadDataResourceBytes(
98 ScaleFactor scale_factor
) = 0;
100 // Retrieve a raw data resource. Return true if a resource was provided or
101 // false to attempt retrieval of the default resource.
102 virtual bool GetRawDataResource(int resource_id
,
103 ScaleFactor scale_factor
,
104 base::StringPiece
* value
) = 0;
106 // Retrieve a localized string. Return true if a string was provided or
107 // false to attempt retrieval of the default string.
108 virtual bool GetLocalizedString(int message_id
, base::string16
* value
) = 0;
110 // Returns a font or NULL to attempt retrieval of the default resource.
111 virtual scoped_ptr
<gfx::Font
> GetFont(FontStyle style
) = 0;
114 virtual ~Delegate() {}
117 // Initialize the ResourceBundle for this process. Does not take ownership of
118 // the |delegate| value. Returns the language selected.
119 // NOTE: Mac ignores this and always loads up resources for the language
120 // defined by the Cocoa UI (i.e., NSBundle does the language work).
122 // TODO(sergeyu): This method also loads common resources (i.e. chrome.pak).
123 // There is no way to specify which resource files are loaded, i.e. names of
124 // the files are hardcoded in ResourceBundle. Fix it to allow to specify which
125 // files are loaded (e.g. add a new method in Delegate).
126 static std::string
InitSharedInstanceWithLocale(
127 const std::string
& pref_locale
, Delegate
* delegate
);
129 // Same as InitSharedInstanceWithLocale(), but loads only localized resources,
130 // without default resource packs.
131 static std::string
InitSharedInstanceLocaleOnly(
132 const std::string
& pref_locale
, Delegate
* delegate
);
134 // Initialize the ResourceBundle using given file. The second argument
135 // controls whether or not ResourceBundle::LoadCommonResources is called.
136 // This allows the use of this function in a sandbox without local file
137 // access (as on Android).
138 static void InitSharedInstanceWithPakFile(base::File file
,
139 bool should_load_common_resources
);
141 // Initialize the ResourceBundle using given data pack path for testing.
142 static void InitSharedInstanceWithPakPath(const base::FilePath
& path
);
144 // Delete the ResourceBundle for this process if it exists.
145 static void CleanupSharedInstance();
147 // Returns true after the global resource loader instance has been created.
148 static bool HasSharedInstance();
150 // Return the global resource loader instance.
151 static ResourceBundle
& GetSharedInstance();
153 // Check if the .pak for the given locale exists.
154 bool LocaleDataPakExists(const std::string
& locale
);
156 // Registers additional data pack files with this ResourceBundle. When
157 // looking for a DataResource, we will search these files after searching the
158 // main module. |path| should be the complete path to the pack file if known
159 // or just the pack file name otherwise (the delegate may optionally override
160 // this value). |scale_factor| is the scale of images in this resource pak
161 // relative to the images in the 1x resource pak. This method is not thread
162 // safe! You should call it immediately after calling InitSharedInstance.
163 void AddDataPackFromPath(const base::FilePath
& path
,
164 ScaleFactor scale_factor
);
166 // Same as above but using an already open file.
167 void AddDataPackFromFile(base::File file
, ScaleFactor scale_factor
);
169 // Same as AddDataPackFromPath but does not log an error if the pack fails to
171 void AddOptionalDataPackFromPath(const base::FilePath
& path
,
172 ScaleFactor scale_factor
);
174 // Changes the locale for an already-initialized ResourceBundle, returning the
175 // name of the newly-loaded locale. Future calls to get strings will return
176 // the strings for this new locale. This has no effect on existing or future
177 // image resources. |locale_resources_data_| is protected by a lock for the
178 // duration of the swap, as GetLocalizedString() may be concurrently invoked
179 // on another thread.
180 std::string
ReloadLocaleResources(const std::string
& pref_locale
);
182 // Gets image with the specified resource_id from the current module data.
183 // Returns a pointer to a shared instance of gfx::ImageSkia. This shared
184 // instance is owned by the resource bundle and should not be freed.
185 // TODO(pkotwicz): Make method return const gfx::ImageSkia*
187 // NOTE: GetNativeImageNamed is preferred for cross-platform gfx::Image use.
188 gfx::ImageSkia
* GetImageSkiaNamed(int resource_id
);
190 // Gets an image resource from the current module data. This will load the
191 // image in Skia format by default. The ResourceBundle owns this.
192 gfx::Image
& GetImageNamed(int resource_id
);
194 // Similar to GetImageNamed, but rather than loading the image in Skia format,
195 // it will load in the native platform type. This can avoid conversion from
196 // one image type to another. ResourceBundle owns the result.
198 // Note that if the same resource has already been loaded in GetImageNamed(),
199 // gfx::Image will perform a conversion, rather than using the native image
200 // loading code of ResourceBundle.
202 // If |rtl| is RTL_ENABLED then the image is flipped in RTL locales.
203 gfx::Image
& GetNativeImageNamed(int resource_id
, ImageRTL rtl
);
205 // Same as GetNativeImageNamed() except that RTL is not enabled.
206 gfx::Image
& GetNativeImageNamed(int resource_id
);
208 // Loads the raw bytes of a scale independent data resource.
209 base::RefCountedStaticMemory
* LoadDataResourceBytes(int resource_id
) const;
211 // Loads the raw bytes of a data resource nearest the scale factor
212 // |scale_factor| into |bytes|, without doing any processing or
213 // interpretation of the resource. Use ResourceHandle::SCALE_FACTOR_NONE
214 // for scale independent image resources (such as wallpaper).
215 // Returns NULL if we fail to read the resource.
216 base::RefCountedStaticMemory
* LoadDataResourceBytesForScale(
218 ScaleFactor scale_factor
) const;
220 // Return the contents of a scale independent resource in a
221 // StringPiece given the resource id
222 base::StringPiece
GetRawDataResource(int resource_id
) const;
224 // Return the contents of a resource in a StringPiece given the resource id
225 // nearest the scale factor |scale_factor|.
226 // Use ResourceHandle::SCALE_FACTOR_NONE for scale independent image resources
227 // (such as wallpaper).
228 base::StringPiece
GetRawDataResourceForScale(int resource_id
,
229 ScaleFactor scale_factor
) const;
231 // Get a localized string given a message id. Returns an empty
232 // string if the message_id is not found.
233 base::string16
GetLocalizedString(int message_id
);
235 // Returns the font list for the specified style.
236 const gfx::FontList
& GetFontList(FontStyle style
);
238 // Returns the font for the specified style.
239 const gfx::Font
& GetFont(FontStyle style
);
241 // Resets and reloads the cached fonts. This is useful when the fonts of the
242 // system have changed, for example, when the locale has changed.
245 // Overrides the path to the pak file from which the locale resources will be
246 // loaded. Pass an empty path to undo.
247 void OverrideLocalePakForTest(const base::FilePath
& pak_path
);
249 // Returns the full pathname of the locale file to load. May return an empty
250 // string if no locale data files are found and |test_file_exists| is true.
251 // Used on Android to load the local file in the browser process and pass it
252 // to the sandboxed renderer process.
253 base::FilePath
GetLocaleFilePath(const std::string
& app_locale
,
254 bool test_file_exists
);
256 // Returns the maximum scale factor currently loaded.
257 // Returns SCALE_FACTOR_100P if no resource is loaded.
258 ScaleFactor
GetMaxScaleFactor() const;
261 FRIEND_TEST_ALL_PREFIXES(ResourceBundleTest
, DelegateGetPathForLocalePack
);
262 FRIEND_TEST_ALL_PREFIXES(ResourceBundleTest
, DelegateGetImageNamed
);
263 FRIEND_TEST_ALL_PREFIXES(ResourceBundleTest
, DelegateGetNativeImageNamed
);
265 friend class ResourceBundleImageTest
;
266 friend class ResourceBundleTest
;
268 class ResourceBundleImageSource
;
269 friend class ResourceBundleImageSource
;
271 // Ctor/dtor are private, since we're a singleton.
272 explicit ResourceBundle(Delegate
* delegate
);
275 // Shared initialization.
276 static void InitSharedInstance(Delegate
* delegate
);
278 // Free skia_images_.
281 // Load the main resources.
282 void LoadCommonResources();
284 // Implementation for AddDataPackFromPath and AddOptionalDataPackFromPath, if
285 // the pack is not |optional| logs an error on failure to load.
286 void AddDataPackFromPathInternal(const base::FilePath
& path
,
287 ScaleFactor scale_factor
,
290 // Inserts |data_pack| to |data_pack_| and updates |max_scale_factor_|
292 void AddDataPack(DataPack
* data_pack
);
294 // Try to load the locale specific strings from an external data module.
295 // Returns the locale that is loaded.
296 std::string
LoadLocaleResources(const std::string
& pref_locale
);
298 // Load test resources in given paths. If either path is empty an empty
299 // resource pack is loaded.
300 void LoadTestResources(const base::FilePath
& path
,
301 const base::FilePath
& locale_path
);
303 // Unload the locale specific strings and prepares to load new ones. See
304 // comments for ReloadLocaleResources().
305 void UnloadLocaleResources();
307 // Initializes all the gfx::FontList members if they haven't yet been
309 void LoadFontsIfNecessary();
311 // Returns a FontList or NULL by calling Delegate::GetFont and converting
312 // scoped_ptr<gfx::Font> to scoped_ptr<gfx::FontList>.
313 scoped_ptr
<gfx::FontList
> GetFontListFromDelegate(FontStyle style
);
315 // Fills the |bitmap| given the data file to look in and the |resource_id|.
316 // Returns false if the resource does not exist.
318 // If the call succeeds, |fell_back_to_1x| indicates whether Chrome's custom
319 // csCl PNG chunk is present (added by GRIT if it falls back to a 100% image).
320 bool LoadBitmap(const ResourceHandle
& data_handle
,
323 bool* fell_back_to_1x
) const;
325 // Fills the |bitmap| given the |resource_id| and |scale_factor|.
326 // Returns false if the resource does not exist. This may fall back to
327 // the data pack with SCALE_FACTOR_NONE, and when this happens,
328 // |scale_factor| will be set to SCALE_FACTOR_100P.
329 bool LoadBitmap(int resource_id
,
330 ScaleFactor
* scale_factor
,
332 bool* fell_back_to_1x
) const;
334 // Returns true if missing scaled resources should be visually indicated when
335 // drawing the fallback (e.g., by tinting the image).
336 static bool ShouldHighlightMissingScaledResources();
338 // Returns true if the data in |buf| is a PNG that has the special marker
339 // added by GRIT that indicates that the image is actually 1x data.
340 static bool PNGContainsFallbackMarker(const unsigned char* buf
, size_t size
);
342 // A wrapper for PNGCodec::Decode that returns information about custom
343 // chunks. For security reasons we can't alter PNGCodec to return this
344 // information. Our PNG files are preprocessed by GRIT, and any special chunks
345 // should occur immediately after the IHDR chunk.
346 static bool DecodePNG(const unsigned char* buf
,
349 bool* fell_back_to_1x
);
351 // Returns an empty image for when a resource cannot be loaded. This is a
352 // bright red bitmap.
353 gfx::Image
& GetEmptyImage();
355 const base::FilePath
& GetOverriddenPakPath();
357 // This pointer is guaranteed to outlive the ResourceBundle instance and may
361 // Protects |images_| and font-related members.
362 scoped_ptr
<base::Lock
> images_and_fonts_lock_
;
364 // Protects |locale_resources_data_|.
365 scoped_ptr
<base::Lock
> locale_resources_data_lock_
;
367 // Handles for data sources.
368 scoped_ptr
<ResourceHandle
> locale_resources_data_
;
369 ScopedVector
<ResourceHandle
> data_packs_
;
371 // The maximum scale factor currently loaded.
372 ScaleFactor max_scale_factor_
;
374 // Cached images. The ResourceBundle caches all retrieved images and keeps
375 // ownership of the pointers.
376 typedef std::map
<int, gfx::Image
> ImageMap
;
379 gfx::Image empty_image_
;
381 // The various font lists used. Cached to avoid repeated GDI
382 // creation/destruction.
383 scoped_ptr
<gfx::FontList
> base_font_list_
;
384 scoped_ptr
<gfx::FontList
> bold_font_list_
;
385 scoped_ptr
<gfx::FontList
> small_font_list_
;
386 scoped_ptr
<gfx::FontList
> small_bold_font_list_
;
387 scoped_ptr
<gfx::FontList
> medium_font_list_
;
388 scoped_ptr
<gfx::FontList
> medium_bold_font_list_
;
389 scoped_ptr
<gfx::FontList
> large_font_list_
;
390 scoped_ptr
<gfx::FontList
> large_bold_font_list_
;
391 scoped_ptr
<gfx::FontList
> web_font_list_
;
393 base::FilePath overridden_pak_path_
;
395 DISALLOW_COPY_AND_ASSIGN(ResourceBundle
);
400 // TODO(beng): Someday, maybe, get rid of this.
401 using ui::ResourceBundle
;
403 #endif // UI_BASE_RESOURCE_RESOURCE_BUNDLE_H_