Home | History | Annotate | Download | only in resource
      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 "build/build_config.h"
      9 
     10 #include <map>
     11 #include <string>
     12 
     13 #include "base/basictypes.h"
     14 #include "base/files/file_path.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/platform_file.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_export.h"
     23 #include "ui/gfx/font_list.h"
     24 #include "ui/gfx/image/image.h"
     25 #include "ui/gfx/native_widget_types.h"
     26 
     27 class SkBitmap;
     28 
     29 namespace base {
     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_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, 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:
    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 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(
    139       base::PlatformFile file, bool should_load_common_resources);
    140 
    141   // Initialize the ResourceBundle using given data pack path for testing.
    142   static void InitSharedInstanceWithPakPath(const base::FilePath& path);
    143 
    144   // Delete the ResourceBundle for this process if it exists.
    145   static void CleanupSharedInstance();
    146 
    147   // Returns true after the global resource loader instance has been created.
    148   static bool HasSharedInstance();
    149 
    150   // Return the global resource loader instance.
    151   static ResourceBundle& GetSharedInstance();
    152 
    153   // Check if the .pak for the given locale exists.
    154   bool LocaleDataPakExists(const std::string& locale);
    155 
    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);
    165 
    166   // Same as above but using an already open file.
    167   void AddDataPackFromFile(base::PlatformFile file, ScaleFactor scale_factor);
    168 
    169   // Same as AddDataPackFromPath but does not log an error if the pack fails to
    170   // load.
    171   void AddOptionalDataPackFromPath(const base::FilePath& path,
    172                                    ScaleFactor scale_factor);
    173 
    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);
    181 
    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*
    186   //
    187   // NOTE: GetNativeImageNamed is preferred for cross-platform gfx::Image use.
    188   gfx::ImageSkia* GetImageSkiaNamed(int resource_id);
    189 
    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);
    193 
    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.
    197   //
    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.
    201   //
    202   // If |rtl| is RTL_ENABLED then the image is flipped in RTL locales.
    203   gfx::Image& GetNativeImageNamed(int resource_id, ImageRTL rtl);
    204 
    205   // Same as GetNativeImageNamed() except that RTL is not enabled.
    206   gfx::Image& GetNativeImageNamed(int resource_id);
    207 
    208   // Loads the raw bytes of a scale independent data resource.
    209   base::RefCountedStaticMemory* LoadDataResourceBytes(int resource_id) const;
    210 
    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(
    217       int resource_id,
    218       ScaleFactor scale_factor) const;
    219 
    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;
    223 
    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;
    230 
    231   // Get a localized string given a message id.  Returns an empty
    232   // string if the message_id is not found.
    233   string16 GetLocalizedString(int message_id);
    234 
    235   // Returns the font list for the specified style.
    236   const gfx::FontList& GetFontList(FontStyle style);
    237 
    238   // Returns the font for the specified style.
    239   const gfx::Font& GetFont(FontStyle style);
    240 
    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.
    243   void ReloadFonts();
    244 
    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);
    248 
    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);
    255 
    256   // Returns the maximum scale factor currently loaded.
    257   // Returns SCALE_FACTOR_100P if no resource is loaded.
    258   ScaleFactor max_scale_factor() const {
    259     return max_scale_factor_;
    260   }
    261 
    262  private:
    263   FRIEND_TEST_ALL_PREFIXES(ResourceBundleTest, DelegateGetPathForLocalePack);
    264   FRIEND_TEST_ALL_PREFIXES(ResourceBundleTest, DelegateGetImageNamed);
    265   FRIEND_TEST_ALL_PREFIXES(ResourceBundleTest, DelegateGetNativeImageNamed);
    266 
    267   friend class ResourceBundleImageTest;
    268   friend class ResourceBundleTest;
    269 
    270   class ResourceBundleImageSource;
    271   friend class ResourceBundleImageSource;
    272 
    273   // Ctor/dtor are private, since we're a singleton.
    274   explicit ResourceBundle(Delegate* delegate);
    275   ~ResourceBundle();
    276 
    277   // Free skia_images_.
    278   void FreeImages();
    279 
    280   // Load the main resources.
    281   void LoadCommonResources();
    282 
    283   // Implementation for AddDataPackFromPath and AddOptionalDataPackFromPath, if
    284   // the pack is not |optional| logs an error on failure to load.
    285   void AddDataPackFromPathInternal(const base::FilePath& path,
    286                                    ScaleFactor scale_factor,
    287                                    bool optional);
    288 
    289   // Inserts |data_pack| to |data_pack_| and updates |max_scale_factor_|
    290   // accordingly.
    291   void AddDataPack(DataPack* data_pack);
    292 
    293   // Try to load the locale specific strings from an external data module.
    294   // Returns the locale that is loaded.
    295   std::string LoadLocaleResources(const std::string& pref_locale);
    296 
    297   // Load test resources in given paths. If either path is empty an empty
    298   // resource pack is loaded.
    299   void LoadTestResources(const base::FilePath& path,
    300                          const base::FilePath& locale_path);
    301 
    302   // Unload the locale specific strings and prepares to load new ones. See
    303   // comments for ReloadLocaleResources().
    304   void UnloadLocaleResources();
    305 
    306   // Initializes all the gfx::FontList members if they haven't yet been
    307   // initialized.
    308   void LoadFontsIfNecessary();
    309 
    310   // Returns a FontList or NULL by calling Delegate::GetFont and converting
    311   // scoped_ptr<gfx::Font> to scoped_ptr<gfx::FontList>.
    312   scoped_ptr<gfx::FontList> GetFontListFromDelegate(FontStyle style);
    313 
    314   // Fills the |bitmap| given the data file to look in and the |resource_id|.
    315   // Returns false if the resource does not exist.
    316   //
    317   // If the call succeeds, |fell_back_to_1x| indicates whether Chrome's custom
    318   // csCl PNG chunk is present (added by GRIT if it falls back to a 100% image).
    319   bool LoadBitmap(const ResourceHandle& data_handle,
    320                   int resource_id,
    321                   SkBitmap* bitmap,
    322                   bool* fell_back_to_1x) const;
    323 
    324   // Fills the |bitmap| given the |resource_id| and |scale_factor|.
    325   // Returns false if the resource does not exist. This may fall back to
    326   // the data pack with SCALE_FACTOR_NONE, and when this happens,
    327   // |scale_factor| will be set to SCALE_FACTOR_100P.
    328   bool LoadBitmap(int resource_id,
    329                   ScaleFactor* scale_factor,
    330                   SkBitmap* bitmap,
    331                   bool* fell_back_to_1x) const;
    332 
    333   // Returns true if missing scaled resources should be visually indicated when
    334   // drawing the fallback (e.g., by tinting the image).
    335   static bool ShouldHighlightMissingScaledResources();
    336 
    337   // Returns true if the data in |buf| is a PNG that has the special marker
    338   // added by GRIT that indicates that the image is actually 1x data.
    339   static bool PNGContainsFallbackMarker(const unsigned char* buf, size_t size);
    340 
    341   // A wrapper for PNGCodec::Decode that returns information about custom
    342   // chunks. For security reasons we can't alter PNGCodec to return this
    343   // information. Our PNG files are preprocessed by GRIT, and any special chunks
    344   // should occur immediately after the IHDR chunk.
    345   static bool DecodePNG(const unsigned char* buf,
    346                         size_t size,
    347                         SkBitmap* bitmap,
    348                         bool* fell_back_to_1x);
    349 
    350   // Returns an empty image for when a resource cannot be loaded. This is a
    351   // bright red bitmap.
    352   gfx::Image& GetEmptyImage();
    353 
    354   const base::FilePath& GetOverriddenPakPath();
    355 
    356   // This pointer is guaranteed to outlive the ResourceBundle instance and may
    357   // be NULL.
    358   Delegate* delegate_;
    359 
    360   // Protects |images_| and font-related members.
    361   scoped_ptr<base::Lock> images_and_fonts_lock_;
    362 
    363   // Protects |locale_resources_data_|.
    364   scoped_ptr<base::Lock> locale_resources_data_lock_;
    365 
    366   // Handles for data sources.
    367   scoped_ptr<ResourceHandle> locale_resources_data_;
    368   ScopedVector<ResourceHandle> data_packs_;
    369 
    370   // The maximum scale factor currently loaded.
    371   ScaleFactor max_scale_factor_;
    372 
    373   // Cached images. The ResourceBundle caches all retrieved images and keeps
    374   // ownership of the pointers.
    375   typedef std::map<int, gfx::Image> ImageMap;
    376   ImageMap images_;
    377 
    378   gfx::Image empty_image_;
    379 
    380   // The various font lists used. Cached to avoid repeated GDI
    381   // creation/destruction.
    382   scoped_ptr<gfx::FontList> base_font_list_;
    383   scoped_ptr<gfx::FontList> bold_font_list_;
    384   scoped_ptr<gfx::FontList> small_font_list_;
    385   scoped_ptr<gfx::FontList> small_bold_font_list_;
    386   scoped_ptr<gfx::FontList> medium_font_list_;
    387   scoped_ptr<gfx::FontList> medium_bold_font_list_;
    388   scoped_ptr<gfx::FontList> large_font_list_;
    389   scoped_ptr<gfx::FontList> large_bold_font_list_;
    390   scoped_ptr<gfx::FontList> web_font_list_;
    391 
    392   base::FilePath overridden_pak_path_;
    393 
    394   DISALLOW_COPY_AND_ASSIGN(ResourceBundle);
    395 };
    396 
    397 }  // namespace ui
    398 
    399 // TODO(beng): Someday, maybe, get rid of this.
    400 using ui::ResourceBundle;
    401 
    402 #endif  // UI_BASE_RESOURCE_RESOURCE_BUNDLE_H_
    403