• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2 *******************************************************************************
3 * Copyright (C) 2013, International Business Machines Corporation and
4 * others. All Rights Reserved.
5 *******************************************************************************
6 *
7 *
8 * File REGION.CPP
9 *
10 * Modification History:*
11 *   Date        Name        Description
12 * 01/15/13      Emmons      Original Port from ICU4J
13 ********************************************************************************
14 */
15 
16 /**
17  * \file
18  * \brief C++ API: Region classes (territory containment)
19  */
20 
21 #include "unicode/region.h"
22 #include "unicode/utypes.h"
23 #include "unicode/uobject.h"
24 #include "unicode/unistr.h"
25 #include "unicode/ures.h"
26 #include "unicode/decimfmt.h"
27 #include "ucln_in.h"
28 #include "cstring.h"
29 #include "uhash.h"
30 #include "umutex.h"
31 #include "uresimp.h"
32 #include "region_impl.h"
33 
34 #if !UCONFIG_NO_FORMATTING
35 
36 
37 U_CDECL_BEGIN
38 
39 static void U_CALLCONV
deleteRegion(void * obj)40 deleteRegion(void *obj) {
41     delete (icu::Region *)obj;
42 }
43 
44 /**
45  * Cleanup callback func
46  */
region_cleanup(void)47 static UBool U_CALLCONV region_cleanup(void)
48 {
49     icu::Region::cleanupRegionData();
50 
51     return TRUE;
52 }
53 
54 U_CDECL_END
55 
56 U_NAMESPACE_BEGIN
57 
58 static UMutex gRegionDataLock = U_MUTEX_INITIALIZER;
59 static UBool regionDataIsLoaded = false;
60 static UVector* availableRegions[URGN_LIMIT];
61 
62 static UHashtable *regionAliases;
63 static UHashtable *regionIDMap;
64 static UHashtable *numericCodeMap;
65 
66 static const UChar UNKNOWN_REGION_ID [] = { 0x5A, 0x5A, 0 };  /* "ZZ" */
67 static const UChar OUTLYING_OCEANIA_REGION_ID [] = { 0x51, 0x4F, 0 };  /* "QO" */
68 static const UChar WORLD_ID [] = { 0x30, 0x30, 0x31, 0 };  /* "001" */
69 
UOBJECT_DEFINE_RTTI_IMPLEMENTATION(RegionNameEnumeration)70 UOBJECT_DEFINE_RTTI_IMPLEMENTATION(RegionNameEnumeration)
71 
72 /*
73  * Initializes the region data from the ICU resource bundles.  The region data
74  * contains the basic relationships such as which regions are known, what the numeric
75  * codes are, any known aliases, and the territory containment data.
76  *
77  * If the region data has already loaded, then this method simply returns without doing
78  * anything meaningful.
79  */
80 void Region::loadRegionData() {
81 
82     if (regionDataIsLoaded) {
83         return;
84     }
85 
86     umtx_lock(&gRegionDataLock);
87 
88     if (regionDataIsLoaded) { // In case another thread gets to it before we do...
89         umtx_unlock(&gRegionDataLock);
90         return;
91     }
92 
93 
94     UErrorCode status = U_ZERO_ERROR;
95 
96     UResourceBundle* regionCodes = NULL;
97     UResourceBundle* territoryAlias = NULL;
98     UResourceBundle* codeMappings = NULL;
99     UResourceBundle* worldContainment = NULL;
100     UResourceBundle* territoryContainment = NULL;
101     UResourceBundle* groupingContainment = NULL;
102 
103     DecimalFormat *df = new DecimalFormat(status);
104     if (U_FAILURE(status)) {
105         umtx_unlock(&gRegionDataLock);
106         return;
107     }
108     df->setParseIntegerOnly(TRUE);
109 
110     regionIDMap = uhash_open(uhash_hashUnicodeString,uhash_compareUnicodeString,NULL,&status);
111     uhash_setValueDeleter(regionIDMap, deleteRegion);
112 
113     numericCodeMap = uhash_open(uhash_hashLong,uhash_compareLong,NULL,&status);
114 
115     regionAliases = uhash_open(uhash_hashUnicodeString,uhash_compareUnicodeString,NULL,&status);
116     uhash_setKeyDeleter(regionAliases,uprv_deleteUObject);
117 
118     UResourceBundle *rb = ures_openDirect(NULL,"metadata",&status);
119     regionCodes = ures_getByKey(rb,"regionCodes",NULL,&status);
120     territoryAlias = ures_getByKey(rb,"territoryAlias",NULL,&status);
121 
122     UResourceBundle *rb2 = ures_openDirect(NULL,"supplementalData",&status);
123     codeMappings = ures_getByKey(rb2,"codeMappings",NULL,&status);
124 
125     territoryContainment = ures_getByKey(rb2,"territoryContainment",NULL,&status);
126     worldContainment = ures_getByKey(territoryContainment,"001",NULL,&status);
127     groupingContainment = ures_getByKey(territoryContainment,"grouping",NULL,&status);
128 
129     UVector *continents = new UVector(uprv_deleteUObject, uhash_compareUnicodeString, status);
130 
131     while ( ures_hasNext(worldContainment) ) {
132         UnicodeString *continentName = new UnicodeString(ures_getNextUnicodeString(worldContainment,NULL,&status));
133         continents->addElement(continentName,status);
134     }
135 
136     UVector *groupings = new UVector(uprv_deleteUObject, uhash_compareUnicodeString, status);
137     while ( ures_hasNext(groupingContainment) ) {
138         UnicodeString *groupingName = new UnicodeString(ures_getNextUnicodeString(groupingContainment,NULL,&status));
139         groupings->addElement(groupingName,status);
140     }
141 
142     while ( ures_hasNext(regionCodes) ) {
143         UnicodeString regionID = ures_getNextUnicodeString(regionCodes,NULL,&status);
144         Region *r = new Region();
145         r->idStr = regionID;
146         r->idStr.extract(0,r->idStr.length(),r->id,sizeof(r->id),US_INV);
147         r->type = URGN_TERRITORY; // Only temporary - figure out the real type later once the aliases are known.
148 
149         uhash_put(regionIDMap,(void *)&(r->idStr),(void *)r,&status);
150         Formattable result;
151         UErrorCode ps = U_ZERO_ERROR;
152         df->parse(r->idStr,result,ps);
153         if ( U_SUCCESS(ps) ) {
154             r->code = result.getLong(); // Convert string to number
155             uhash_iput(numericCodeMap,r->code,(void *)r,&status);
156             r->type = URGN_SUBCONTINENT;
157         } else {
158             r->code = -1;
159         }
160     }
161 
162 
163     // Process the territory aliases
164     while ( ures_hasNext(territoryAlias) ) {
165         UResourceBundle *res = ures_getNextResource(territoryAlias,NULL,&status);
166         const char *aliasFrom = ures_getKey(res);
167         UnicodeString* aliasFromStr = new UnicodeString(aliasFrom, -1, US_INV);
168         UnicodeString aliasTo = ures_getUnicodeString(res,&status);
169         ures_close(res);
170 
171         Region *aliasToRegion = (Region *) uhash_get(regionIDMap,&aliasTo);
172         Region *aliasFromRegion = (Region *)uhash_get(regionIDMap,aliasFromStr);
173 
174         if ( aliasToRegion != NULL && aliasFromRegion == NULL ) { // This is just an alias from some string to a region
175             uhash_put(regionAliases,(void *)aliasFromStr, (void *)aliasToRegion,&status);
176         } else {
177             if ( aliasFromRegion == NULL ) { // Deprecated region code not in the master codes list - so need to create a deprecated region for it.
178                 aliasFromRegion = new Region();
179                 aliasFromRegion->idStr.setTo(*aliasFromStr);
180                 aliasFromRegion->idStr.extract(0,aliasFromRegion->idStr.length(),aliasFromRegion->id,sizeof(aliasFromRegion->id),US_INV);
181                 uhash_put(regionIDMap,(void *)&(aliasFromRegion->idStr),(void *)aliasFromRegion,&status);
182                 Formattable result;
183                 UErrorCode ps = U_ZERO_ERROR;
184                 df->parse(aliasFromRegion->idStr,result,ps);
185                 if ( U_SUCCESS(ps) ) {
186                     aliasFromRegion->code = result.getLong(); // Convert string to number
187                     uhash_iput(numericCodeMap,aliasFromRegion->code,(void *)aliasFromRegion,&status);
188                 } else {
189                     aliasFromRegion->code = -1;
190                 }
191                 aliasFromRegion->type = URGN_DEPRECATED;
192             } else {
193                 aliasFromRegion->type = URGN_DEPRECATED;
194             }
195             delete aliasFromStr;
196 
197             aliasFromRegion->preferredValues = new UVector(uprv_deleteUObject, uhash_compareUnicodeString, status);
198             UnicodeString currentRegion;
199             currentRegion.remove();
200             for (int32_t i = 0 ; i < aliasTo.length() ; i++ ) {
201                 if ( aliasTo.charAt(i) != 0x0020 ) {
202                     currentRegion.append(aliasTo.charAt(i));
203                 }
204                 if ( aliasTo.charAt(i) == 0x0020 || i+1 == aliasTo.length() ) {
205                     Region *target = (Region *)uhash_get(regionIDMap,(void *)&currentRegion);
206                     if (target) {
207                         UnicodeString *preferredValue = new UnicodeString(target->idStr);
208                         aliasFromRegion->preferredValues->addElement((void *)preferredValue,status);
209                     }
210                     currentRegion.remove();
211                 }
212             }
213         }
214     }
215 
216     // Process the code mappings - This will allow us to assign numeric codes to most of the territories.
217     while ( ures_hasNext(codeMappings) ) {
218         UResourceBundle *mapping = ures_getNextResource(codeMappings,NULL,&status);
219         if ( ures_getType(mapping) == URES_ARRAY && ures_getSize(mapping) == 3) {
220             UnicodeString codeMappingID = ures_getUnicodeStringByIndex(mapping,0,&status);
221             UnicodeString codeMappingNumber = ures_getUnicodeStringByIndex(mapping,1,&status);
222             UnicodeString codeMapping3Letter = ures_getUnicodeStringByIndex(mapping,2,&status);
223 
224             Region *r = (Region *)uhash_get(regionIDMap,(void *)&codeMappingID);
225             if ( r ) {
226                 Formattable result;
227                 UErrorCode ps = U_ZERO_ERROR;
228                 df->parse(codeMappingNumber,result,ps);
229                 if ( U_SUCCESS(ps) ) {
230                     r->code = result.getLong(); // Convert string to number
231                     uhash_iput(numericCodeMap,r->code,(void *)r,&status);
232                 }
233                 UnicodeString *code3 = new UnicodeString(codeMapping3Letter);
234                 uhash_put(regionAliases,(void *)code3, (void *)r,&status);
235             }
236         }
237         ures_close(mapping);
238     }
239 
240     // Now fill in the special cases for WORLD, UNKNOWN, CONTINENTS, and GROUPINGS
241     Region *r;
242 	UnicodeString WORLD_ID_STRING(WORLD_ID);
243     r = (Region *) uhash_get(regionIDMap,(void *)&WORLD_ID_STRING);
244     if ( r ) {
245         r->type = URGN_WORLD;
246     }
247 
248 	UnicodeString UNKNOWN_REGION_ID_STRING(UNKNOWN_REGION_ID);
249     r = (Region *) uhash_get(regionIDMap,(void *)&UNKNOWN_REGION_ID_STRING);
250     if ( r ) {
251         r->type = URGN_UNKNOWN;
252     }
253 
254     for ( int32_t i = 0 ; i < continents->size() ; i++ ) {
255         r = (Region *) uhash_get(regionIDMap,(void *)continents->elementAt(i));
256         if ( r ) {
257             r->type = URGN_CONTINENT;
258         }
259     }
260     delete continents;
261 
262     for ( int32_t i = 0 ; i < groupings->size() ; i++ ) {
263         r = (Region *) uhash_get(regionIDMap,(void *)groupings->elementAt(i));
264         if ( r ) {
265             r->type = URGN_GROUPING;
266         }
267     }
268     delete groupings;
269 
270     // Special case: The region code "QO" (Outlying Oceania) is a subcontinent code added by CLDR
271     // even though it looks like a territory code.  Need to handle it here.
272 
273 	UnicodeString OUTLYING_OCEANIA_REGION_ID_STRING(OUTLYING_OCEANIA_REGION_ID);
274     r = (Region *) uhash_get(regionIDMap,(void *)&OUTLYING_OCEANIA_REGION_ID_STRING);
275     if ( r ) {
276         r->type = URGN_SUBCONTINENT;
277     }
278 
279     // Load territory containment info from the supplemental data.
280     while ( ures_hasNext(territoryContainment) ) {
281         UResourceBundle *mapping = ures_getNextResource(territoryContainment,NULL,&status);
282         const char *parent = ures_getKey(mapping);
283         UnicodeString parentStr = UnicodeString(parent, -1 , US_INV);
284         Region *parentRegion = (Region *) uhash_get(regionIDMap,(void *)&parentStr);
285 
286         for ( int j = 0 ; j < ures_getSize(mapping); j++ ) {
287             UnicodeString child = ures_getUnicodeStringByIndex(mapping,j,&status);
288             Region *childRegion = (Region *) uhash_get(regionIDMap,(void *)&child);
289             if ( parentRegion != NULL && childRegion != NULL ) {
290 
291                 // Add the child region to the set of regions contained by the parent
292                 if (parentRegion->containedRegions == NULL) {
293                     parentRegion->containedRegions = new UVector(uprv_deleteUObject, uhash_compareUnicodeString, status);
294                 }
295 
296                 UnicodeString *childStr = new UnicodeString();
297                 childStr->fastCopyFrom(childRegion->idStr);
298                 parentRegion->containedRegions->addElement((void *)childStr,status);
299 
300                 // Set the parent region to be the containing region of the child.
301                 // Regions of type GROUPING can't be set as the parent, since another region
302                 // such as a SUBCONTINENT, CONTINENT, or WORLD must always be the parent.
303                 if ( parentRegion->type != URGN_GROUPING) {
304                     childRegion->containingRegion = parentRegion;
305                 }
306             }
307         }
308         ures_close(mapping);
309     }
310 
311     // Create the availableRegions lists
312     int32_t pos = -1;
313     while ( const UHashElement* element = uhash_nextElement(regionIDMap,&pos)) {
314         Region *ar = (Region *)element->value.pointer;
315         if ( availableRegions[ar->type] == NULL ) {
316             availableRegions[ar->type] = new UVector(uprv_deleteUObject, uhash_compareUnicodeString, status);
317         }
318         UnicodeString *arString = new UnicodeString(ar->idStr);
319         availableRegions[ar->type]->addElement((void *)arString,status);
320     }
321 
322     ures_close(territoryContainment);
323     ures_close(worldContainment);
324     ures_close(groupingContainment);
325 
326     ures_close(codeMappings);
327     ures_close(rb2);
328     ures_close(territoryAlias);
329     ures_close(regionCodes);
330     ures_close(rb);
331 
332     delete df;
333 
334     ucln_i18n_registerCleanup(UCLN_I18N_REGION, region_cleanup);
335 
336     regionDataIsLoaded = true;
337     umtx_unlock(&gRegionDataLock);
338 
339 }
340 
cleanupRegionData()341 void Region::cleanupRegionData() {
342 
343     for (int32_t i = 0 ; i < URGN_LIMIT ; i++ ) {
344         if ( availableRegions[i] ) {
345             delete availableRegions[i];
346         }
347     }
348 
349     if (regionAliases) {
350         uhash_close(regionAliases);
351     }
352 
353     if (numericCodeMap) {
354         uhash_close(numericCodeMap);
355     }
356 
357     if (regionIDMap) {
358         uhash_close(regionIDMap);
359     }
360 }
361 
Region()362 Region::Region ()
363         : code(-1),
364           type(URGN_UNKNOWN),
365           containingRegion(NULL),
366           containedRegions(NULL),
367           preferredValues(NULL) {
368     id[0] = 0;
369 }
370 
~Region()371 Region::~Region () {
372         if (containedRegions) {
373             delete containedRegions;
374         }
375         if (preferredValues) {
376             delete preferredValues;
377         }
378 }
379 
380 /**
381  * Returns true if the two regions are equal.
382  */
383 UBool
operator ==(const Region & that) const384 Region::operator==(const Region &that) const {
385     return (idStr == that.idStr);
386 }
387 
388 /**
389  * Returns true if the two regions are NOT equal; that is, if operator ==() returns false.
390  */
391 UBool
operator !=(const Region & that) const392 Region::operator!=(const Region &that) const {
393         return (idStr != that.idStr);
394 }
395 
396 /**
397  * Returns a pointer to a Region using the given region code.  The region code can be either 2-letter ISO code,
398  * 3-letter ISO code,  UNM.49 numeric code, or other valid Unicode Region Code as defined by the LDML specification.
399  * The identifier will be canonicalized internally using the supplemental metadata as defined in the CLDR.
400  * If the region code is NULL or not recognized, the appropriate error code will be set ( U_ILLEGAL_ARGUMENT_ERROR )
401  */
402 const Region* U_EXPORT2
getInstance(const char * region_code,UErrorCode & status)403 Region::getInstance(const char *region_code, UErrorCode &status) {
404 
405     if ( !region_code ) {
406         status = U_ILLEGAL_ARGUMENT_ERROR;
407         return NULL;
408     }
409 
410     loadRegionData();
411 
412     if (regionIDMap == NULL) {
413         status = U_ILLEGAL_ARGUMENT_ERROR;
414         return NULL;
415     }
416 
417     UnicodeString regionCodeString = UnicodeString(region_code, -1, US_INV);
418     Region *r = (Region *)uhash_get(regionIDMap,(void *)&regionCodeString);
419 
420     if ( !r ) {
421         r = (Region *)uhash_get(regionAliases,(void *)&regionCodeString);
422     }
423 
424     if ( !r ) { // Unknown region code
425         status = U_ILLEGAL_ARGUMENT_ERROR;
426         return NULL;
427     }
428 
429     if ( r->type == URGN_DEPRECATED && r->preferredValues->size() == 1) {
430         StringEnumeration *pv = r->getPreferredValues();
431         pv->reset(status);
432         const UnicodeString *ustr = pv->snext(status);
433         r = (Region *)uhash_get(regionIDMap,(void *)ustr);
434         delete pv;
435     }
436 
437     return r;
438 
439 }
440 
441 /**
442  * Returns a pointer to a Region using the given numeric region code. If the numeric region code is not recognized,
443  * the appropriate error code will be set ( U_ILLEGAL_ARGUMENT_ERROR ).
444  */
445 const Region* U_EXPORT2
getInstance(int32_t code,UErrorCode & status)446 Region::getInstance (int32_t code, UErrorCode &status) {
447 
448     loadRegionData();
449 
450     if (numericCodeMap == NULL) {
451         status = U_ILLEGAL_ARGUMENT_ERROR;
452         return NULL;
453     }
454 
455     Region *r = (Region *)uhash_iget(numericCodeMap,code);
456 
457     if ( !r ) { // Just in case there's an alias that's numeric, try to find it.
458         UErrorCode fs = U_ZERO_ERROR;
459         UnicodeString pat = UNICODE_STRING_SIMPLE("00#");
460         DecimalFormat *df = new DecimalFormat(pat,fs);
461 
462         UnicodeString id;
463         id.remove();
464         df->format(code,id);
465         delete df;
466         r = (Region *)uhash_get(regionAliases,&id);
467     }
468 
469     if ( !r ) {
470         status = U_ILLEGAL_ARGUMENT_ERROR;
471         return NULL;
472     }
473 
474     if ( r->type == URGN_DEPRECATED && r->preferredValues->size() == 1) {
475         StringEnumeration *pv = r->getPreferredValues();
476         pv->reset(status);
477         const UnicodeString *ustr = pv->snext(status);
478         r = (Region *)uhash_get(regionIDMap,(void *)ustr);
479         delete pv;
480     }
481 
482     return r;
483 }
484 
485 
486 /**
487  * Returns an enumeration over the IDs of all known regions that match the given type.
488  */
489 StringEnumeration* U_EXPORT2
getAvailable(URegionType type)490 Region::getAvailable(URegionType type) {
491 
492     loadRegionData();
493     UErrorCode status = U_ZERO_ERROR;
494     return new RegionNameEnumeration(availableRegions[type],status);
495 
496     return NULL;
497 }
498 
499 /**
500  * Returns a pointer to the region that contains this region.  Returns NULL if this region is code "001" (World)
501  * or "ZZ" (Unknown region). For example, calling this method with region "IT" (Italy) returns the
502  * region "039" (Southern Europe).
503  */
504 const Region*
getContainingRegion() const505 Region::getContainingRegion() const {
506     loadRegionData();
507     return containingRegion;
508 }
509 
510 /**
511  * Return a pointer to the region that geographically contains this region and matches the given type,
512  * moving multiple steps up the containment chain if necessary.  Returns NULL if no containing region can be found
513  * that matches the given type. Note: The URegionTypes = "URGN_GROUPING", "URGN_DEPRECATED", or "URGN_UNKNOWN"
514  * are not appropriate for use in this API. NULL will be returned in this case. For example, calling this method
515  * with region "IT" (Italy) for type "URGN_CONTINENT" returns the region "150" ( Europe ).
516  */
517 const Region*
getContainingRegion(URegionType type) const518 Region::getContainingRegion(URegionType type) const {
519     loadRegionData();
520     if ( containingRegion == NULL ) {
521         return NULL;
522     }
523 
524     if ( containingRegion->type == type ) {
525         return containingRegion;
526     } else {
527         return containingRegion->getContainingRegion(type);
528     }
529 }
530 
531 /**
532  * Return an enumeration over the IDs of all the regions that are immediate children of this region in the
533  * region hierarchy. These returned regions could be either macro regions, territories, or a mixture of the two,
534  * depending on the containment data as defined in CLDR.  This API may return NULL if this region doesn't have
535  * any sub-regions. For example, calling this method with region "150" (Europe) returns an enumeration containing
536  * the various sub regions of Europe - "039" (Southern Europe) - "151" (Eastern Europe) - "154" (Northern Europe)
537  * and "155" (Western Europe).
538  */
539 StringEnumeration*
getContainedRegions() const540 Region::getContainedRegions() const {
541     loadRegionData();
542     UErrorCode status = U_ZERO_ERROR;
543     return new RegionNameEnumeration(containedRegions,status);
544 }
545 
546 /**
547  * Returns an enumeration over the IDs of all the regions that are children of this region anywhere in the region
548  * hierarchy and match the given type.  This API may return an empty enumeration if this region doesn't have any
549  * sub-regions that match the given type. For example, calling this method with region "150" (Europe) and type
550  * "URGN_TERRITORY" returns a set containing all the territories in Europe ( "FR" (France) - "IT" (Italy) - "DE" (Germany) etc. )
551  */
552 StringEnumeration*
getContainedRegions(URegionType type) const553 Region::getContainedRegions( URegionType type ) const {
554     loadRegionData();
555 
556     UErrorCode status = U_ZERO_ERROR;
557     UVector *result = new UVector(NULL, uhash_compareChars, status);
558 
559     StringEnumeration *cr = getContainedRegions();
560 
561     for ( int32_t i = 0 ; i < cr->count(status) ; i++ ) {
562         const char *id = cr->next(NULL,status);
563         const Region *r = Region::getInstance(id,status);
564         if ( r->getType() == type ) {
565             result->addElement((void *)&r->idStr,status);
566         } else {
567             StringEnumeration *children = r->getContainedRegions(type);
568             for ( int32_t j = 0 ; j < children->count(status) ; j++ ) {
569                 const char *id2 = children->next(NULL,status);
570                 const Region *r2 = Region::getInstance(id2,status);
571                 result->addElement((void *)&r2->idStr,status);
572             }
573             delete children;
574         }
575     }
576     delete cr;
577     StringEnumeration* resultEnumeration = new RegionNameEnumeration(result,status);
578     delete result;
579     return resultEnumeration;
580 }
581 
582 /**
583  * Returns true if this region contains the supplied other region anywhere in the region hierarchy.
584  */
585 UBool
contains(const Region & other) const586 Region::contains(const Region &other) const {
587     loadRegionData();
588 
589     if (!containedRegions) {
590           return FALSE;
591     }
592     if (containedRegions->contains((void *)&other.idStr)) {
593         return TRUE;
594     } else {
595         for ( int32_t i = 0 ; i < containedRegions->size() ; i++ ) {
596             UnicodeString *crStr = (UnicodeString *)containedRegions->elementAt(i);
597             Region *cr = (Region *) uhash_get(regionIDMap,(void *)crStr);
598             if ( cr && cr->contains(other) ) {
599                 return TRUE;
600             }
601         }
602     }
603 
604     return FALSE;
605 }
606 
607 /**
608  * For deprecated regions, return an enumeration over the IDs of the regions that are the preferred replacement
609  * regions for this region.  Returns NULL for a non-deprecated region.  For example, calling this method with region
610  * "SU" (Soviet Union) would return a list of the regions containing "RU" (Russia), "AM" (Armenia), "AZ" (Azerbaijan), etc...
611  */
612 StringEnumeration*
getPreferredValues() const613 Region::getPreferredValues() const {
614     loadRegionData();
615     UErrorCode status = U_ZERO_ERROR;
616     if ( type == URGN_DEPRECATED ) {
617         return new RegionNameEnumeration(preferredValues,status);
618     } else {
619         return NULL;
620     }
621 }
622 
623 
624 /**
625  * Return this region's canonical region code.
626  */
627 const char*
getRegionCode() const628 Region::getRegionCode() const {
629     return id;
630 }
631 
632 int32_t
getNumericCode() const633 Region::getNumericCode() const {
634     return code;
635 }
636 
637 /**
638  * Returns the region type of this region.
639  */
640 URegionType
getType() const641 Region::getType() const {
642     return type;
643 }
644 
RegionNameEnumeration(UVector * fNameList,UErrorCode & status)645 RegionNameEnumeration::RegionNameEnumeration(UVector *fNameList, UErrorCode& status) {
646     pos=0;
647     if (fNameList && U_SUCCESS(status)) {
648         fRegionNames = new UVector(uprv_deleteUObject, uhash_compareUnicodeString, fNameList->size(),status);
649         for ( int32_t i = 0 ; i < fNameList->size() ; i++ ) {
650             UnicodeString* this_region_name = (UnicodeString *)fNameList->elementAt(i);
651             UnicodeString* new_region_name = new UnicodeString(*this_region_name);
652             fRegionNames->addElement((void *)new_region_name,status);
653         }
654     }
655     else {
656         fRegionNames = NULL;
657     }
658 }
659 
660 const UnicodeString*
snext(UErrorCode & status)661 RegionNameEnumeration::snext(UErrorCode& status) {
662   if (U_FAILURE(status) || (fRegionNames==NULL)) {
663     return NULL;
664   }
665   const UnicodeString* nextStr = (const UnicodeString *)fRegionNames->elementAt(pos);
666   if (nextStr!=NULL) {
667     pos++;
668   }
669   return nextStr;
670 }
671 
672 void
reset(UErrorCode &)673 RegionNameEnumeration::reset(UErrorCode& /*status*/) {
674     pos=0;
675 }
676 
677 int32_t
count(UErrorCode &) const678 RegionNameEnumeration::count(UErrorCode& /*status*/) const {
679     return (fRegionNames==NULL) ? 0 : fRegionNames->size();
680 }
681 
~RegionNameEnumeration()682 RegionNameEnumeration::~RegionNameEnumeration() {
683     delete fRegionNames;
684 }
685 
686 U_NAMESPACE_END
687 
688 #endif /* #if !UCONFIG_NO_FORMATTING */
689 
690 //eof
691