1 // © 2016 and later: Unicode, Inc. and others.
2 // License & terms of use: http://www.unicode.org/copyright.html#License
3 /*
4  *******************************************************************************
5  * Copyright (C) 2004-2016, International Business Machines Corporation and
6  * others. All Rights Reserved.
7  *******************************************************************************
8  */
9 
10 package com.ibm.icu.util;
11 
12 import java.nio.ByteBuffer;
13 import java.util.Collections;
14 import java.util.Enumeration;
15 import java.util.HashMap;
16 import java.util.Locale;
17 import java.util.Map;
18 import java.util.MissingResourceException;
19 import java.util.ResourceBundle;
20 import java.util.Set;
21 import java.util.TreeSet;
22 import java.util.concurrent.ConcurrentHashMap;
23 
24 import com.ibm.icu.impl.ICUData;
25 import com.ibm.icu.impl.ICUResourceBundle;
26 import com.ibm.icu.impl.ICUResourceBundleReader;
27 import com.ibm.icu.impl.ResourceBundleWrapper;
28 
29 /**
30  * {@icuenhanced java.util.ResourceBundle}.{@icu _usage_}
31  *
32  * <p>A class representing a collection of resource information pertaining to a given
33  * locale. A resource bundle provides a way of accessing locale- specific information in a
34  * data file. You create a resource bundle that manages the resources for a given locale
35  * and then ask it for individual resources.
36  *
37  * <p>In ResourceBundle, an object is created and the sub-items are fetched using the
38  * getString and getObject methods.  In UResourceBundle, each individual element of a
39  * resource is a resource by itself.
40  *
41  * <p>Resource bundles in ICU are currently defined using text files that conform to the
42  * following <a
43  * href="http://source.icu-project.org/repos/icu/icuhtml/trunk/design/bnf_rb.txt">BNF
44  * definition</a>.  More on resource bundle concepts and syntax can be found in the <a
45  * href="http://www.icu-project.org/userguide/ResourceManagement.html">Users Guide</a>.
46  *
47  * <p>The packaging of ICU *.res files can be of two types
48  * ICU4C:
49  * <pre>
50  *       root.res
51  *         |
52  *      --------
53  *     |        |
54  *   fr.res  en.res
55  *     |
56  *   --------
57  *  |        |
58  * fr_CA.res fr_FR.res
59  * </pre>
60  * JAVA/JDK:
61  * <pre>
62  *    LocaleElements.res
63  *         |
64  *      -------------------
65  *     |                   |
66  * LocaleElements_fr.res  LocaleElements_en.res
67  *     |
68  *   ---------------------------
69  *  |                            |
70  * LocaleElements_fr_CA.res   LocaleElements_fr_FR.res
71  * </pre>
72  *
73  * Depending on the organization of your resources, the syntax to getBundleInstance will
74  * change.  To open ICU style organization use:
75  *
76  * <pre>
77  *      UResourceBundle bundle =
78  *          UResourceBundle.getBundleInstance("com/mycompany/resources",
79  *                                            "en_US", myClassLoader);
80  * </pre>
81  * To open Java/JDK style organization use:
82  * <pre>
83  *      UResourceBundle bundle =
84  *          UResourceBundle.getBundleInstance("com.mycompany.resources.LocaleElements",
85  *                                            "en_US", myClassLoader);
86  * </pre>
87  *
88  * <p>Note: Please use pass a class loader for loading non-ICU resources. Java security does not
89  * allow loading of resources across jar files. You must provide your class loader
90  * to load the resources
91 
92  * @stable ICU 3.0
93  * @author ram
94  */
95 public abstract class UResourceBundle extends ResourceBundle {
96 
97 
98     /**
99      * {@icu} Creates a resource bundle using the specified base name and locale.
100      * ICU_DATA_CLASS is used as the default root.
101      * @param baseName string containing the name of the data package.
102      *                    If null the default ICU package name is used.
103      * @param localeName the locale for which a resource bundle is desired
104      * @throws MissingResourceException If no resource bundle for the specified base name
105      * can be found
106      * @return a resource bundle for the given base name and locale
107      * @stable ICU 3.0
108      */
getBundleInstance(String baseName, String localeName)109     public static UResourceBundle getBundleInstance(String baseName, String localeName){
110         return getBundleInstance(baseName, localeName, ICUResourceBundle.ICU_DATA_CLASS_LOADER,
111                                  false);
112     }
113 
114     /**
115      * {@icu} Creates a resource bundle using the specified base name, locale, and class root.
116      *
117      * @param baseName string containing the name of the data package.
118      *                    If null the default ICU package name is used.
119      * @param localeName the locale for which a resource bundle is desired
120      * @param root the class object from which to load the resource bundle
121      * @throws MissingResourceException If no resource bundle for the specified base name
122      * can be found
123      * @return a resource bundle for the given base name and locale
124      * @stable ICU 3.0
125      */
getBundleInstance(String baseName, String localeName, ClassLoader root)126     public static UResourceBundle getBundleInstance(String baseName, String localeName,
127                                                     ClassLoader root){
128         return getBundleInstance(baseName, localeName, root, false);
129     }
130 
131     /**
132      * {@icu} Creates a resource bundle using the specified base name, locale, and class
133      * root.
134      *
135      * @param baseName string containing the name of the data package.
136      *                    If null the default ICU package name is used.
137      * @param localeName the locale for which a resource bundle is desired
138      * @param root the class object from which to load the resource bundle
139      * @param disableFallback Option to disable locale inheritence.
140      *                          If true the fallback chain will not be built.
141      * @throws MissingResourceException
142      *     if no resource bundle for the specified base name can be found
143      * @return a resource bundle for the given base name and locale
144      * @stable ICU 3.0
145      *
146      */
getBundleInstance(String baseName, String localeName, ClassLoader root, boolean disableFallback)147     protected static UResourceBundle getBundleInstance(String baseName, String localeName,
148                                                        ClassLoader root, boolean disableFallback) {
149         return instantiateBundle(baseName, localeName, root, disableFallback);
150     }
151 
152     /**
153      * {@icu} Sole constructor.  (For invocation by subclass constructors, typically
154      * implicit.)  This is public for compatibility with Java, whose compiler
155      * will generate public default constructors for an abstract class.
156      * @stable ICU 3.0
157      */
UResourceBundle()158     public UResourceBundle() {
159     }
160 
161     /**
162      * {@icu} Creates a UResourceBundle for the locale specified, from which users can extract
163      * resources by using their corresponding keys.
164      * @param locale  specifies the locale for which we want to open the resource.
165      *                If null the bundle for default locale is opened.
166      * @return a resource bundle for the given locale
167      * @stable ICU 3.0
168      */
getBundleInstance(ULocale locale)169     public static UResourceBundle getBundleInstance(ULocale locale) {
170         if (locale==null) {
171             locale = ULocale.getDefault();
172         }
173         return getBundleInstance(ICUData.ICU_BASE_NAME, locale.getBaseName(),
174                                  ICUResourceBundle.ICU_DATA_CLASS_LOADER, false);
175     }
176 
177     /**
178      * {@icu} Creates a UResourceBundle for the default locale and specified base name,
179      * from which users can extract resources by using their corresponding keys.
180      * @param baseName string containing the name of the data package.
181      *                    If null the default ICU package name is used.
182      * @return a resource bundle for the given base name and default locale
183      * @stable ICU 3.0
184      */
getBundleInstance(String baseName)185     public static UResourceBundle getBundleInstance(String baseName) {
186         if (baseName == null) {
187             baseName = ICUData.ICU_BASE_NAME;
188         }
189         ULocale uloc = ULocale.getDefault();
190         return getBundleInstance(baseName, uloc.getBaseName(), ICUResourceBundle.ICU_DATA_CLASS_LOADER,
191                                  false);
192     }
193 
194     /**
195      * {@icu} Creates a UResourceBundle for the specified locale and specified base name,
196      * from which users can extract resources by using their corresponding keys.
197      * @param baseName string containing the name of the data package.
198      *                    If null the default ICU package name is used.
199      * @param locale  specifies the locale for which we want to open the resource.
200      *                If null the bundle for default locale is opened.
201      * @return a resource bundle for the given base name and locale
202      * @stable ICU 3.0
203      */
204 
getBundleInstance(String baseName, Locale locale)205     public static UResourceBundle getBundleInstance(String baseName, Locale locale) {
206         if (baseName == null) {
207             baseName = ICUData.ICU_BASE_NAME;
208         }
209         ULocale uloc = locale == null ? ULocale.getDefault() : ULocale.forLocale(locale);
210 
211         return getBundleInstance(baseName, uloc.getBaseName(),
212                                  ICUResourceBundle.ICU_DATA_CLASS_LOADER, false);
213     }
214 
215     /**
216      * {@icu} Creates a UResourceBundle, from which users can extract resources by using
217      * their corresponding keys.
218      * @param baseName string containing the name of the data package.
219      *                    If null the default ICU package name is used.
220      * @param locale  specifies the locale for which we want to open the resource.
221      *                If null the bundle for default locale is opened.
222      * @return a resource bundle for the given base name and locale
223      * @stable ICU 3.0
224      */
getBundleInstance(String baseName, ULocale locale)225     public static UResourceBundle getBundleInstance(String baseName, ULocale locale) {
226         if (baseName == null) {
227             baseName = ICUData.ICU_BASE_NAME;
228         }
229         if (locale == null) {
230             locale = ULocale.getDefault();
231         }
232         return getBundleInstance(baseName, locale.getBaseName(),
233                                  ICUResourceBundle.ICU_DATA_CLASS_LOADER, false);
234     }
235 
236     /**
237      * {@icu} Creates a UResourceBundle for the specified locale and specified base name,
238      * from which users can extract resources by using their corresponding keys.
239      * @param baseName string containing the name of the data package.
240      *                    If null the default ICU package name is used.
241      * @param locale  specifies the locale for which we want to open the resource.
242      *                If null the bundle for default locale is opened.
243      * @param loader  the loader to use
244      * @return a resource bundle for the given base name and locale
245      * @stable ICU 3.8
246      */
getBundleInstance(String baseName, Locale locale, ClassLoader loader)247     public static UResourceBundle getBundleInstance(String baseName, Locale locale,
248                                                     ClassLoader loader) {
249         if (baseName == null) {
250             baseName = ICUData.ICU_BASE_NAME;
251         }
252         ULocale uloc = locale == null ? ULocale.getDefault() : ULocale.forLocale(locale);
253         return getBundleInstance(baseName, uloc.getBaseName(), loader, false);
254     }
255 
256     /**
257      * {@icu} Creates a UResourceBundle, from which users can extract resources by using
258      * their corresponding keys.<br><br>
259      * Note: Please use this API for loading non-ICU resources. Java security does not
260      * allow loading of resources across jar files. You must provide your class loader
261      * to load the resources
262      * @param baseName string containing the name of the data package.
263      *                    If null the default ICU package name is used.
264      * @param locale  specifies the locale for which we want to open the resource.
265      *                If null the bundle for default locale is opened.
266      * @param loader  the loader to use
267      * @return a resource bundle for the given base name and locale
268      * @stable ICU 3.8
269      */
getBundleInstance(String baseName, ULocale locale, ClassLoader loader)270     public static UResourceBundle getBundleInstance(String baseName, ULocale locale,
271                                                     ClassLoader loader) {
272         if (baseName == null) {
273             baseName = ICUData.ICU_BASE_NAME;
274         }
275         if (locale == null) {
276             locale = ULocale.getDefault();
277         }
278         return getBundleInstance(baseName, locale.getBaseName(), loader, false);
279     }
280 
281     /**
282      * {@icu} Returns the RFC 3066 conformant locale id of this resource bundle.
283      * This method can be used after a call to getBundleInstance() to
284      * determine whether the resource bundle returned really
285      * corresponds to the requested locale or is a fallback.
286      *
287      * @return the locale of this resource bundle
288      * @stable ICU 3.0
289      */
getULocale()290     public abstract ULocale getULocale();
291 
292     /**
293      * {@icu} Returns the localeID
294      * @return The string representation of the localeID
295      * @stable ICU 3.0
296      */
getLocaleID()297     protected abstract String getLocaleID();
298 
299     /**
300      * {@icu} Returns the base name of the resource bundle
301      * @return The string representation of the base name
302      * @stable ICU 3.0
303      */
getBaseName()304     protected abstract String getBaseName();
305 
306     /**
307      * {@icu} Returns the parent bundle
308      * @return The parent bundle
309      * @stable ICU 3.0
310      */
getParent()311     protected abstract UResourceBundle getParent();
312 
313 
314     /**
315      * Returns the locale of this bundle
316      * @return the locale of this resource bundle
317      * @stable ICU 3.0
318      */
319     @Override
getLocale()320     public Locale getLocale(){
321         return getULocale().toLocale();
322     }
323 
324     private enum RootType { MISSING, ICU, JAVA }
325 
326     private static Map<String, RootType> ROOT_CACHE = new ConcurrentHashMap<String, RootType>();
327 
getRootType(String baseName, ClassLoader root)328     private static RootType getRootType(String baseName, ClassLoader root) {
329         RootType rootType = ROOT_CACHE.get(baseName);
330 
331         if (rootType == null) {
332             String rootLocale = (baseName.indexOf('.')==-1) ? "root" : "";
333             try{
334                 ICUResourceBundle.getBundleInstance(baseName, rootLocale, root, true);
335                 rootType = RootType.ICU;
336             }catch(MissingResourceException ex){
337                 try{
338                     ResourceBundleWrapper.getBundleInstance(baseName, rootLocale, root, true);
339                     rootType = RootType.JAVA;
340                 }catch(MissingResourceException e){
341                     //throw away the exception
342                     rootType = RootType.MISSING;
343                 }
344             }
345 
346             ROOT_CACHE.put(baseName, rootType);
347         }
348 
349         return rootType;
350     }
351 
setRootType(String baseName, RootType rootType)352     private static void setRootType(String baseName, RootType rootType) {
353         ROOT_CACHE.put(baseName, rootType);
354     }
355 
356     /**
357      * {@icu} Loads a new resource bundle for the given base name, locale and class loader.
358      * Optionally will disable loading of fallback bundles.
359      * @param baseName string containing the name of the data package.
360      *                    If null the default ICU package name is used.
361      * @param localeName the locale for which a resource bundle is desired
362      * @param root the class object from which to load the resource bundle
363      * @param disableFallback disables loading of fallback lookup chain
364      * @throws MissingResourceException If no resource bundle for the specified base name
365      * can be found
366      * @return a resource bundle for the given base name and locale
367      * @stable ICU 3.0
368      */
instantiateBundle(String baseName, String localeName, ClassLoader root, boolean disableFallback)369     protected static UResourceBundle instantiateBundle(String baseName, String localeName,
370                                                        ClassLoader root, boolean disableFallback) {
371         RootType rootType = getRootType(baseName, root);
372 
373         switch (rootType) {
374         case ICU:
375             return ICUResourceBundle.getBundleInstance(baseName, localeName, root, disableFallback);
376 
377         case JAVA:
378             return ResourceBundleWrapper.getBundleInstance(baseName, localeName, root,
379                                                            disableFallback);
380 
381         case MISSING:
382         default:
383             UResourceBundle b;
384             try{
385                 b = ICUResourceBundle.getBundleInstance(baseName, localeName, root,
386                                                         disableFallback);
387                 setRootType(baseName, RootType.ICU);
388             }catch(MissingResourceException ex){
389                 b = ResourceBundleWrapper.getBundleInstance(baseName, localeName, root,
390                                                             disableFallback);
391                 setRootType(baseName, RootType.JAVA);
392             }
393             return b;
394         }
395     }
396 
397     /**
398      * {@icu} Returns a binary data item from a binary resource, as a read-only ByteBuffer.
399      *
400      * @return a pointer to a chunk of unsigned bytes which live in a memory mapped/DLL
401      * file.
402      * @see #getIntVector
403      * @see #getInt
404      * @throws MissingResourceException If no resource bundle can be found.
405      * @throws UResourceTypeMismatchException If the resource has a type mismatch.
406      * @stable ICU 3.8
407      */
getBinary()408     public ByteBuffer getBinary() {
409         throw new UResourceTypeMismatchException("");
410     }
411 
412     /**
413      * Returns a string from a string resource type
414      *
415      * @return a string
416      * @see #getBinary()
417      * @see #getIntVector
418      * @see #getInt
419      * @throws MissingResourceException If resource bundle is missing.
420      * @throws UResourceTypeMismatchException If resource bundle has a type mismatch.
421      * @stable ICU 3.8
422      */
getString()423     public String getString() {
424         throw new UResourceTypeMismatchException("");
425     }
426 
427     /**
428      * Returns a string array from a array resource type
429      *
430      * @return a string
431      * @see #getString()
432      * @see #getIntVector
433      * @throws MissingResourceException If resource bundle is missing.
434      * @throws UResourceTypeMismatchException If resource bundle has a type mismatch.
435      * @stable ICU 3.8
436      */
getStringArray()437     public String[] getStringArray() {
438         throw new UResourceTypeMismatchException("");
439     }
440 
441     /**
442      * {@icu} Returns a binary data from a binary resource, as a byte array with a copy
443      * of the bytes from the resource bundle.
444      *
445      * @param ba  The byte array to write the bytes to. A null variable is OK.
446      * @return an array of bytes containing the binary data from the resource.
447      * @see #getIntVector
448      * @see #getInt
449      * @throws MissingResourceException If resource bundle is missing.
450      * @throws UResourceTypeMismatchException If resource bundle has a type mismatch.
451      * @stable ICU 3.8
452      */
getBinary(byte[] ba)453     public byte[] getBinary(byte[] ba) {
454         throw new UResourceTypeMismatchException("");
455     }
456 
457     /**
458      * {@icu} Returns a 32 bit integer array from a resource.
459      *
460      * @return a pointer to a chunk of unsigned bytes which live in a memory mapped/DLL file.
461      * @see #getBinary()
462      * @see #getInt
463      * @throws MissingResourceException If resource bundle is missing.
464      * @throws UResourceTypeMismatchException If resource bundle has a type mismatch.
465      * @stable ICU 3.8
466      */
getIntVector()467     public int[] getIntVector() {
468         throw new UResourceTypeMismatchException("");
469     }
470 
471     /**
472      * {@icu} Returns a signed integer from a resource.
473      *
474      * @return an integer value
475      * @see #getIntVector
476      * @see #getBinary()
477      * @throws MissingResourceException If resource bundle is missing.
478      * @throws UResourceTypeMismatchException If resource bundle type mismatch.
479      * @stable ICU 3.8
480      */
getInt()481     public int getInt() {
482         throw new UResourceTypeMismatchException("");
483     }
484 
485     /**
486      * {@icu} Returns a unsigned integer from a resource.
487      * This integer is originally 28 bit and the sign gets propagated.
488      *
489      * @return an integer value
490      * @see #getIntVector
491      * @see #getBinary()
492      * @throws MissingResourceException If resource bundle is missing.
493      * @throws UResourceTypeMismatchException If resource bundle type mismatch.
494      * @stable ICU 3.8
495      */
getUInt()496     public int getUInt() {
497         throw new UResourceTypeMismatchException("");
498     }
499 
500     /**
501      * {@icu} Returns a resource in a given resource that has a given key.
502      *
503      * @param aKey               a key associated with the wanted resource
504      * @return                  a resource bundle object representing the resource
505      * @throws MissingResourceException If resource bundle is missing.
506      * @stable ICU 3.8
507      */
get(String aKey)508     public UResourceBundle get(String aKey) {
509         UResourceBundle obj = findTopLevel(aKey);
510         if (obj == null) {
511             String fullName = ICUResourceBundleReader.getFullName(getBaseName(), getLocaleID());
512             throw new MissingResourceException(
513                     "Can't find resource for bundle " + fullName + ", key "
514                     + aKey, this.getClass().getName(), aKey);
515         }
516         return obj;
517     }
518 
519     /**
520      * Returns a resource in a given resource that has a given key, or null if the
521      * resource is not found.
522      *
523      * @param aKey the key associated with the wanted resource
524      * @return the resource, or null
525      * @see #get(String)
526      * @internal
527      * @deprecated This API is ICU internal only.
528      */
529     @Deprecated
findTopLevel(String aKey)530     protected UResourceBundle findTopLevel(String aKey) {
531         // NOTE: this only works for top-level resources.  For resources at lower
532         // levels, it fails when you fall back to the parent, since you're now
533         // looking at root resources, not at the corresponding nested resource.
534         for (UResourceBundle res = this; res != null; res = res.getParent()) {
535             UResourceBundle obj = res.handleGet(aKey, null, this);
536             if (obj != null) {
537                 return obj;
538             }
539         }
540         return null;
541     }
542 
543     /**
544      * Returns the string in a given resource at the specified index.
545      *
546      * @param index an index to the wanted string.
547      * @return a string which lives in the resource.
548      * @throws IndexOutOfBoundsException If the index value is out of bounds of accepted values.
549      * @throws UResourceTypeMismatchException If resource bundle type mismatch.
550      * @stable ICU 3.8
551      */
getString(int index)552     public String getString(int index) {
553         ICUResourceBundle temp = (ICUResourceBundle)get(index);
554         if (temp.getType() == STRING) {
555             return temp.getString();
556         }
557         throw new UResourceTypeMismatchException("");
558     }
559 
560     /**
561      * {@icu} Returns the resource in a given resource at the specified index.
562      *
563      * @param index an index to the wanted resource.
564      * @return the sub resource UResourceBundle object
565      * @throws IndexOutOfBoundsException If the index value is out of bounds of accepted values.
566      * @throws MissingResourceException If the resource bundle is missing.
567      * @stable ICU 3.8
568      */
get(int index)569     public UResourceBundle get(int index) {
570         UResourceBundle obj = handleGet(index, null, this);
571         if (obj == null) {
572             obj = getParent();
573             if (obj != null) {
574                 obj = obj.get(index);
575             }
576             if (obj == null)
577                 throw new MissingResourceException(
578                         "Can't find resource for bundle "
579                                 + this.getClass().getName() + ", key "
580                                 + getKey(), this.getClass().getName(), getKey());
581         }
582         return obj;
583     }
584 
585     /**
586      * Returns a resource in a given resource that has a given index, or null if the
587      * resource is not found.
588      *
589      * @param index the index of the resource
590      * @return the resource, or null
591      * @see #get(int)
592      * @internal
593      * @deprecated This API is ICU internal only.
594      */
595     @Deprecated
findTopLevel(int index)596     protected UResourceBundle findTopLevel(int index) {
597         // NOTE: this _barely_ works for top-level resources.  For resources at lower
598         // levels, it fails when you fall back to the parent, since you're now
599         // looking at root resources, not at the corresponding nested resource.
600         // Not only that, but unless the indices correspond 1-to-1, the index will
601         // lose meaning.  Essentially this only works if the child resource arrays
602         // are prefixes of their parent arrays.
603         for (UResourceBundle res = this; res != null; res = res.getParent()) {
604             UResourceBundle obj = res.handleGet(index, null, this);
605             if (obj != null) {
606                 return obj;
607             }
608         }
609         return null;
610     }
611 
612     /**
613      * Returns the keys in this bundle as an enumeration
614      * @return an enumeration containing key strings,
615      *         which is empty if this is not a bundle or a table resource
616      * @stable ICU 3.8
617      */
618     @Override
getKeys()619     public Enumeration<String> getKeys() {
620         return Collections.enumeration(keySet());
621     }
622 
623     /**
624      * Returns a Set of all keys contained in this ResourceBundle and its parent bundles.
625      * @return a Set of all keys contained in this ResourceBundle and its parent bundles,
626      *         which is empty if this is not a bundle or a table resource
627      * @internal
628      * @deprecated This API is ICU internal only.
629      */
630     @Override
631     @Deprecated
keySet()632     public Set<String> keySet() {
633         // TODO: Java 6 ResourceBundle has keySet() which calls handleKeySet()
634         // and caches the results.
635         // When we upgrade to Java 6, we still need to check for isTopLevelResource().
636         // Keep the else branch as is. The if body should just return super.keySet().
637         // Remove then-redundant caching of the keys.
638         Set<String> keys = null;
639         ICUResourceBundle icurb = null;
640         if(isTopLevelResource() && this instanceof ICUResourceBundle) {
641             // We do not cache the top-level keys in this base class so that
642             // not every string/int/binary... resource has to have a keys cache field.
643             icurb = (ICUResourceBundle)this;
644             keys = icurb.getTopLevelKeySet();
645         }
646         if(keys == null) {
647             if(isTopLevelResource()) {
648                 TreeSet<String> newKeySet;
649                 if(parent == null) {
650                     newKeySet = new TreeSet<String>();
651                 } else if(parent instanceof UResourceBundle) {
652                     newKeySet = new TreeSet<String>(((UResourceBundle)parent).keySet());
653                 } else {
654                     // TODO: Java 6 ResourceBundle has keySet(); use it when we upgrade to Java 6
655                     // and remove this else branch.
656                     newKeySet = new TreeSet<String>();
657                     Enumeration<String> parentKeys = parent.getKeys();
658                     while(parentKeys.hasMoreElements()) {
659                         newKeySet.add(parentKeys.nextElement());
660                     }
661                 }
662                 newKeySet.addAll(handleKeySet());
663                 keys = Collections.unmodifiableSet(newKeySet);
664                 if(icurb != null) {
665                     icurb.setTopLevelKeySet(keys);
666                 }
667             } else {
668                 return handleKeySet();
669             }
670         }
671         return keys;
672     }
673 
674     /**
675      * Returns a Set of the keys contained <i>only</i> in this ResourceBundle.
676      * This does not include further keys from parent bundles.
677      * @return a Set of the keys contained only in this ResourceBundle,
678      *         which is empty if this is not a bundle or a table resource
679      * @internal
680      * @deprecated This API is ICU internal only.
681      */
682     @Override
683     @Deprecated
handleKeySet()684     protected Set<String> handleKeySet() {
685         return Collections.emptySet();
686     }
687 
688     /**
689      * {@icu} Returns the size of a resource. Size for scalar types is always 1, and for
690      * vector/table types is the number of child resources.
691      *
692      * <br><b>Note:</b> Integer array is treated as a scalar type. There are no APIs to
693      * access individual members of an integer array. It is always returned as a whole.
694      * @return number of resources in a given resource.
695      * @stable ICU 3.8
696      */
getSize()697     public int getSize() {
698         return 1;
699     }
700 
701     /**
702      * {@icu} Returns the type of a resource.
703      * Available types are {@link #INT INT}, {@link #ARRAY ARRAY},
704      * {@link #BINARY BINARY}, {@link #INT_VECTOR INT_VECTOR},
705      * {@link #STRING STRING}, {@link #TABLE TABLE}.
706      *
707      * @return type of the given resource.
708      * @stable ICU 3.8
709      */
getType()710     public int getType() {
711         return NONE;
712     }
713 
714     /**
715      * {@icu} Return the version number associated with this UResourceBundle as an
716      * VersionInfo object.
717      * @return VersionInfo object containing the version of the bundle
718      * @stable ICU 3.8
719      */
getVersion()720     public VersionInfo getVersion() {
721         return null;
722     }
723 
724     /**
725      * {@icu} Returns the iterator which iterates over this
726      * resource bundle
727      * @return UResourceBundleIterator that iterates over the resources in the bundle
728      * @stable ICU 3.8
729      */
getIterator()730     public UResourceBundleIterator getIterator() {
731         return new UResourceBundleIterator(this);
732     }
733 
734     /**
735      * {@icu} Returns the key associated with a given resource. Not all the resources have
736      * a key - only those that are members of a table.
737      * @return a key associated to this resource, or null if it doesn't have a key
738      * @stable ICU 3.8
739      */
getKey()740     public String getKey() {
741         return null;
742     }
743 
744     /**
745      * {@icu} Resource type constant for "no resource".
746      * @stable ICU 3.8
747      */
748     public static final int NONE = -1;
749 
750     /**
751      * {@icu} Resource type constant for strings.
752      * @stable ICU 3.8
753      */
754     public static final int STRING = 0;
755 
756     /**
757      * {@icu} Resource type constant for binary data.
758      * @stable ICU 3.8
759      */
760     public static final int BINARY = 1;
761 
762     /**
763      * {@icu} Resource type constant for tables of key-value pairs.
764      * @stable ICU 3.8
765      */
766     public static final int TABLE = 2;
767 
768     /**
769      * {@icu} Resource type constant for a single 28-bit integer, interpreted as
770      * signed or unsigned by the getInt() function.
771      * @see #getInt
772      * @stable ICU 3.8
773      */
774     public static final int INT = 7;
775 
776     /**
777      * {@icu} Resource type constant for arrays of resources.
778      * @stable ICU 3.8
779      */
780     public static final int ARRAY = 8;
781 
782     /**
783      * Resource type constant for vectors of 32-bit integers.
784      * @see #getIntVector
785      * @stable ICU 3.8
786      */
787     public static final int INT_VECTOR = 14;
788 
789     //====== protected members ==============
790 
791     /**
792      * {@icu} Actual worker method for fetching a resource based on the given key.
793      * Sub classes must override this method if they support resources with keys.
794      * @param aKey the key string of the resource to be fetched
795      * @param aliasesVisited hashtable object to hold references of resources already seen
796      * @param requested the original resource bundle object on which the get method was invoked.
797      *                  The requested bundle and the bundle on which this method is invoked
798      *                  are the same, except in the cases where aliases are involved.
799      * @return UResourceBundle a resource associated with the key
800      * @stable ICU 3.8
801      */
handleGet(String aKey, HashMap<String, String> aliasesVisited, UResourceBundle requested)802     protected UResourceBundle handleGet(String aKey, HashMap<String, String> aliasesVisited,
803                                         UResourceBundle requested) {
804         return null;
805     }
806 
807     /**
808      * {@icu} Actual worker method for fetching a resource based on the given index.
809      * Sub classes must override this method if they support arrays of resources.
810      * @param index the index of the resource to be fetched
811      * @param aliasesVisited hashtable object to hold references of resources already seen
812      * @param requested the original resource bundle object on which the get method was invoked.
813      *                  The requested bundle and the bundle on which this method is invoked
814      *                  are the same, except in the cases where aliases are involved.
815      * @return UResourceBundle a resource associated with the index
816      * @stable ICU 3.8
817      */
handleGet(int index, HashMap<String, String> aliasesVisited, UResourceBundle requested)818     protected UResourceBundle handleGet(int index, HashMap<String, String> aliasesVisited,
819                                         UResourceBundle requested) {
820         return null;
821     }
822 
823     /**
824      * {@icu} Actual worker method for fetching the array of strings in a resource.
825      * Sub classes must override this method if they support arrays of strings.
826      * @return String[] An array of strings containing strings
827      * @stable ICU 3.8
828      */
handleGetStringArray()829     protected String[] handleGetStringArray() {
830         return null;
831     }
832 
833     /**
834      * {@icu} Actual worker method for fetching the keys of resources contained in the resource.
835      * Sub classes must override this method if they support keys and associated resources.
836      *
837      * @return Enumeration An enumeration of all the keys in this resource.
838      * @stable ICU 3.8
839      */
handleGetKeys()840     protected Enumeration<String> handleGetKeys(){
841         return null;
842     }
843 
844     /**
845      * {@inheritDoc}
846      * @stable ICU 3.8
847      */
848     // this method is declared in ResourceBundle class
849     // so cannot change the signature
850     // Override this method
851     @Override
handleGetObject(String aKey)852     protected Object handleGetObject(String aKey) {
853         return handleGetObjectImpl(aKey, this);
854     }
855 
856     /**
857      * Override the superclass method
858      */
859     // To facilitate XPath style aliases we need a way to pass the reference
860     // to requested locale. The only way I could figure out is to implement
861     // the look up logic here. This has a disadvantage that if the client
862     // loads an ICUResourceBundle, calls ResourceBundle.getObject method
863     // with a key that does not exist in the bundle then the lookup is
864     // done twice before throwing a MissingResourceExpection.
handleGetObjectImpl(String aKey, UResourceBundle requested)865     private Object handleGetObjectImpl(String aKey, UResourceBundle requested) {
866         Object obj = resolveObject(aKey, requested);
867         if (obj == null) {
868             UResourceBundle parentBundle = getParent();
869             if (parentBundle != null) {
870                 obj = parentBundle.handleGetObjectImpl(aKey, requested);
871             }
872             if (obj == null)
873                 throw new MissingResourceException(
874                     "Can't find resource for bundle "
875                     + this.getClass().getName() + ", key " + aKey,
876                     this.getClass().getName(), aKey);
877         }
878         return obj;
879     }
880 
881     // Routine for figuring out the type of object to be returned
882     // string or string array
resolveObject(String aKey, UResourceBundle requested)883     private Object resolveObject(String aKey, UResourceBundle requested) {
884         if (getType() == STRING) {
885             return getString();
886         }
887         UResourceBundle obj = handleGet(aKey, null, requested);
888         if (obj != null) {
889             if (obj.getType() == STRING) {
890                 return obj.getString();
891             }
892             try {
893                 if (obj.getType() == ARRAY) {
894                     return obj.handleGetStringArray();
895                 }
896             } catch (UResourceTypeMismatchException ex) {
897                 return obj;
898             }
899         }
900         return obj;
901     }
902 
903     /**
904      * Is this a top-level resource, that is, a whole bundle?
905      * @return true if this is a top-level resource
906      * @internal
907      * @deprecated This API is ICU internal only.
908      */
909     @Deprecated
isTopLevelResource()910     protected boolean isTopLevelResource() {
911         return true;
912     }
913 }
914