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. 4 5 #ifndef UI_BASE_RESOURCE_RESOURCE_BUNDLE_H_ 6 #define UI_BASE_RESOURCE_RESOURCE_BUNDLE_H_ 7 8 #include <map> 9 #include <string> 10 11 #include "base/basictypes.h" 12 #include "base/files/file_path.h" 13 #include "base/files/memory_mapped_file.h" 14 #include "base/gtest_prod_util.h" 15 #include "base/memory/scoped_ptr.h" 16 #include "base/memory/scoped_vector.h" 17 #include "base/strings/string16.h" 18 #include "base/strings/string_piece.h" 19 #include "build/build_config.h" 20 #include "ui/base/layout.h" 21 #include "ui/base/ui_base_export.h" 22 #include "ui/gfx/font_list.h" 23 #include "ui/gfx/image/image.h" 24 #include "ui/gfx/native_widget_types.h" 25 26 class SkBitmap; 27 28 namespace base { 29 class File; 30 class Lock; 31 class RefCountedStaticMemory; 32 } 33 34 namespace ui { 35 36 class DataPack; 37 class ResourceHandle; 38 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 { 42 public: 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. 46 enum FontStyle { 47 // NOTE: depending upon the locale, using one of the *BoldFont below 48 // may *not* actually result in a bold font. 49 SmallFont, 50 SmallBoldFont, 51 BaseFont, 52 BoldFont, 53 MediumFont, 54 MediumBoldFont, 55 LargeFont, 56 LargeBoldFont, 57 }; 58 59 enum ImageRTL { 60 // Images are flipped in RTL locales. 61 RTL_ENABLED, 62 // Images are never flipped. 63 RTL_DISABLED, 64 }; 65 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. 68 class Delegate { 69 public: 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; 77 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; 85 86 // Return an image resource or an empty value to attempt retrieval of the 87 // default resource. 88 virtual gfx::Image GetImageNamed(int resource_id) = 0; 89 90 // Return an image resource or an empty value to attempt retrieval of the 91 // default resource. 92 virtual gfx::Image GetNativeImageNamed(int resource_id, ImageRTL rtl) = 0; 93 94 // Return a static memory resource or NULL to attempt retrieval of the 95 // default resource. 96 virtual base::RefCountedStaticMemory* LoadDataResourceBytes( 97 int resource_id, 98 ScaleFactor scale_factor) = 0; 99 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; 105 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; 109 110 // Returns a font or NULL to attempt retrieval of the default resource. 111 virtual scoped_ptr<gfx::Font> GetFont(FontStyle style) = 0; 112 113 protected: ~Delegate()114 virtual ~Delegate() {} 115 }; 116 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). 121 // 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); 128 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); 133 134 // Initialize the ResourceBundle using the given file region. If |region| is 135 // MemoryMappedFile::Region::kWholeFile, the entire |pak_file| is used. 136 // |should_load_common_resources| controls whether or not LoadCommonResources 137 // is called. 138 // This allows the use of this function in a sandbox without local file 139 // access (as on Android). 140 static void InitSharedInstanceWithPakFileRegion( 141 base::File pak_file, 142 const base::MemoryMappedFile::Region& region, 143 bool should_load_common_resources); 144 145 // Initialize the ResourceBundle using given data pack path for testing. 146 static void InitSharedInstanceWithPakPath(const base::FilePath& path); 147 148 // Delete the ResourceBundle for this process if it exists. 149 static void CleanupSharedInstance(); 150 151 // Returns true after the global resource loader instance has been created. 152 static bool HasSharedInstance(); 153 154 // Return the global resource loader instance. 155 static ResourceBundle& GetSharedInstance(); 156 157 // Check if the .pak for the given locale exists. 158 bool LocaleDataPakExists(const std::string& locale); 159 160 // Registers additional data pack files with this ResourceBundle. When 161 // looking for a DataResource, we will search these files after searching the 162 // main module. |path| should be the complete path to the pack file if known 163 // or just the pack file name otherwise (the delegate may optionally override 164 // this value). |scale_factor| is the scale of images in this resource pak 165 // relative to the images in the 1x resource pak. This method is not thread 166 // safe! You should call it immediately after calling InitSharedInstance. 167 void AddDataPackFromPath(const base::FilePath& path, 168 ScaleFactor scale_factor); 169 170 // Same as above but using an already open file. 171 void AddDataPackFromFile(base::File file, ScaleFactor scale_factor); 172 173 // Same as above but using only a region (offset + size) of the file. 174 void AddDataPackFromFileRegion(base::File file, 175 const base::MemoryMappedFile::Region& region, 176 ScaleFactor scale_factor); 177 178 // Same as AddDataPackFromPath but does not log an error if the pack fails to 179 // load. 180 void AddOptionalDataPackFromPath(const base::FilePath& path, 181 ScaleFactor scale_factor); 182 183 // Changes the locale for an already-initialized ResourceBundle, returning the 184 // name of the newly-loaded locale. Future calls to get strings will return 185 // the strings for this new locale. This has no effect on existing or future 186 // image resources. |locale_resources_data_| is protected by a lock for the 187 // duration of the swap, as GetLocalizedString() may be concurrently invoked 188 // on another thread. 189 std::string ReloadLocaleResources(const std::string& pref_locale); 190 191 // Gets image with the specified resource_id from the current module data. 192 // Returns a pointer to a shared instance of gfx::ImageSkia. This shared 193 // instance is owned by the resource bundle and should not be freed. 194 // TODO(pkotwicz): Make method return const gfx::ImageSkia* 195 // 196 // NOTE: GetNativeImageNamed is preferred for cross-platform gfx::Image use. 197 gfx::ImageSkia* GetImageSkiaNamed(int resource_id); 198 199 // Gets an image resource from the current module data. This will load the 200 // image in Skia format by default. The ResourceBundle owns this. 201 gfx::Image& GetImageNamed(int resource_id); 202 203 // Similar to GetImageNamed, but rather than loading the image in Skia format, 204 // it will load in the native platform type. This can avoid conversion from 205 // one image type to another. ResourceBundle owns the result. 206 // 207 // Note that if the same resource has already been loaded in GetImageNamed(), 208 // gfx::Image will perform a conversion, rather than using the native image 209 // loading code of ResourceBundle. 210 // 211 // If |rtl| is RTL_ENABLED then the image is flipped in RTL locales. 212 gfx::Image& GetNativeImageNamed(int resource_id, ImageRTL rtl); 213 214 // Same as GetNativeImageNamed() except that RTL is not enabled. 215 gfx::Image& GetNativeImageNamed(int resource_id); 216 217 // Loads the raw bytes of a scale independent data resource. 218 base::RefCountedStaticMemory* LoadDataResourceBytes(int resource_id) const; 219 220 // Loads the raw bytes of a data resource nearest the scale factor 221 // |scale_factor| into |bytes|, without doing any processing or 222 // interpretation of the resource. Use ResourceHandle::SCALE_FACTOR_NONE 223 // for scale independent image resources (such as wallpaper). 224 // Returns NULL if we fail to read the resource. 225 base::RefCountedStaticMemory* LoadDataResourceBytesForScale( 226 int resource_id, 227 ScaleFactor scale_factor) const; 228 229 // Return the contents of a scale independent resource in a 230 // StringPiece given the resource id 231 base::StringPiece GetRawDataResource(int resource_id) const; 232 233 // Return the contents of a resource in a StringPiece given the resource id 234 // nearest the scale factor |scale_factor|. 235 // Use ResourceHandle::SCALE_FACTOR_NONE for scale independent image resources 236 // (such as wallpaper). 237 base::StringPiece GetRawDataResourceForScale(int resource_id, 238 ScaleFactor scale_factor) const; 239 240 // Get a localized string given a message id. Returns an empty 241 // string if the message_id is not found. 242 base::string16 GetLocalizedString(int message_id); 243 244 // Returns the font list for the specified style. 245 const gfx::FontList& GetFontList(FontStyle style); 246 247 // Returns the font for the specified style. 248 const gfx::Font& GetFont(FontStyle style); 249 250 // Resets and reloads the cached fonts. This is useful when the fonts of the 251 // system have changed, for example, when the locale has changed. 252 void ReloadFonts(); 253 254 // Overrides the path to the pak file from which the locale resources will be 255 // loaded. Pass an empty path to undo. 256 void OverrideLocalePakForTest(const base::FilePath& pak_path); 257 258 // Returns the full pathname of the locale file to load. May return an empty 259 // string if no locale data files are found and |test_file_exists| is true. 260 // Used on Android to load the local file in the browser process and pass it 261 // to the sandboxed renderer process. 262 base::FilePath GetLocaleFilePath(const std::string& app_locale, 263 bool test_file_exists); 264 265 // Returns the maximum scale factor currently loaded. 266 // Returns SCALE_FACTOR_100P if no resource is loaded. 267 ScaleFactor GetMaxScaleFactor() const; 268 269 protected: 270 // Returns true if |scale_factor| is supported by this platform. 271 static bool IsScaleFactorSupported(ScaleFactor scale_factor); 272 273 private: 274 FRIEND_TEST_ALL_PREFIXES(ResourceBundleTest, DelegateGetPathForLocalePack); 275 FRIEND_TEST_ALL_PREFIXES(ResourceBundleTest, DelegateGetImageNamed); 276 FRIEND_TEST_ALL_PREFIXES(ResourceBundleTest, DelegateGetNativeImageNamed); 277 278 friend class ResourceBundleImageTest; 279 friend class ResourceBundleTest; 280 281 class ResourceBundleImageSource; 282 friend class ResourceBundleImageSource; 283 284 // Ctor/dtor are private, since we're a singleton. 285 explicit ResourceBundle(Delegate* delegate); 286 ~ResourceBundle(); 287 288 // Shared initialization. 289 static void InitSharedInstance(Delegate* delegate); 290 291 // Free skia_images_. 292 void FreeImages(); 293 294 // Load the main resources. 295 void LoadCommonResources(); 296 297 // Implementation for AddDataPackFromPath and AddOptionalDataPackFromPath, if 298 // the pack is not |optional| logs an error on failure to load. 299 void AddDataPackFromPathInternal(const base::FilePath& path, 300 ScaleFactor scale_factor, 301 bool optional); 302 303 // Inserts |data_pack| to |data_pack_| and updates |max_scale_factor_| 304 // accordingly. 305 void AddDataPack(DataPack* data_pack); 306 307 // Try to load the locale specific strings from an external data module. 308 // Returns the locale that is loaded. 309 std::string LoadLocaleResources(const std::string& pref_locale); 310 311 // Load test resources in given paths. If either path is empty an empty 312 // resource pack is loaded. 313 void LoadTestResources(const base::FilePath& path, 314 const base::FilePath& locale_path); 315 316 // Unload the locale specific strings and prepares to load new ones. See 317 // comments for ReloadLocaleResources(). 318 void UnloadLocaleResources(); 319 320 // Initializes all the gfx::FontList members if they haven't yet been 321 // initialized. 322 void LoadFontsIfNecessary(); 323 324 // Returns a FontList or NULL by calling Delegate::GetFont and converting 325 // scoped_ptr<gfx::Font> to scoped_ptr<gfx::FontList>. 326 scoped_ptr<gfx::FontList> GetFontListFromDelegate(FontStyle style); 327 328 // Fills the |bitmap| given the data file to look in and the |resource_id|. 329 // Returns false if the resource does not exist. 330 // 331 // If the call succeeds, |fell_back_to_1x| indicates whether Chrome's custom 332 // csCl PNG chunk is present (added by GRIT if it falls back to a 100% image). 333 bool LoadBitmap(const ResourceHandle& data_handle, 334 int resource_id, 335 SkBitmap* bitmap, 336 bool* fell_back_to_1x) const; 337 338 // Fills the |bitmap| given the |resource_id| and |scale_factor|. 339 // Returns false if the resource does not exist. This may fall back to 340 // the data pack with SCALE_FACTOR_NONE, and when this happens, 341 // |scale_factor| will be set to SCALE_FACTOR_100P. 342 bool LoadBitmap(int resource_id, 343 ScaleFactor* scale_factor, 344 SkBitmap* bitmap, 345 bool* fell_back_to_1x) const; 346 347 // Returns true if missing scaled resources should be visually indicated when 348 // drawing the fallback (e.g., by tinting the image). 349 static bool ShouldHighlightMissingScaledResources(); 350 351 // Returns true if the data in |buf| is a PNG that has the special marker 352 // added by GRIT that indicates that the image is actually 1x data. 353 static bool PNGContainsFallbackMarker(const unsigned char* buf, size_t size); 354 355 // A wrapper for PNGCodec::Decode that returns information about custom 356 // chunks. For security reasons we can't alter PNGCodec to return this 357 // information. Our PNG files are preprocessed by GRIT, and any special chunks 358 // should occur immediately after the IHDR chunk. 359 static bool DecodePNG(const unsigned char* buf, 360 size_t size, 361 SkBitmap* bitmap, 362 bool* fell_back_to_1x); 363 364 // Returns an empty image for when a resource cannot be loaded. This is a 365 // bright red bitmap. 366 gfx::Image& GetEmptyImage(); 367 368 const base::FilePath& GetOverriddenPakPath(); 369 370 // This pointer is guaranteed to outlive the ResourceBundle instance and may 371 // be NULL. 372 Delegate* delegate_; 373 374 // Protects |images_| and font-related members. 375 scoped_ptr<base::Lock> images_and_fonts_lock_; 376 377 // Protects |locale_resources_data_|. 378 scoped_ptr<base::Lock> locale_resources_data_lock_; 379 380 // Handles for data sources. 381 scoped_ptr<ResourceHandle> locale_resources_data_; 382 ScopedVector<ResourceHandle> data_packs_; 383 384 // The maximum scale factor currently loaded. 385 ScaleFactor max_scale_factor_; 386 387 // Cached images. The ResourceBundle caches all retrieved images and keeps 388 // ownership of the pointers. 389 typedef std::map<int, gfx::Image> ImageMap; 390 ImageMap images_; 391 392 gfx::Image empty_image_; 393 394 // The various font lists used. Cached to avoid repeated GDI 395 // creation/destruction. 396 scoped_ptr<gfx::FontList> base_font_list_; 397 scoped_ptr<gfx::FontList> bold_font_list_; 398 scoped_ptr<gfx::FontList> small_font_list_; 399 scoped_ptr<gfx::FontList> small_bold_font_list_; 400 scoped_ptr<gfx::FontList> medium_font_list_; 401 scoped_ptr<gfx::FontList> medium_bold_font_list_; 402 scoped_ptr<gfx::FontList> large_font_list_; 403 scoped_ptr<gfx::FontList> large_bold_font_list_; 404 scoped_ptr<gfx::FontList> web_font_list_; 405 406 base::FilePath overridden_pak_path_; 407 408 DISALLOW_COPY_AND_ASSIGN(ResourceBundle); 409 }; 410 411 } // namespace ui 412 413 // TODO(beng): Someday, maybe, get rid of this. 414 using ui::ResourceBundle; 415 416 #endif // UI_BASE_RESOURCE_RESOURCE_BUNDLE_H_ 417