Durable Storage: Refactor browser test and test the basic "deny" flow.
[chromium-blink-merge.git] / ui / base / resource / resource_bundle.h
blobb7533ed6517b34b506d0e3833a175a7127d3d8b3
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 <map>
9 #include <string>
11 #include "base/basictypes.h"
12 #include "base/containers/hash_tables.h"
13 #include "base/files/file_path.h"
14 #include "base/files/memory_mapped_file.h"
15 #include "base/gtest_prod_util.h"
16 #include "base/memory/scoped_ptr.h"
17 #include "base/memory/scoped_vector.h"
18 #include "base/strings/string16.h"
19 #include "base/strings/string_piece.h"
20 #include "build/build_config.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"
27 class SkBitmap;
29 namespace base {
30 class File;
31 class Lock;
32 class RefCountedStaticMemory;
35 namespace ui {
37 class DataPack;
38 class ResourceHandle;
40 // ResourceBundle is a central facility to load images and other resources,
41 // such as theme graphics. Every resource is loaded only once.
42 class UI_BASE_EXPORT ResourceBundle {
43 public:
44 // An enumeration of the various font styles used throughout Chrome.
45 // The following holds true for the font sizes:
46 // Small <= SmallBold <= Base <= Bold <= Medium <= MediumBold <= Large.
47 enum FontStyle {
48 // NOTE: depending upon the locale, using one of the *BoldFont below
49 // may *not* actually result in a bold font.
50 SmallFont,
51 SmallBoldFont,
52 BaseFont,
53 BoldFont,
54 MediumFont,
55 MediumBoldFont,
56 LargeFont,
57 LargeBoldFont,
60 enum ImageRTL {
61 // Images are flipped in RTL locales.
62 RTL_ENABLED,
63 // Images are never flipped.
64 RTL_DISABLED,
67 enum LoadResources {
68 LOAD_COMMON_RESOURCES,
69 DO_NOT_LOAD_COMMON_RESOURCES
72 // Delegate class that allows interception of pack file loading and resource
73 // requests. The methods of this class may be called on multiple threads.
74 class Delegate {
75 public:
76 // Called before a resource pack file is loaded. Return the full path for
77 // the pack file to continue loading or an empty value to cancel loading.
78 // |pack_path| will contain the complete default path for the pack file if
79 // known or just the pack file name otherwise.
80 virtual base::FilePath GetPathForResourcePack(
81 const base::FilePath& pack_path,
82 ScaleFactor scale_factor) = 0;
84 // Called before a locale pack file is loaded. Return the full path for
85 // the pack file to continue loading or an empty value to cancel loading.
86 // |pack_path| will contain the complete default path for the pack file if
87 // known or just the pack file name otherwise.
88 virtual base::FilePath GetPathForLocalePack(
89 const base::FilePath& pack_path,
90 const std::string& locale) = 0;
92 // Return an image resource or an empty value to attempt retrieval of the
93 // default resource.
94 virtual gfx::Image GetImageNamed(int resource_id) = 0;
96 // Return an image resource or an empty value to attempt retrieval of the
97 // default resource.
98 virtual gfx::Image GetNativeImageNamed(int resource_id, ImageRTL rtl) = 0;
100 // Return a static memory resource or NULL to attempt retrieval of the
101 // default resource.
102 virtual base::RefCountedStaticMemory* LoadDataResourceBytes(
103 int resource_id,
104 ScaleFactor scale_factor) = 0;
106 // Retrieve a raw data resource. Return true if a resource was provided or
107 // false to attempt retrieval of the default resource.
108 virtual bool GetRawDataResource(int resource_id,
109 ScaleFactor scale_factor,
110 base::StringPiece* value) = 0;
112 // Retrieve a localized string. Return true if a string was provided or
113 // false to attempt retrieval of the default string.
114 virtual bool GetLocalizedString(int message_id, base::string16* value) = 0;
116 // Returns a font or NULL to attempt retrieval of the default resource.
117 virtual scoped_ptr<gfx::Font> GetFont(FontStyle style) = 0;
119 protected:
120 virtual ~Delegate() {}
123 // Initialize the ResourceBundle for this process. Does not take ownership of
124 // the |delegate| value. Returns the language selected.
125 // NOTE: Mac ignores this and always loads up resources for the language
126 // defined by the Cocoa UI (i.e., NSBundle does the language work).
128 // TODO(sergeyu): This method also loads common resources (i.e. chrome.pak).
129 // There is no way to specify which resource files are loaded, i.e. names of
130 // the files are hardcoded in ResourceBundle. Fix it to allow to specify which
131 // files are loaded (e.g. add a new method in Delegate).
132 // |load_resources| controls whether or not LoadCommonResources is called.
133 static std::string InitSharedInstanceWithLocale(
134 const std::string& pref_locale,
135 Delegate* delegate,
136 LoadResources load_resources);
138 // Initialize the ResourceBundle using the given file region. If |region| is
139 // MemoryMappedFile::Region::kWholeFile, the entire |pak_file| is used.
140 // This allows the use of this function in a sandbox without local file
141 // access (as on Android).
142 static void InitSharedInstanceWithPakFileRegion(
143 base::File pak_file,
144 const base::MemoryMappedFile::Region& region);
146 // Initialize the ResourceBundle using given data pack path for testing.
147 static void InitSharedInstanceWithPakPath(const base::FilePath& path);
149 // Delete the ResourceBundle for this process if it exists.
150 static void CleanupSharedInstance();
152 // Returns true after the global resource loader instance has been created.
153 static bool HasSharedInstance();
155 // Return the global resource loader instance.
156 static ResourceBundle& GetSharedInstance();
158 // Check if the .pak for the given locale exists.
159 bool LocaleDataPakExists(const std::string& locale);
161 // Registers additional data pack files with this ResourceBundle. When
162 // looking for a DataResource, we will search these files after searching the
163 // main module. |path| should be the complete path to the pack file if known
164 // or just the pack file name otherwise (the delegate may optionally override
165 // this value). |scale_factor| is the scale of images in this resource pak
166 // relative to the images in the 1x resource pak. This method is not thread
167 // safe! You should call it immediately after calling InitSharedInstance.
168 void AddDataPackFromPath(const base::FilePath& path,
169 ScaleFactor scale_factor);
171 // Same as above but using an already open file.
172 void AddDataPackFromFile(base::File file, ScaleFactor scale_factor);
174 // Same as above but using only a region (offset + size) of the file.
175 void AddDataPackFromFileRegion(base::File file,
176 const base::MemoryMappedFile::Region& region,
177 ScaleFactor scale_factor);
179 // Same as AddDataPackFromPath but does not log an error if the pack fails to
180 // load.
181 void AddOptionalDataPackFromPath(const base::FilePath& path,
182 ScaleFactor scale_factor);
184 // The same as AddDataPackFromPath() and AddOptionalDataPackFromPath(),
185 // except the data pack is flagged as containing only material design assets.
186 // TODO(tdanderson): These methods are temporary and should be removed after
187 // the transition to material design in the browser UI.
188 void AddMaterialDesignDataPackFromPath(const base::FilePath& path,
189 ScaleFactor scale_factor);
190 void AddOptionalMaterialDesignDataPackFromPath(const base::FilePath& path,
191 ScaleFactor scale_factor);
193 // Changes the locale for an already-initialized ResourceBundle, returning the
194 // name of the newly-loaded locale. Future calls to get strings will return
195 // the strings for this new locale. This has no effect on existing or future
196 // image resources. |locale_resources_data_| is protected by a lock for the
197 // duration of the swap, as GetLocalizedString() may be concurrently invoked
198 // on another thread.
199 std::string ReloadLocaleResources(const std::string& pref_locale);
201 // Gets image with the specified resource_id from the current module data.
202 // Returns a pointer to a shared instance of gfx::ImageSkia. This shared
203 // instance is owned by the resource bundle and should not be freed.
204 // TODO(pkotwicz): Make method return const gfx::ImageSkia*
206 // NOTE: GetNativeImageNamed is preferred for cross-platform gfx::Image use.
207 gfx::ImageSkia* GetImageSkiaNamed(int resource_id);
209 // Gets an image resource from the current module data. This will load the
210 // image in Skia format by default. The ResourceBundle owns this.
211 gfx::Image& GetImageNamed(int resource_id);
213 // Similar to GetImageNamed, but rather than loading the image in Skia format,
214 // it will load in the native platform type. This can avoid conversion from
215 // one image type to another. ResourceBundle owns the result.
217 // Note that if the same resource has already been loaded in GetImageNamed(),
218 // gfx::Image will perform a conversion, rather than using the native image
219 // loading code of ResourceBundle.
221 // If |rtl| is RTL_ENABLED then the image is flipped in RTL locales.
222 gfx::Image& GetNativeImageNamed(int resource_id, ImageRTL rtl);
224 // Same as GetNativeImageNamed() except that RTL is not enabled.
225 gfx::Image& GetNativeImageNamed(int resource_id);
227 // Loads the raw bytes of a scale independent data resource.
228 base::RefCountedStaticMemory* LoadDataResourceBytes(int resource_id) const;
230 // Loads the raw bytes of a data resource nearest the scale factor
231 // |scale_factor| into |bytes|, without doing any processing or
232 // interpretation of the resource. Use ResourceHandle::SCALE_FACTOR_NONE
233 // for scale independent image resources (such as wallpaper).
234 // Returns NULL if we fail to read the resource.
235 base::RefCountedStaticMemory* LoadDataResourceBytesForScale(
236 int resource_id,
237 ScaleFactor scale_factor) const;
239 // Return the contents of a scale independent resource in a
240 // StringPiece given the resource id
241 base::StringPiece GetRawDataResource(int resource_id) const;
243 // Return the contents of a resource in a StringPiece given the resource id
244 // nearest the scale factor |scale_factor|.
245 // Use ResourceHandle::SCALE_FACTOR_NONE for scale independent image resources
246 // (such as wallpaper).
247 base::StringPiece GetRawDataResourceForScale(int resource_id,
248 ScaleFactor scale_factor) const;
250 // Get a localized string given a message id. Returns an empty
251 // string if the message_id is not found.
252 base::string16 GetLocalizedString(int message_id);
254 // Returns the font list for the specified style.
255 const gfx::FontList& GetFontList(FontStyle style);
257 // Returns the font for the specified style.
258 const gfx::Font& GetFont(FontStyle style);
260 // Resets and reloads the cached fonts. This is useful when the fonts of the
261 // system have changed, for example, when the locale has changed.
262 void ReloadFonts();
264 // Overrides the path to the pak file from which the locale resources will be
265 // loaded. Pass an empty path to undo.
266 void OverrideLocalePakForTest(const base::FilePath& pak_path);
268 // Overrides a localized string resource with the given string. If no delegate
269 // is present, the |string| will be returned when getting the localized string
270 // |message_id|. If |ReloadLocaleResources| is called, all overrides are
271 // cleared. This is intended to be used in conjunction with field trials and
272 // the variations service to experiment with different UI strings. This method
273 // is not thread safe!
274 void OverrideLocaleStringResource(int message_id,
275 const base::string16& string);
277 // Returns the full pathname of the locale file to load. May return an empty
278 // string if no locale data files are found and |test_file_exists| is true.
279 // Used on Android to load the local file in the browser process and pass it
280 // to the sandboxed renderer process.
281 base::FilePath GetLocaleFilePath(const std::string& app_locale,
282 bool test_file_exists);
284 // Returns the maximum scale factor currently loaded.
285 // Returns SCALE_FACTOR_100P if no resource is loaded.
286 ScaleFactor GetMaxScaleFactor() const;
288 protected:
289 // Returns true if |scale_factor| is supported by this platform.
290 static bool IsScaleFactorSupported(ScaleFactor scale_factor);
292 private:
293 FRIEND_TEST_ALL_PREFIXES(ResourceBundleTest, DelegateGetPathForLocalePack);
294 FRIEND_TEST_ALL_PREFIXES(ResourceBundleTest, DelegateGetImageNamed);
295 FRIEND_TEST_ALL_PREFIXES(ResourceBundleTest, DelegateGetNativeImageNamed);
296 FRIEND_TEST_ALL_PREFIXES(ResourceBundleImageTest,
297 CountMaterialDesignDataPacksInResourceBundle);
299 friend class ResourceBundleImageTest;
300 friend class ResourceBundleTest;
302 class ResourceBundleImageSource;
303 friend class ResourceBundleImageSource;
305 typedef base::hash_map<int, base::string16> IdToStringMap;
307 // Ctor/dtor are private, since we're a singleton.
308 explicit ResourceBundle(Delegate* delegate);
309 ~ResourceBundle();
311 // Shared initialization.
312 static void InitSharedInstance(Delegate* delegate);
314 // Free skia_images_.
315 void FreeImages();
317 // Load the main resources.
318 void LoadCommonResources();
320 // Implementation for the public methods which add a DataPack from a path. If
321 // |optional| is false, an error is logged on failure to load. Sets the
322 // member |has_only_material_design_assets_| on the created DataPack to the
323 // value of |has_only_material_assets|.
324 void AddDataPackFromPathInternal(const base::FilePath& path,
325 ScaleFactor scale_factor,
326 bool optional,
327 bool has_only_material_assets);
329 // Inserts |data_pack| to |data_pack_| and updates |max_scale_factor_|
330 // accordingly.
331 void AddDataPack(DataPack* data_pack);
333 // Try to load the locale specific strings from an external data module.
334 // Returns the locale that is loaded.
335 std::string LoadLocaleResources(const std::string& pref_locale);
337 // Load test resources in given paths. If either path is empty an empty
338 // resource pack is loaded.
339 void LoadTestResources(const base::FilePath& path,
340 const base::FilePath& locale_path);
342 // Unload the locale specific strings and prepares to load new ones. See
343 // comments for ReloadLocaleResources().
344 void UnloadLocaleResources();
346 // Initializes the font description of default gfx::FontList.
347 void InitDefaultFontList();
349 // Initializes all the gfx::FontList members if they haven't yet been
350 // initialized.
351 void LoadFontsIfNecessary();
353 // Returns a FontList or NULL by calling Delegate::GetFont and converting
354 // scoped_ptr<gfx::Font> to scoped_ptr<gfx::FontList>.
355 scoped_ptr<gfx::FontList> GetFontListFromDelegate(FontStyle style);
357 // Fills the |bitmap| given the data file to look in and the |resource_id|.
358 // Returns false if the resource does not exist.
360 // If the call succeeds, |fell_back_to_1x| indicates whether Chrome's custom
361 // csCl PNG chunk is present (added by GRIT if it falls back to a 100% image).
362 bool LoadBitmap(const ResourceHandle& data_handle,
363 int resource_id,
364 SkBitmap* bitmap,
365 bool* fell_back_to_1x) const;
367 // Fills the |bitmap| given the |resource_id| and |scale_factor|.
368 // Returns false if the resource does not exist. This may fall back to
369 // the data pack with SCALE_FACTOR_NONE, and when this happens,
370 // |scale_factor| will be set to SCALE_FACTOR_100P.
371 bool LoadBitmap(int resource_id,
372 ScaleFactor* scale_factor,
373 SkBitmap* bitmap,
374 bool* fell_back_to_1x) const;
376 // Returns true if missing scaled resources should be visually indicated when
377 // drawing the fallback (e.g., by tinting the image).
378 static bool ShouldHighlightMissingScaledResources();
380 // Returns true if the data in |buf| is a PNG that has the special marker
381 // added by GRIT that indicates that the image is actually 1x data.
382 static bool PNGContainsFallbackMarker(const unsigned char* buf, size_t size);
384 // A wrapper for PNGCodec::Decode that returns information about custom
385 // chunks. For security reasons we can't alter PNGCodec to return this
386 // information. Our PNG files are preprocessed by GRIT, and any special chunks
387 // should occur immediately after the IHDR chunk.
388 static bool DecodePNG(const unsigned char* buf,
389 size_t size,
390 SkBitmap* bitmap,
391 bool* fell_back_to_1x);
393 // Returns an empty image for when a resource cannot be loaded. This is a
394 // bright red bitmap.
395 gfx::Image& GetEmptyImage();
397 const base::FilePath& GetOverriddenPakPath();
399 // This pointer is guaranteed to outlive the ResourceBundle instance and may
400 // be NULL.
401 Delegate* delegate_;
403 // Protects |images_| and font-related members.
404 scoped_ptr<base::Lock> images_and_fonts_lock_;
406 // Protects |locale_resources_data_|.
407 scoped_ptr<base::Lock> locale_resources_data_lock_;
409 // Handles for data sources.
410 scoped_ptr<ResourceHandle> locale_resources_data_;
411 ScopedVector<ResourceHandle> data_packs_;
413 // The maximum scale factor currently loaded.
414 ScaleFactor max_scale_factor_;
416 // Cached images. The ResourceBundle caches all retrieved images and keeps
417 // ownership of the pointers.
418 typedef std::map<int, gfx::Image> ImageMap;
419 ImageMap images_;
421 gfx::Image empty_image_;
423 // The various font lists used. Cached to avoid repeated GDI
424 // creation/destruction.
425 scoped_ptr<gfx::FontList> base_font_list_;
426 scoped_ptr<gfx::FontList> bold_font_list_;
427 scoped_ptr<gfx::FontList> small_font_list_;
428 scoped_ptr<gfx::FontList> small_bold_font_list_;
429 scoped_ptr<gfx::FontList> medium_font_list_;
430 scoped_ptr<gfx::FontList> medium_bold_font_list_;
431 scoped_ptr<gfx::FontList> large_font_list_;
432 scoped_ptr<gfx::FontList> large_bold_font_list_;
433 scoped_ptr<gfx::FontList> web_font_list_;
435 base::FilePath overridden_pak_path_;
437 IdToStringMap overridden_locale_strings_;
439 DISALLOW_COPY_AND_ASSIGN(ResourceBundle);
442 } // namespace ui
444 // TODO(beng): Someday, maybe, get rid of this.
445 using ui::ResourceBundle;
447 #endif // UI_BASE_RESOURCE_RESOURCE_BUNDLE_H_