1 // Copyright (C) 2016 and later: Unicode, Inc. and others.
2 // License & terms of use: http://www.unicode.org/copyright.html
3 /*
4 *******************************************************************************
5 *
6 * Copyright (C) 1997-2016, International Business Machines
7 * Corporation and others. All Rights Reserved.
8 *
9 *******************************************************************************
10 * file name: locdispnames.cpp
11 * encoding: US-ASCII
12 * tab size: 8 (not used)
13 * indentation:4
14 *
15 * created on: 2010feb25
16 * created by: Markus W. Scherer
17 *
18 * Code for locale display names, separated out from other .cpp files
19 * that then do not depend on resource bundle code and display name data.
20 */
21
22 #include "unicode/utypes.h"
23 #include "unicode/brkiter.h"
24 #include "unicode/locid.h"
25 #include "unicode/uloc.h"
26 #include "unicode/ures.h"
27 #include "unicode/ustring.h"
28 #include "cmemory.h"
29 #include "cstring.h"
30 #include "putilimp.h"
31 #include "ulocimp.h"
32 #include "uresimp.h"
33 #include "ureslocs.h"
34 #include "ustr_imp.h"
35
36 // C++ API ----------------------------------------------------------------- ***
37
38 U_NAMESPACE_BEGIN
39
40 UnicodeString&
getDisplayLanguage(UnicodeString & dispLang) const41 Locale::getDisplayLanguage(UnicodeString& dispLang) const
42 {
43 return this->getDisplayLanguage(getDefault(), dispLang);
44 }
45
46 /*We cannot make any assumptions on the size of the output display strings
47 * Yet, since we are calling through to a C API, we need to set limits on
48 * buffer size. For all the following getDisplay functions we first attempt
49 * to fill up a stack allocated buffer. If it is to small we heap allocated
50 * the exact buffer we need copy it to the UnicodeString and delete it*/
51
52 UnicodeString&
getDisplayLanguage(const Locale & displayLocale,UnicodeString & result) const53 Locale::getDisplayLanguage(const Locale &displayLocale,
54 UnicodeString &result) const {
55 UChar *buffer;
56 UErrorCode errorCode=U_ZERO_ERROR;
57 int32_t length;
58
59 buffer=result.getBuffer(ULOC_FULLNAME_CAPACITY);
60 if(buffer==0) {
61 result.truncate(0);
62 return result;
63 }
64
65 length=uloc_getDisplayLanguage(fullName, displayLocale.fullName,
66 buffer, result.getCapacity(),
67 &errorCode);
68 result.releaseBuffer(U_SUCCESS(errorCode) ? length : 0);
69
70 if(errorCode==U_BUFFER_OVERFLOW_ERROR) {
71 buffer=result.getBuffer(length);
72 if(buffer==0) {
73 result.truncate(0);
74 return result;
75 }
76 errorCode=U_ZERO_ERROR;
77 length=uloc_getDisplayLanguage(fullName, displayLocale.fullName,
78 buffer, result.getCapacity(),
79 &errorCode);
80 result.releaseBuffer(U_SUCCESS(errorCode) ? length : 0);
81 }
82
83 return result;
84 }
85
86 UnicodeString&
getDisplayScript(UnicodeString & dispScript) const87 Locale::getDisplayScript(UnicodeString& dispScript) const
88 {
89 return this->getDisplayScript(getDefault(), dispScript);
90 }
91
92 UnicodeString&
getDisplayScript(const Locale & displayLocale,UnicodeString & result) const93 Locale::getDisplayScript(const Locale &displayLocale,
94 UnicodeString &result) const {
95 UChar *buffer;
96 UErrorCode errorCode=U_ZERO_ERROR;
97 int32_t length;
98
99 buffer=result.getBuffer(ULOC_FULLNAME_CAPACITY);
100 if(buffer==0) {
101 result.truncate(0);
102 return result;
103 }
104
105 length=uloc_getDisplayScript(fullName, displayLocale.fullName,
106 buffer, result.getCapacity(),
107 &errorCode);
108 result.releaseBuffer(U_SUCCESS(errorCode) ? length : 0);
109
110 if(errorCode==U_BUFFER_OVERFLOW_ERROR) {
111 buffer=result.getBuffer(length);
112 if(buffer==0) {
113 result.truncate(0);
114 return result;
115 }
116 errorCode=U_ZERO_ERROR;
117 length=uloc_getDisplayScript(fullName, displayLocale.fullName,
118 buffer, result.getCapacity(),
119 &errorCode);
120 result.releaseBuffer(U_SUCCESS(errorCode) ? length : 0);
121 }
122
123 return result;
124 }
125
126 UnicodeString&
getDisplayCountry(UnicodeString & dispCntry) const127 Locale::getDisplayCountry(UnicodeString& dispCntry) const
128 {
129 return this->getDisplayCountry(getDefault(), dispCntry);
130 }
131
132 UnicodeString&
getDisplayCountry(const Locale & displayLocale,UnicodeString & result) const133 Locale::getDisplayCountry(const Locale &displayLocale,
134 UnicodeString &result) const {
135 UChar *buffer;
136 UErrorCode errorCode=U_ZERO_ERROR;
137 int32_t length;
138
139 buffer=result.getBuffer(ULOC_FULLNAME_CAPACITY);
140 if(buffer==0) {
141 result.truncate(0);
142 return result;
143 }
144
145 length=uloc_getDisplayCountry(fullName, displayLocale.fullName,
146 buffer, result.getCapacity(),
147 &errorCode);
148 result.releaseBuffer(U_SUCCESS(errorCode) ? length : 0);
149
150 if(errorCode==U_BUFFER_OVERFLOW_ERROR) {
151 buffer=result.getBuffer(length);
152 if(buffer==0) {
153 result.truncate(0);
154 return result;
155 }
156 errorCode=U_ZERO_ERROR;
157 length=uloc_getDisplayCountry(fullName, displayLocale.fullName,
158 buffer, result.getCapacity(),
159 &errorCode);
160 result.releaseBuffer(U_SUCCESS(errorCode) ? length : 0);
161 }
162
163 return result;
164 }
165
166 UnicodeString&
getDisplayVariant(UnicodeString & dispVar) const167 Locale::getDisplayVariant(UnicodeString& dispVar) const
168 {
169 return this->getDisplayVariant(getDefault(), dispVar);
170 }
171
172 UnicodeString&
getDisplayVariant(const Locale & displayLocale,UnicodeString & result) const173 Locale::getDisplayVariant(const Locale &displayLocale,
174 UnicodeString &result) const {
175 UChar *buffer;
176 UErrorCode errorCode=U_ZERO_ERROR;
177 int32_t length;
178
179 buffer=result.getBuffer(ULOC_FULLNAME_CAPACITY);
180 if(buffer==0) {
181 result.truncate(0);
182 return result;
183 }
184
185 length=uloc_getDisplayVariant(fullName, displayLocale.fullName,
186 buffer, result.getCapacity(),
187 &errorCode);
188 result.releaseBuffer(U_SUCCESS(errorCode) ? length : 0);
189
190 if(errorCode==U_BUFFER_OVERFLOW_ERROR) {
191 buffer=result.getBuffer(length);
192 if(buffer==0) {
193 result.truncate(0);
194 return result;
195 }
196 errorCode=U_ZERO_ERROR;
197 length=uloc_getDisplayVariant(fullName, displayLocale.fullName,
198 buffer, result.getCapacity(),
199 &errorCode);
200 result.releaseBuffer(U_SUCCESS(errorCode) ? length : 0);
201 }
202
203 return result;
204 }
205
206 UnicodeString&
getDisplayName(UnicodeString & name) const207 Locale::getDisplayName( UnicodeString& name ) const
208 {
209 return this->getDisplayName(getDefault(), name);
210 }
211
212 UnicodeString&
getDisplayName(const Locale & displayLocale,UnicodeString & result) const213 Locale::getDisplayName(const Locale &displayLocale,
214 UnicodeString &result) const {
215 UChar *buffer;
216 UErrorCode errorCode=U_ZERO_ERROR;
217 int32_t length;
218
219 buffer=result.getBuffer(ULOC_FULLNAME_CAPACITY);
220 if(buffer==0) {
221 result.truncate(0);
222 return result;
223 }
224
225 length=uloc_getDisplayName(fullName, displayLocale.fullName,
226 buffer, result.getCapacity(),
227 &errorCode);
228 result.releaseBuffer(U_SUCCESS(errorCode) ? length : 0);
229
230 if(errorCode==U_BUFFER_OVERFLOW_ERROR) {
231 buffer=result.getBuffer(length);
232 if(buffer==0) {
233 result.truncate(0);
234 return result;
235 }
236 errorCode=U_ZERO_ERROR;
237 length=uloc_getDisplayName(fullName, displayLocale.fullName,
238 buffer, result.getCapacity(),
239 &errorCode);
240 result.releaseBuffer(U_SUCCESS(errorCode) ? length : 0);
241 }
242
243 return result;
244 }
245
246 #if ! UCONFIG_NO_BREAK_ITERATION
247
248 // -------------------------------------
249 // Gets the objectLocale display name in the default locale language.
250 UnicodeString& U_EXPORT2
getDisplayName(const Locale & objectLocale,UnicodeString & name)251 BreakIterator::getDisplayName(const Locale& objectLocale,
252 UnicodeString& name)
253 {
254 return objectLocale.getDisplayName(name);
255 }
256
257 // -------------------------------------
258 // Gets the objectLocale display name in the displayLocale language.
259 UnicodeString& U_EXPORT2
getDisplayName(const Locale & objectLocale,const Locale & displayLocale,UnicodeString & name)260 BreakIterator::getDisplayName(const Locale& objectLocale,
261 const Locale& displayLocale,
262 UnicodeString& name)
263 {
264 return objectLocale.getDisplayName(displayLocale, name);
265 }
266
267 #endif
268
269
270 U_NAMESPACE_END
271
272 // C API ------------------------------------------------------------------- ***
273
274 U_NAMESPACE_USE
275
276 /* ### Constants **************************************************/
277
278 /* These strings describe the resources we attempt to load from
279 the locale ResourceBundle data file.*/
280 static const char _kLanguages[] = "Languages";
281 static const char _kScripts[] = "Scripts";
282 static const char _kScriptsStandAlone[] = "Scripts%stand-alone";
283 static const char _kCountries[] = "Countries";
284 static const char _kVariants[] = "Variants";
285 static const char _kKeys[] = "Keys";
286 static const char _kTypes[] = "Types";
287 //static const char _kRootName[] = "root";
288 static const char _kCurrency[] = "currency";
289 static const char _kCurrencies[] = "Currencies";
290 static const char _kLocaleDisplayPattern[] = "localeDisplayPattern";
291 static const char _kPattern[] = "pattern";
292 static const char _kSeparator[] = "separator";
293
294 /* ### Display name **************************************************/
295
296 static int32_t
_getStringOrCopyKey(const char * path,const char * locale,const char * tableKey,const char * subTableKey,const char * itemKey,const char * substitute,UChar * dest,int32_t destCapacity,UErrorCode * pErrorCode)297 _getStringOrCopyKey(const char *path, const char *locale,
298 const char *tableKey,
299 const char* subTableKey,
300 const char *itemKey,
301 const char *substitute,
302 UChar *dest, int32_t destCapacity,
303 UErrorCode *pErrorCode) {
304 const UChar *s = NULL;
305 int32_t length = 0;
306
307 if(itemKey==NULL) {
308 /* top-level item: normal resource bundle access */
309 UResourceBundle *rb;
310
311 rb=ures_open(path, locale, pErrorCode);
312
313 if(U_SUCCESS(*pErrorCode)) {
314 s=ures_getStringByKey(rb, tableKey, &length, pErrorCode);
315 /* see comment about closing rb near "return item;" in _res_getTableStringWithFallback() */
316 ures_close(rb);
317 }
318 } else {
319 /* Language code should not be a number. If it is, set the error code. */
320 if (!uprv_strncmp(tableKey, "Languages", 9) && uprv_strtol(itemKey, NULL, 10)) {
321 *pErrorCode = U_MISSING_RESOURCE_ERROR;
322 } else {
323 /* second-level item, use special fallback */
324 s=uloc_getTableStringWithFallback(path, locale,
325 tableKey,
326 subTableKey,
327 itemKey,
328 &length,
329 pErrorCode);
330 }
331 }
332
333 if(U_SUCCESS(*pErrorCode)) {
334 int32_t copyLength=uprv_min(length, destCapacity);
335 if(copyLength>0 && s != NULL) {
336 u_memcpy(dest, s, copyLength);
337 }
338 } else {
339 /* no string from a resource bundle: convert the substitute */
340 length=(int32_t)uprv_strlen(substitute);
341 u_charsToUChars(substitute, dest, uprv_min(length, destCapacity));
342 *pErrorCode=U_USING_DEFAULT_WARNING;
343 }
344
345 return u_terminateUChars(dest, destCapacity, length, pErrorCode);
346 }
347
348 typedef int32_t U_CALLCONV UDisplayNameGetter(const char *, char *, int32_t, UErrorCode *);
349
350 static int32_t
_getDisplayNameForComponent(const char * locale,const char * displayLocale,UChar * dest,int32_t destCapacity,UDisplayNameGetter * getter,const char * tag,UErrorCode * pErrorCode)351 _getDisplayNameForComponent(const char *locale,
352 const char *displayLocale,
353 UChar *dest, int32_t destCapacity,
354 UDisplayNameGetter *getter,
355 const char *tag,
356 UErrorCode *pErrorCode) {
357 char localeBuffer[ULOC_FULLNAME_CAPACITY*4];
358 int32_t length;
359 UErrorCode localStatus;
360 const char* root = NULL;
361
362 /* argument checking */
363 if(pErrorCode==NULL || U_FAILURE(*pErrorCode)) {
364 return 0;
365 }
366
367 if(destCapacity<0 || (destCapacity>0 && dest==NULL)) {
368 *pErrorCode=U_ILLEGAL_ARGUMENT_ERROR;
369 return 0;
370 }
371
372 localStatus = U_ZERO_ERROR;
373 length=(*getter)(locale, localeBuffer, sizeof(localeBuffer), &localStatus);
374 if(U_FAILURE(localStatus) || localStatus==U_STRING_NOT_TERMINATED_WARNING) {
375 *pErrorCode=U_ILLEGAL_ARGUMENT_ERROR;
376 return 0;
377 }
378 if(length==0) {
379 return u_terminateUChars(dest, destCapacity, 0, pErrorCode);
380 }
381
382 root = tag == _kCountries ? U_ICUDATA_REGION : U_ICUDATA_LANG;
383
384 return _getStringOrCopyKey(root, displayLocale,
385 tag, NULL, localeBuffer,
386 localeBuffer,
387 dest, destCapacity,
388 pErrorCode);
389 }
390
391 U_CAPI int32_t U_EXPORT2
uloc_getDisplayLanguage(const char * locale,const char * displayLocale,UChar * dest,int32_t destCapacity,UErrorCode * pErrorCode)392 uloc_getDisplayLanguage(const char *locale,
393 const char *displayLocale,
394 UChar *dest, int32_t destCapacity,
395 UErrorCode *pErrorCode) {
396 return _getDisplayNameForComponent(locale, displayLocale, dest, destCapacity,
397 uloc_getLanguage, _kLanguages, pErrorCode);
398 }
399
400 U_CAPI int32_t U_EXPORT2
uloc_getDisplayScript(const char * locale,const char * displayLocale,UChar * dest,int32_t destCapacity,UErrorCode * pErrorCode)401 uloc_getDisplayScript(const char* locale,
402 const char* displayLocale,
403 UChar *dest, int32_t destCapacity,
404 UErrorCode *pErrorCode)
405 {
406 UErrorCode err = U_ZERO_ERROR;
407 int32_t res = _getDisplayNameForComponent(locale, displayLocale, dest, destCapacity,
408 uloc_getScript, _kScriptsStandAlone, &err);
409
410 if ( err == U_USING_DEFAULT_WARNING ) {
411 return _getDisplayNameForComponent(locale, displayLocale, dest, destCapacity,
412 uloc_getScript, _kScripts, pErrorCode);
413 } else {
414 *pErrorCode = err;
415 return res;
416 }
417 }
418
419 U_INTERNAL int32_t U_EXPORT2
uloc_getDisplayScriptInContext(const char * locale,const char * displayLocale,UChar * dest,int32_t destCapacity,UErrorCode * pErrorCode)420 uloc_getDisplayScriptInContext(const char* locale,
421 const char* displayLocale,
422 UChar *dest, int32_t destCapacity,
423 UErrorCode *pErrorCode)
424 {
425 return _getDisplayNameForComponent(locale, displayLocale, dest, destCapacity,
426 uloc_getScript, _kScripts, pErrorCode);
427 }
428
429 U_CAPI int32_t U_EXPORT2
uloc_getDisplayCountry(const char * locale,const char * displayLocale,UChar * dest,int32_t destCapacity,UErrorCode * pErrorCode)430 uloc_getDisplayCountry(const char *locale,
431 const char *displayLocale,
432 UChar *dest, int32_t destCapacity,
433 UErrorCode *pErrorCode) {
434 return _getDisplayNameForComponent(locale, displayLocale, dest, destCapacity,
435 uloc_getCountry, _kCountries, pErrorCode);
436 }
437
438 /*
439 * TODO separate variant1_variant2_variant3...
440 * by getting each tag's display string and concatenating them with ", "
441 * in between - similar to uloc_getDisplayName()
442 */
443 U_CAPI int32_t U_EXPORT2
uloc_getDisplayVariant(const char * locale,const char * displayLocale,UChar * dest,int32_t destCapacity,UErrorCode * pErrorCode)444 uloc_getDisplayVariant(const char *locale,
445 const char *displayLocale,
446 UChar *dest, int32_t destCapacity,
447 UErrorCode *pErrorCode) {
448 return _getDisplayNameForComponent(locale, displayLocale, dest, destCapacity,
449 uloc_getVariant, _kVariants, pErrorCode);
450 }
451
452 /* Instead of having a separate pass for 'special' patterns, reintegrate the two
453 * so we don't get bitten by preflight bugs again. We can be reasonably efficient
454 * without two separate code paths, this code isn't that performance-critical.
455 *
456 * This code is general enough to deal with patterns that have a prefix or swap the
457 * language and remainder components, since we gave developers enough rope to do such
458 * things if they futz with the pattern data. But since we don't give them a way to
459 * specify a pattern for arbitrary combinations of components, there's not much use in
460 * that. I don't think our data includes such patterns, the only variable I know if is
461 * whether there is a space before the open paren, or not. Oh, and zh uses different
462 * chars than the standard open/close paren (which ja and ko use, btw).
463 */
464 U_CAPI int32_t U_EXPORT2
uloc_getDisplayName(const char * locale,const char * displayLocale,UChar * dest,int32_t destCapacity,UErrorCode * pErrorCode)465 uloc_getDisplayName(const char *locale,
466 const char *displayLocale,
467 UChar *dest, int32_t destCapacity,
468 UErrorCode *pErrorCode)
469 {
470 static const UChar defaultSeparator[9] = { 0x007b, 0x0030, 0x007d, 0x002c, 0x0020, 0x007b, 0x0031, 0x007d, 0x0000 }; /* "{0}, {1}" */
471 static const UChar sub0[4] = { 0x007b, 0x0030, 0x007d , 0x0000 } ; /* {0} */
472 static const UChar sub1[4] = { 0x007b, 0x0031, 0x007d , 0x0000 } ; /* {1} */
473 static const int32_t subLen = 3;
474 static const UChar defaultPattern[10] = {
475 0x007b, 0x0030, 0x007d, 0x0020, 0x0028, 0x007b, 0x0031, 0x007d, 0x0029, 0x0000
476 }; /* {0} ({1}) */
477 static const int32_t defaultPatLen = 9;
478 static const int32_t defaultSub0Pos = 0;
479 static const int32_t defaultSub1Pos = 5;
480
481 int32_t length; /* of formatted result */
482
483 const UChar *separator;
484 int32_t sepLen = 0;
485 const UChar *pattern;
486 int32_t patLen = 0;
487 int32_t sub0Pos, sub1Pos;
488
489 UChar formatOpenParen = 0x0028; // (
490 UChar formatReplaceOpenParen = 0x005B; // [
491 UChar formatCloseParen = 0x0029; // )
492 UChar formatReplaceCloseParen = 0x005D; // ]
493
494 UBool haveLang = TRUE; /* assume true, set false if we find we don't have
495 a lang component in the locale */
496 UBool haveRest = TRUE; /* assume true, set false if we find we don't have
497 any other component in the locale */
498 UBool retry = FALSE; /* set true if we need to retry, see below */
499
500 int32_t langi = 0; /* index of the language substitution (0 or 1), virtually always 0 */
501
502 if(pErrorCode==NULL || U_FAILURE(*pErrorCode)) {
503 return 0;
504 }
505
506 if(destCapacity<0 || (destCapacity>0 && dest==NULL)) {
507 *pErrorCode=U_ILLEGAL_ARGUMENT_ERROR;
508 return 0;
509 }
510
511 {
512 UErrorCode status = U_ZERO_ERROR;
513 UResourceBundle* locbundle=ures_open(U_ICUDATA_LANG, displayLocale, &status);
514 UResourceBundle* dspbundle=ures_getByKeyWithFallback(locbundle, _kLocaleDisplayPattern,
515 NULL, &status);
516
517 separator=ures_getStringByKeyWithFallback(dspbundle, _kSeparator, &sepLen, &status);
518 pattern=ures_getStringByKeyWithFallback(dspbundle, _kPattern, &patLen, &status);
519
520 ures_close(dspbundle);
521 ures_close(locbundle);
522 }
523
524 /* If we couldn't find any data, then use the defaults */
525 if(sepLen == 0) {
526 separator = defaultSeparator;
527 }
528 /* #10244: Even though separator is now a pattern, it is awkward to handle it as such
529 * here since we are trying to build the display string in place in the dest buffer,
530 * and to handle it as a pattern would entail having separate storage for the
531 * substrings that need to be combined (the first of which may be the result of
532 * previous such combinations). So for now we continue to treat the portion between
533 * {0} and {1} as a string to be appended when joining substrings, ignoring anything
534 * that is before {0} or after {1} (no existing separator pattern has any such thing).
535 * This is similar to how pattern is handled below.
536 */
537 {
538 UChar *p0=u_strstr(separator, sub0);
539 UChar *p1=u_strstr(separator, sub1);
540 if (p0==NULL || p1==NULL || p1<p0) {
541 *pErrorCode=U_ILLEGAL_ARGUMENT_ERROR;
542 return 0;
543 }
544 separator = (const UChar *)p0 + subLen;
545 sepLen = p1 - separator;
546 }
547
548 if(patLen==0 || (patLen==defaultPatLen && !u_strncmp(pattern, defaultPattern, patLen))) {
549 pattern=defaultPattern;
550 patLen=defaultPatLen;
551 sub0Pos=defaultSub0Pos;
552 sub1Pos=defaultSub1Pos;
553 // use default formatOpenParen etc. set above
554 } else { /* non-default pattern */
555 UChar *p0=u_strstr(pattern, sub0);
556 UChar *p1=u_strstr(pattern, sub1);
557 if (p0==NULL || p1==NULL) {
558 *pErrorCode=U_ILLEGAL_ARGUMENT_ERROR;
559 return 0;
560 }
561 sub0Pos=p0-pattern;
562 sub1Pos=p1-pattern;
563 if (sub1Pos < sub0Pos) { /* a very odd pattern */
564 int32_t t=sub0Pos; sub0Pos=sub1Pos; sub1Pos=t;
565 langi=1;
566 }
567 if (u_strchr(pattern, 0xFF08) != NULL) {
568 formatOpenParen = 0xFF08; // fullwidth (
569 formatReplaceOpenParen = 0xFF3B; // fullwidth [
570 formatCloseParen = 0xFF09; // fullwidth )
571 formatReplaceCloseParen = 0xFF3D; // fullwidth ]
572 }
573 }
574
575 /* We loop here because there is one case in which after the first pass we could need to
576 * reextract the data. If there's initial padding before the first element, we put in
577 * the padding and then write that element. If it turns out there's no second element,
578 * we didn't need the padding. If we do need the data (no preflight), and the first element
579 * would have fit but for the padding, we need to reextract. In this case (only) we
580 * adjust the parameters so padding is not added, and repeat.
581 */
582 do {
583 UChar* p=dest;
584 int32_t patPos=0; /* position in the pattern, used for non-substitution portions */
585 int32_t langLen=0; /* length of language substitution */
586 int32_t langPos=0; /* position in output of language substitution */
587 int32_t restLen=0; /* length of 'everything else' substitution */
588 int32_t restPos=0; /* position in output of 'everything else' substitution */
589 UEnumeration* kenum = NULL; /* keyword enumeration */
590
591 /* prefix of pattern, extremely likely to be empty */
592 if(sub0Pos) {
593 if(destCapacity >= sub0Pos) {
594 while (patPos < sub0Pos) {
595 *p++ = pattern[patPos++];
596 }
597 } else {
598 patPos=sub0Pos;
599 }
600 length=sub0Pos;
601 } else {
602 length=0;
603 }
604
605 for(int32_t subi=0,resti=0;subi<2;) { /* iterate through patterns 0 and 1*/
606 UBool subdone = FALSE; /* set true when ready to move to next substitution */
607
608 /* prep p and cap for calls to get display components, pin cap to 0 since
609 they complain if cap is negative */
610 int32_t cap=destCapacity-length;
611 if (cap <= 0) {
612 cap=0;
613 } else {
614 p=dest+length;
615 }
616
617 if (subi == langi) { /* {0}*/
618 if(haveLang) {
619 langPos=length;
620 langLen=uloc_getDisplayLanguage(locale, displayLocale, p, cap, pErrorCode);
621 length+=langLen;
622 haveLang=langLen>0;
623 }
624 subdone=TRUE;
625 } else { /* {1} */
626 if(!haveRest) {
627 subdone=TRUE;
628 } else {
629 int32_t len; /* length of component (plus other stuff) we just fetched */
630 switch(resti++) {
631 case 0:
632 restPos=length;
633 len=uloc_getDisplayScriptInContext(locale, displayLocale, p, cap, pErrorCode);
634 break;
635 case 1:
636 len=uloc_getDisplayCountry(locale, displayLocale, p, cap, pErrorCode);
637 break;
638 case 2:
639 len=uloc_getDisplayVariant(locale, displayLocale, p, cap, pErrorCode);
640 break;
641 case 3:
642 kenum = uloc_openKeywords(locale, pErrorCode);
643 U_FALLTHROUGH;
644 default: {
645 const char* kw=uenum_next(kenum, &len, pErrorCode);
646 if (kw == NULL) {
647 uenum_close(kenum);
648 len=0; /* mark that we didn't add a component */
649 subdone=TRUE;
650 } else {
651 /* incorporating this behavior into the loop made it even more complex,
652 so just special case it here */
653 len = uloc_getDisplayKeyword(kw, displayLocale, p, cap, pErrorCode);
654 if(len) {
655 if(len < cap) {
656 p[len]=0x3d; /* '=', assume we'll need it */
657 }
658 len+=1;
659
660 /* adjust for call to get keyword */
661 cap-=len;
662 if(cap <= 0) {
663 cap=0;
664 } else {
665 p+=len;
666 }
667 }
668 /* reset for call below */
669 if(*pErrorCode == U_BUFFER_OVERFLOW_ERROR) {
670 *pErrorCode=U_ZERO_ERROR;
671 }
672 int32_t vlen = uloc_getDisplayKeywordValue(locale, kw, displayLocale,
673 p, cap, pErrorCode);
674 if(len) {
675 if(vlen==0) {
676 --len; /* remove unneeded '=' */
677 }
678 /* restore cap and p to what they were at start */
679 cap=destCapacity-length;
680 if(cap <= 0) {
681 cap=0;
682 } else {
683 p=dest+length;
684 }
685 }
686 len+=vlen; /* total we added for key + '=' + value */
687 }
688 } break;
689 } /* end switch */
690
691 if (len>0) {
692 /* we addeed a component, so add separator and write it if there's room. */
693 if(len+sepLen<=cap) {
694 const UChar * plimit = p + len;
695 for (; p < plimit; p++) {
696 if (*p == formatOpenParen) {
697 *p = formatReplaceOpenParen;
698 } else if (*p == formatCloseParen) {
699 *p = formatReplaceCloseParen;
700 }
701 }
702 for(int32_t i=0;i<sepLen;++i) {
703 *p++=separator[i];
704 }
705 }
706 length+=len+sepLen;
707 } else if(subdone) {
708 /* remove separator if we added it */
709 if (length!=restPos) {
710 length-=sepLen;
711 }
712 restLen=length-restPos;
713 haveRest=restLen>0;
714 }
715 }
716 }
717
718 if(*pErrorCode == U_BUFFER_OVERFLOW_ERROR) {
719 *pErrorCode=U_ZERO_ERROR;
720 }
721
722 if(subdone) {
723 if(haveLang && haveRest) {
724 /* append internal portion of pattern, the first time,
725 or last portion of pattern the second time */
726 int32_t padLen;
727 patPos+=subLen;
728 padLen=(subi==0 ? sub1Pos : patLen)-patPos;
729 if(length+padLen < destCapacity) {
730 p=dest+length;
731 for(int32_t i=0;i<padLen;++i) {
732 *p++=pattern[patPos++];
733 }
734 } else {
735 patPos+=padLen;
736 }
737 length+=padLen;
738 } else if(subi==0) {
739 /* don't have first component, reset for second component */
740 sub0Pos=0;
741 length=0;
742 } else if(length>0) {
743 /* true length is the length of just the component we got. */
744 length=haveLang?langLen:restLen;
745 if(dest && sub0Pos!=0) {
746 if (sub0Pos+length<=destCapacity) {
747 /* first component not at start of result,
748 but we have full component in buffer. */
749 u_memmove(dest, dest+(haveLang?langPos:restPos), length);
750 } else {
751 /* would have fit, but didn't because of pattern prefix. */
752 sub0Pos=0; /* stops initial padding (and a second retry,
753 so we won't end up here again) */
754 retry=TRUE;
755 }
756 }
757 }
758
759 ++subi; /* move on to next substitution */
760 }
761 }
762 } while(retry);
763
764 return u_terminateUChars(dest, destCapacity, length, pErrorCode);
765 }
766
767 U_CAPI int32_t U_EXPORT2
uloc_getDisplayKeyword(const char * keyword,const char * displayLocale,UChar * dest,int32_t destCapacity,UErrorCode * status)768 uloc_getDisplayKeyword(const char* keyword,
769 const char* displayLocale,
770 UChar* dest,
771 int32_t destCapacity,
772 UErrorCode* status){
773
774 /* argument checking */
775 if(status==NULL || U_FAILURE(*status)) {
776 return 0;
777 }
778
779 if(destCapacity<0 || (destCapacity>0 && dest==NULL)) {
780 *status=U_ILLEGAL_ARGUMENT_ERROR;
781 return 0;
782 }
783
784
785 /* pass itemKey=NULL to look for a top-level item */
786 return _getStringOrCopyKey(U_ICUDATA_LANG, displayLocale,
787 _kKeys, NULL,
788 keyword,
789 keyword,
790 dest, destCapacity,
791 status);
792
793 }
794
795
796 #define UCURRENCY_DISPLAY_NAME_INDEX 1
797
798 U_CAPI int32_t U_EXPORT2
uloc_getDisplayKeywordValue(const char * locale,const char * keyword,const char * displayLocale,UChar * dest,int32_t destCapacity,UErrorCode * status)799 uloc_getDisplayKeywordValue( const char* locale,
800 const char* keyword,
801 const char* displayLocale,
802 UChar* dest,
803 int32_t destCapacity,
804 UErrorCode* status){
805
806
807 char keywordValue[ULOC_FULLNAME_CAPACITY*4];
808 int32_t capacity = ULOC_FULLNAME_CAPACITY*4;
809 int32_t keywordValueLen =0;
810
811 /* argument checking */
812 if(status==NULL || U_FAILURE(*status)) {
813 return 0;
814 }
815
816 if(destCapacity<0 || (destCapacity>0 && dest==NULL)) {
817 *status=U_ILLEGAL_ARGUMENT_ERROR;
818 return 0;
819 }
820
821 /* get the keyword value */
822 keywordValue[0]=0;
823 keywordValueLen = uloc_getKeywordValue(locale, keyword, keywordValue, capacity, status);
824
825 /*
826 * if the keyword is equal to currency .. then to get the display name
827 * we need to do the fallback ourselves
828 */
829 if(uprv_stricmp(keyword, _kCurrency)==0){
830
831 int32_t dispNameLen = 0;
832 const UChar *dispName = NULL;
833
834 UResourceBundle *bundle = ures_open(U_ICUDATA_CURR, displayLocale, status);
835 UResourceBundle *currencies = ures_getByKey(bundle, _kCurrencies, NULL, status);
836 UResourceBundle *currency = ures_getByKeyWithFallback(currencies, keywordValue, NULL, status);
837
838 dispName = ures_getStringByIndex(currency, UCURRENCY_DISPLAY_NAME_INDEX, &dispNameLen, status);
839
840 /*close the bundles */
841 ures_close(currency);
842 ures_close(currencies);
843 ures_close(bundle);
844
845 if(U_FAILURE(*status)){
846 if(*status == U_MISSING_RESOURCE_ERROR){
847 /* we just want to write the value over if nothing is available */
848 *status = U_USING_DEFAULT_WARNING;
849 }else{
850 return 0;
851 }
852 }
853
854 /* now copy the dispName over if not NULL */
855 if(dispName != NULL){
856 if(dispNameLen <= destCapacity){
857 u_memcpy(dest, dispName, dispNameLen);
858 return u_terminateUChars(dest, destCapacity, dispNameLen, status);
859 }else{
860 *status = U_BUFFER_OVERFLOW_ERROR;
861 return dispNameLen;
862 }
863 }else{
864 /* we have not found the display name for the value .. just copy over */
865 if(keywordValueLen <= destCapacity){
866 u_charsToUChars(keywordValue, dest, keywordValueLen);
867 return u_terminateUChars(dest, destCapacity, keywordValueLen, status);
868 }else{
869 *status = U_BUFFER_OVERFLOW_ERROR;
870 return keywordValueLen;
871 }
872 }
873
874
875 }else{
876
877 return _getStringOrCopyKey(U_ICUDATA_LANG, displayLocale,
878 _kTypes, keyword,
879 keywordValue,
880 keywordValue,
881 dest, destCapacity,
882 status);
883 }
884 }
885