1 /*
2 *******************************************************************************
3 * Copyright (C) 2010-2013, International Business Machines Corporation and
4 * others. All Rights Reserved.
5 *******************************************************************************
6 */
7
8 #include "unicode/utypes.h"
9
10 #if !UCONFIG_NO_FORMATTING
11
12 #include "unicode/locdspnm.h"
13 #include "unicode/msgfmt.h"
14 #include "unicode/ures.h"
15 #include "unicode/brkiter.h"
16
17 #include "cmemory.h"
18 #include "cstring.h"
19 #include "ulocimp.h"
20 #include "ureslocs.h"
21 #include "uresimp.h"
22
23 #include <stdarg.h>
24
25 /**
26 * Concatenate a number of null-terminated strings to buffer, leaving a
27 * null-terminated string. The last argument should be the null pointer.
28 * Return the length of the string in the buffer, not counting the trailing
29 * null. Return -1 if there is an error (buffer is null, or buflen < 1).
30 */
ncat(char * buffer,uint32_t buflen,...)31 static int32_t ncat(char *buffer, uint32_t buflen, ...) {
32 va_list args;
33 char *str;
34 char *p = buffer;
35 const char* e = buffer + buflen - 1;
36
37 if (buffer == NULL || buflen < 1) {
38 return -1;
39 }
40
41 va_start(args, buflen);
42 while ((str = va_arg(args, char *))) {
43 char c;
44 while (p != e && (c = *str++)) {
45 *p++ = c;
46 }
47 }
48 *p = 0;
49 va_end(args);
50
51 return p - buffer;
52 }
53
54 U_NAMESPACE_BEGIN
55
56 ////////////////////////////////////////////////////////////////////////////////////////////////////
57
58 // Access resource data for locale components.
59 // Wrap code in uloc.c for now.
60 class ICUDataTable {
61 const char* path;
62 Locale locale;
63
64 public:
65 ICUDataTable(const char* path, const Locale& locale);
66 ~ICUDataTable();
67
68 const Locale& getLocale();
69
70 UnicodeString& get(const char* tableKey, const char* itemKey,
71 UnicodeString& result) const;
72 UnicodeString& get(const char* tableKey, const char* subTableKey, const char* itemKey,
73 UnicodeString& result) const;
74
75 UnicodeString& getNoFallback(const char* tableKey, const char* itemKey,
76 UnicodeString &result) const;
77 UnicodeString& getNoFallback(const char* tableKey, const char* subTableKey, const char* itemKey,
78 UnicodeString &result) const;
79 };
80
81 inline UnicodeString &
get(const char * tableKey,const char * itemKey,UnicodeString & result) const82 ICUDataTable::get(const char* tableKey, const char* itemKey, UnicodeString& result) const {
83 return get(tableKey, NULL, itemKey, result);
84 }
85
86 inline UnicodeString &
getNoFallback(const char * tableKey,const char * itemKey,UnicodeString & result) const87 ICUDataTable::getNoFallback(const char* tableKey, const char* itemKey, UnicodeString& result) const {
88 return getNoFallback(tableKey, NULL, itemKey, result);
89 }
90
ICUDataTable(const char * path,const Locale & locale)91 ICUDataTable::ICUDataTable(const char* path, const Locale& locale)
92 : path(NULL), locale(Locale::getRoot())
93 {
94 if (path) {
95 int32_t len = uprv_strlen(path);
96 this->path = (const char*) uprv_malloc(len + 1);
97 if (this->path) {
98 uprv_strcpy((char *)this->path, path);
99 this->locale = locale;
100 }
101 }
102 }
103
~ICUDataTable()104 ICUDataTable::~ICUDataTable() {
105 if (path) {
106 uprv_free((void*) path);
107 path = NULL;
108 }
109 }
110
111 const Locale&
getLocale()112 ICUDataTable::getLocale() {
113 return locale;
114 }
115
116 UnicodeString &
get(const char * tableKey,const char * subTableKey,const char * itemKey,UnicodeString & result) const117 ICUDataTable::get(const char* tableKey, const char* subTableKey, const char* itemKey,
118 UnicodeString &result) const {
119 UErrorCode status = U_ZERO_ERROR;
120 int32_t len = 0;
121
122 const UChar *s = uloc_getTableStringWithFallback(path, locale.getName(),
123 tableKey, subTableKey, itemKey,
124 &len, &status);
125 if (U_SUCCESS(status) && len > 0) {
126 return result.setTo(s, len);
127 }
128 return result.setTo(UnicodeString(itemKey, -1, US_INV));
129 }
130
131 UnicodeString &
getNoFallback(const char * tableKey,const char * subTableKey,const char * itemKey,UnicodeString & result) const132 ICUDataTable::getNoFallback(const char* tableKey, const char* subTableKey, const char* itemKey,
133 UnicodeString& result) const {
134 UErrorCode status = U_ZERO_ERROR;
135 int32_t len = 0;
136
137 const UChar *s = uloc_getTableStringWithFallback(path, locale.getName(),
138 tableKey, subTableKey, itemKey,
139 &len, &status);
140 if (U_SUCCESS(status)) {
141 return result.setTo(s, len);
142 }
143
144 result.setToBogus();
145 return result;
146 }
147
148 ////////////////////////////////////////////////////////////////////////////////////////////////////
149
~LocaleDisplayNames()150 LocaleDisplayNames::~LocaleDisplayNames() {}
151
152 ////////////////////////////////////////////////////////////////////////////////////////////////////
153
154 #if 0 // currently unused
155
156 class DefaultLocaleDisplayNames : public LocaleDisplayNames {
157 UDialectHandling dialectHandling;
158
159 public:
160 // constructor
161 DefaultLocaleDisplayNames(UDialectHandling dialectHandling);
162
163 virtual ~DefaultLocaleDisplayNames();
164
165 virtual const Locale& getLocale() const;
166 virtual UDialectHandling getDialectHandling() const;
167
168 virtual UnicodeString& localeDisplayName(const Locale& locale,
169 UnicodeString& result) const;
170 virtual UnicodeString& localeDisplayName(const char* localeId,
171 UnicodeString& result) const;
172 virtual UnicodeString& languageDisplayName(const char* lang,
173 UnicodeString& result) const;
174 virtual UnicodeString& scriptDisplayName(const char* script,
175 UnicodeString& result) const;
176 virtual UnicodeString& scriptDisplayName(UScriptCode scriptCode,
177 UnicodeString& result) const;
178 virtual UnicodeString& regionDisplayName(const char* region,
179 UnicodeString& result) const;
180 virtual UnicodeString& variantDisplayName(const char* variant,
181 UnicodeString& result) const;
182 virtual UnicodeString& keyDisplayName(const char* key,
183 UnicodeString& result) const;
184 virtual UnicodeString& keyValueDisplayName(const char* key,
185 const char* value,
186 UnicodeString& result) const;
187 };
188
189 DefaultLocaleDisplayNames::DefaultLocaleDisplayNames(UDialectHandling dialectHandling)
190 : dialectHandling(dialectHandling) {
191 }
192
193 DefaultLocaleDisplayNames::~DefaultLocaleDisplayNames() {
194 }
195
196 const Locale&
197 DefaultLocaleDisplayNames::getLocale() const {
198 return Locale::getRoot();
199 }
200
201 UDialectHandling
202 DefaultLocaleDisplayNames::getDialectHandling() const {
203 return dialectHandling;
204 }
205
206 UnicodeString&
207 DefaultLocaleDisplayNames::localeDisplayName(const Locale& locale,
208 UnicodeString& result) const {
209 return result = UnicodeString(locale.getName(), -1, US_INV);
210 }
211
212 UnicodeString&
213 DefaultLocaleDisplayNames::localeDisplayName(const char* localeId,
214 UnicodeString& result) const {
215 return result = UnicodeString(localeId, -1, US_INV);
216 }
217
218 UnicodeString&
219 DefaultLocaleDisplayNames::languageDisplayName(const char* lang,
220 UnicodeString& result) const {
221 return result = UnicodeString(lang, -1, US_INV);
222 }
223
224 UnicodeString&
225 DefaultLocaleDisplayNames::scriptDisplayName(const char* script,
226 UnicodeString& result) const {
227 return result = UnicodeString(script, -1, US_INV);
228 }
229
230 UnicodeString&
231 DefaultLocaleDisplayNames::scriptDisplayName(UScriptCode scriptCode,
232 UnicodeString& result) const {
233 const char* name = uscript_getName(scriptCode);
234 if (name) {
235 return result = UnicodeString(name, -1, US_INV);
236 }
237 return result.remove();
238 }
239
240 UnicodeString&
241 DefaultLocaleDisplayNames::regionDisplayName(const char* region,
242 UnicodeString& result) const {
243 return result = UnicodeString(region, -1, US_INV);
244 }
245
246 UnicodeString&
247 DefaultLocaleDisplayNames::variantDisplayName(const char* variant,
248 UnicodeString& result) const {
249 return result = UnicodeString(variant, -1, US_INV);
250 }
251
252 UnicodeString&
253 DefaultLocaleDisplayNames::keyDisplayName(const char* key,
254 UnicodeString& result) const {
255 return result = UnicodeString(key, -1, US_INV);
256 }
257
258 UnicodeString&
259 DefaultLocaleDisplayNames::keyValueDisplayName(const char* /* key */,
260 const char* value,
261 UnicodeString& result) const {
262 return result = UnicodeString(value, -1, US_INV);
263 }
264
265 #endif // currently unused class DefaultLocaleDisplayNames
266
267 ////////////////////////////////////////////////////////////////////////////////////////////////////
268
269 class LocaleDisplayNamesImpl : public LocaleDisplayNames {
270 Locale locale;
271 UDialectHandling dialectHandling;
272 ICUDataTable langData;
273 ICUDataTable regionData;
274 MessageFormat *separatorFormat;
275 MessageFormat *format;
276 MessageFormat *keyTypeFormat;
277 UDisplayContext capitalizationContext;
278 UnicodeString formatOpenParen;
279 UnicodeString formatReplaceOpenParen;
280 UnicodeString formatCloseParen;
281 UnicodeString formatReplaceCloseParen;
282
283 // Constants for capitalization context usage types.
284 enum CapContextUsage {
285 kCapContextUsageLanguage,
286 kCapContextUsageScript,
287 kCapContextUsageTerritory,
288 kCapContextUsageVariant,
289 kCapContextUsageKey,
290 kCapContextUsageType,
291 kCapContextUsageCount
292 };
293 // Capitalization transforms. For each usage type, the first array element indicates
294 // whether to titlecase for uiListOrMenu context, the second indicates whether to
295 // titlecase for stand-alone context.
296 UBool fCapitalization[kCapContextUsageCount][2];
297
298 public:
299 // constructor
300 LocaleDisplayNamesImpl(const Locale& locale, UDialectHandling dialectHandling);
301 LocaleDisplayNamesImpl(const Locale& locale, UDisplayContext *contexts, int32_t length);
302 virtual ~LocaleDisplayNamesImpl();
303
304 virtual const Locale& getLocale() const;
305 virtual UDialectHandling getDialectHandling() const;
306 virtual UDisplayContext getContext(UDisplayContextType type) const;
307
308 virtual UnicodeString& localeDisplayName(const Locale& locale,
309 UnicodeString& result) const;
310 virtual UnicodeString& localeDisplayName(const char* localeId,
311 UnicodeString& result) const;
312 virtual UnicodeString& languageDisplayName(const char* lang,
313 UnicodeString& result) const;
314 virtual UnicodeString& scriptDisplayName(const char* script,
315 UnicodeString& result) const;
316 virtual UnicodeString& scriptDisplayName(UScriptCode scriptCode,
317 UnicodeString& result) const;
318 virtual UnicodeString& regionDisplayName(const char* region,
319 UnicodeString& result) const;
320 virtual UnicodeString& variantDisplayName(const char* variant,
321 UnicodeString& result) const;
322 virtual UnicodeString& keyDisplayName(const char* key,
323 UnicodeString& result) const;
324 virtual UnicodeString& keyValueDisplayName(const char* key,
325 const char* value,
326 UnicodeString& result) const;
327 private:
328 UnicodeString& localeIdName(const char* localeId,
329 UnicodeString& result) const;
330 UnicodeString& appendWithSep(UnicodeString& buffer, const UnicodeString& src) const;
331 UnicodeString& adjustForUsageAndContext(CapContextUsage usage, UnicodeString& result) const;
332 void initialize(void);
333 };
334
LocaleDisplayNamesImpl(const Locale & locale,UDialectHandling dialectHandling)335 LocaleDisplayNamesImpl::LocaleDisplayNamesImpl(const Locale& locale,
336 UDialectHandling dialectHandling)
337 : dialectHandling(dialectHandling)
338 , langData(U_ICUDATA_LANG, locale)
339 , regionData(U_ICUDATA_REGION, locale)
340 , separatorFormat(NULL)
341 , format(NULL)
342 , keyTypeFormat(NULL)
343 , capitalizationContext(UDISPCTX_CAPITALIZATION_NONE)
344 {
345 initialize();
346 }
347
LocaleDisplayNamesImpl(const Locale & locale,UDisplayContext * contexts,int32_t length)348 LocaleDisplayNamesImpl::LocaleDisplayNamesImpl(const Locale& locale,
349 UDisplayContext *contexts, int32_t length)
350 : dialectHandling(ULDN_STANDARD_NAMES)
351 , langData(U_ICUDATA_LANG, locale)
352 , regionData(U_ICUDATA_REGION, locale)
353 , separatorFormat(NULL)
354 , format(NULL)
355 , keyTypeFormat(NULL)
356 , capitalizationContext(UDISPCTX_CAPITALIZATION_NONE)
357 {
358 while (length-- > 0) {
359 UDisplayContext value = *contexts++;
360 UDisplayContextType selector = (UDisplayContextType)((uint32_t)value >> 8);
361 switch (selector) {
362 case UDISPCTX_TYPE_DIALECT_HANDLING:
363 dialectHandling = (UDialectHandling)value;
364 break;
365 case UDISPCTX_TYPE_CAPITALIZATION:
366 capitalizationContext = value;
367 break;
368 default:
369 break;
370 }
371 }
372 initialize();
373 }
374
375 void
initialize(void)376 LocaleDisplayNamesImpl::initialize(void) {
377 LocaleDisplayNamesImpl *nonConstThis = (LocaleDisplayNamesImpl *)this;
378 nonConstThis->locale = langData.getLocale() == Locale::getRoot()
379 ? regionData.getLocale()
380 : langData.getLocale();
381
382 UnicodeString sep;
383 langData.getNoFallback("localeDisplayPattern", "separator", sep);
384 if (sep.isBogus()) {
385 sep = UnicodeString("{0}, {1}", -1, US_INV);
386 }
387 UErrorCode status = U_ZERO_ERROR;
388 separatorFormat = new MessageFormat(sep, status);
389
390 UnicodeString pattern;
391 langData.getNoFallback("localeDisplayPattern", "pattern", pattern);
392 if (pattern.isBogus()) {
393 pattern = UnicodeString("{0} ({1})", -1, US_INV);
394 }
395 format = new MessageFormat(pattern, status);
396 if (pattern.indexOf((UChar)0xFF08) >= 0) {
397 formatOpenParen.setTo((UChar)0xFF08); // fullwidth (
398 formatReplaceOpenParen.setTo((UChar)0xFF3B); // fullwidth [
399 formatCloseParen.setTo((UChar)0xFF09); // fullwidth )
400 formatReplaceCloseParen.setTo((UChar)0xFF3D); // fullwidth ]
401 } else {
402 formatOpenParen.setTo((UChar)0x0028); // (
403 formatReplaceOpenParen.setTo((UChar)0x005B); // [
404 formatCloseParen.setTo((UChar)0x0029); // )
405 formatReplaceCloseParen.setTo((UChar)0x005D); // ]
406 }
407
408 UnicodeString ktPattern;
409 langData.get("localeDisplayPattern", "keyTypePattern", ktPattern);
410 if (ktPattern.isBogus()) {
411 ktPattern = UnicodeString("{0}={1}", -1, US_INV);
412 }
413 keyTypeFormat = new MessageFormat(ktPattern, status);
414
415 uprv_memset(fCapitalization, 0, sizeof(fCapitalization));
416 #if !UCONFIG_NO_BREAK_ITERATION
417 // The following is basically copied from DateFormatSymbols::initializeData
418 typedef struct {
419 const char * usageName;
420 LocaleDisplayNamesImpl::CapContextUsage usageEnum;
421 } ContextUsageNameToEnum;
422 const ContextUsageNameToEnum contextUsageTypeMap[] = {
423 // Entries must be sorted by usageTypeName; entry with NULL name terminates list.
424 { "key", kCapContextUsageKey },
425 { "languages", kCapContextUsageLanguage },
426 { "script", kCapContextUsageScript },
427 { "territory", kCapContextUsageTerritory },
428 { "type", kCapContextUsageType },
429 { "variant", kCapContextUsageVariant },
430 { NULL, (CapContextUsage)0 },
431 };
432 int32_t len = 0;
433 UResourceBundle *localeBundle = ures_open(NULL, locale.getName(), &status);
434 if (U_SUCCESS(status)) {
435 UResourceBundle *contextTransforms = ures_getByKeyWithFallback(localeBundle, "contextTransforms", NULL, &status);
436 if (U_SUCCESS(status)) {
437 UResourceBundle *contextTransformUsage;
438 while ( (contextTransformUsage = ures_getNextResource(contextTransforms, NULL, &status)) != NULL ) {
439 const int32_t * intVector = ures_getIntVector(contextTransformUsage, &len, &status);
440 if (U_SUCCESS(status) && intVector != NULL && len >= 2) {
441 const char* usageKey = ures_getKey(contextTransformUsage);
442 if (usageKey != NULL) {
443 const ContextUsageNameToEnum * typeMapPtr = contextUsageTypeMap;
444 int32_t compResult = 0;
445 // linear search; list is short and we cannot be sure that bsearch is available
446 while ( typeMapPtr->usageName != NULL && (compResult = uprv_strcmp(usageKey, typeMapPtr->usageName)) > 0 ) {
447 ++typeMapPtr;
448 }
449 if (typeMapPtr->usageName != NULL && compResult == 0) {
450 fCapitalization[typeMapPtr->usageEnum][0] = intVector[0];
451 fCapitalization[typeMapPtr->usageEnum][1] = intVector[1];
452 }
453 }
454 }
455 status = U_ZERO_ERROR;
456 ures_close(contextTransformUsage);
457 }
458 ures_close(contextTransforms);
459 }
460 ures_close(localeBundle);
461 }
462 #endif
463 }
464
~LocaleDisplayNamesImpl()465 LocaleDisplayNamesImpl::~LocaleDisplayNamesImpl() {
466 delete separatorFormat;
467 delete format;
468 delete keyTypeFormat;
469 }
470
471 const Locale&
getLocale() const472 LocaleDisplayNamesImpl::getLocale() const {
473 return locale;
474 }
475
476 UDialectHandling
getDialectHandling() const477 LocaleDisplayNamesImpl::getDialectHandling() const {
478 return dialectHandling;
479 }
480
481 UDisplayContext
getContext(UDisplayContextType type) const482 LocaleDisplayNamesImpl::getContext(UDisplayContextType type) const {
483 switch (type) {
484 case UDISPCTX_TYPE_DIALECT_HANDLING:
485 return (UDisplayContext)dialectHandling;
486 case UDISPCTX_TYPE_CAPITALIZATION:
487 return capitalizationContext;
488 default:
489 break;
490 }
491 return (UDisplayContext)0;
492 }
493
494 UnicodeString&
adjustForUsageAndContext(CapContextUsage usage,UnicodeString & result) const495 LocaleDisplayNamesImpl::adjustForUsageAndContext(CapContextUsage usage,
496 UnicodeString& result) const {
497 #if !UCONFIG_NO_BREAK_ITERATION
498 // check to see whether we need to titlecase result
499 UBool titlecase = FALSE;
500 switch (capitalizationContext) {
501 case UDISPCTX_CAPITALIZATION_FOR_BEGINNING_OF_SENTENCE:
502 titlecase = TRUE;
503 break;
504 case UDISPCTX_CAPITALIZATION_FOR_UI_LIST_OR_MENU:
505 titlecase = fCapitalization[usage][0];
506 break;
507 case UDISPCTX_CAPITALIZATION_FOR_STANDALONE:
508 titlecase = fCapitalization[usage][1];
509 break;
510 default:
511 // titlecase = FALSE;
512 break;
513 }
514 if (titlecase) {
515 // TODO: Fix this titlecase hack when we figure out something better to do.
516 // We don't want to titlecase the whole text, only something like the first word,
517 // of the first segment long enough to have a complete cluster, whichever is
518 // shorter. We could have keep a word break iterator around, but I am not sure
519 // that will do the ight thing for the purposes here. For now we assume that in
520 // languages for which titlecasing makes a difference, we can stop at non-letter
521 // characters in 0x0000-0x00FF and only titlecase up to the first occurrence of
522 // any of those, or to a small number of chars, whichever comes first.
523 int32_t stopPos, stopPosLimit = 8, len = result.length();
524 if ( stopPosLimit > len ) {
525 stopPosLimit = len;
526 }
527 for ( stopPos = 0; stopPos < stopPosLimit; stopPos++ ) {
528 UChar32 ch = result.char32At(stopPos);
529 if ( (ch < 0x41) || (ch > 0x5A && ch < 0x61) || (ch > 0x7A && ch < 0xC0) ) {
530 break;
531 }
532 if (ch >= 0x10000) {
533 stopPos++;
534 }
535 }
536 if ( stopPos > 0 && stopPos < len ) {
537 UnicodeString firstWord(result, 0, stopPos);
538 firstWord.toTitle(NULL, locale, U_TITLECASE_NO_LOWERCASE | U_TITLECASE_NO_BREAK_ADJUSTMENT);
539 result.replaceBetween(0, stopPos, firstWord);
540 } else {
541 // no stopPos, titlecase the whole text
542 result.toTitle(NULL, locale, U_TITLECASE_NO_LOWERCASE | U_TITLECASE_NO_BREAK_ADJUSTMENT);
543 }
544 }
545 #endif
546 return result;
547 }
548
549 UnicodeString&
localeDisplayName(const Locale & locale,UnicodeString & result) const550 LocaleDisplayNamesImpl::localeDisplayName(const Locale& locale,
551 UnicodeString& result) const {
552 UnicodeString resultName;
553
554 const char* lang = locale.getLanguage();
555 if (uprv_strlen(lang) == 0) {
556 lang = "root";
557 }
558 const char* script = locale.getScript();
559 const char* country = locale.getCountry();
560 const char* variant = locale.getVariant();
561
562 UBool hasScript = uprv_strlen(script) > 0;
563 UBool hasCountry = uprv_strlen(country) > 0;
564 UBool hasVariant = uprv_strlen(variant) > 0;
565
566 if (dialectHandling == ULDN_DIALECT_NAMES) {
567 char buffer[ULOC_FULLNAME_CAPACITY];
568 do { // loop construct is so we can break early out of search
569 if (hasScript && hasCountry) {
570 ncat(buffer, ULOC_FULLNAME_CAPACITY, lang, "_", script, "_", country, (char *)0);
571 localeIdName(buffer, resultName);
572 if (!resultName.isBogus()) {
573 hasScript = FALSE;
574 hasCountry = FALSE;
575 break;
576 }
577 }
578 if (hasScript) {
579 ncat(buffer, ULOC_FULLNAME_CAPACITY, lang, "_", script, (char *)0);
580 localeIdName(buffer, resultName);
581 if (!resultName.isBogus()) {
582 hasScript = FALSE;
583 break;
584 }
585 }
586 if (hasCountry) {
587 ncat(buffer, ULOC_FULLNAME_CAPACITY, lang, "_", country, (char*)0);
588 localeIdName(buffer, resultName);
589 if (!resultName.isBogus()) {
590 hasCountry = FALSE;
591 break;
592 }
593 }
594 } while (FALSE);
595 }
596 if (resultName.isBogus() || resultName.isEmpty()) {
597 localeIdName(lang, resultName);
598 }
599
600 UnicodeString resultRemainder;
601 UnicodeString temp;
602 StringEnumeration *e = NULL;
603 UErrorCode status = U_ZERO_ERROR;
604
605 if (hasScript) {
606 resultRemainder.append(scriptDisplayName(script, temp));
607 }
608 if (hasCountry) {
609 appendWithSep(resultRemainder, regionDisplayName(country, temp));
610 }
611 if (hasVariant) {
612 appendWithSep(resultRemainder, variantDisplayName(variant, temp));
613 }
614 resultRemainder.findAndReplace(formatOpenParen, formatReplaceOpenParen);
615 resultRemainder.findAndReplace(formatCloseParen, formatReplaceCloseParen);
616
617 e = locale.createKeywords(status);
618 if (e && U_SUCCESS(status)) {
619 UnicodeString temp2;
620 char value[ULOC_KEYWORD_AND_VALUES_CAPACITY]; // sigh, no ULOC_VALUE_CAPACITY
621 const char* key;
622 while ((key = e->next((int32_t *)0, status)) != NULL) {
623 locale.getKeywordValue(key, value, ULOC_KEYWORD_AND_VALUES_CAPACITY, status);
624 keyDisplayName(key, temp);
625 temp.findAndReplace(formatOpenParen, formatReplaceOpenParen);
626 temp.findAndReplace(formatCloseParen, formatReplaceCloseParen);
627 keyValueDisplayName(key, value, temp2);
628 temp2.findAndReplace(formatOpenParen, formatReplaceOpenParen);
629 temp2.findAndReplace(formatCloseParen, formatReplaceCloseParen);
630 if (temp2 != UnicodeString(value, -1, US_INV)) {
631 appendWithSep(resultRemainder, temp2);
632 } else if (temp != UnicodeString(key, -1, US_INV)) {
633 UnicodeString temp3;
634 Formattable data[] = {
635 temp,
636 temp2
637 };
638 FieldPosition fpos;
639 status = U_ZERO_ERROR;
640 keyTypeFormat->format(data, 2, temp3, fpos, status);
641 appendWithSep(resultRemainder, temp3);
642 } else {
643 appendWithSep(resultRemainder, temp)
644 .append((UChar)0x3d /* = */)
645 .append(temp2);
646 }
647 }
648 delete e;
649 }
650
651 if (!resultRemainder.isEmpty()) {
652 Formattable data[] = {
653 resultName,
654 resultRemainder
655 };
656 FieldPosition fpos;
657 status = U_ZERO_ERROR;
658 format->format(data, 2, result, fpos, status);
659 return adjustForUsageAndContext(kCapContextUsageLanguage, result);
660 }
661
662 result = resultName;
663 return adjustForUsageAndContext(kCapContextUsageLanguage, result);
664 }
665
666 UnicodeString&
appendWithSep(UnicodeString & buffer,const UnicodeString & src) const667 LocaleDisplayNamesImpl::appendWithSep(UnicodeString& buffer, const UnicodeString& src) const {
668 if (buffer.isEmpty()) {
669 buffer.setTo(src);
670 } else {
671 UnicodeString combined;
672 Formattable data[] = {
673 buffer,
674 src
675 };
676 FieldPosition fpos;
677 UErrorCode status = U_ZERO_ERROR;
678 separatorFormat->format(data, 2, combined, fpos, status);
679 if (U_SUCCESS(status)) {
680 buffer.setTo(combined);
681 }
682 }
683 return buffer;
684 }
685
686 UnicodeString&
localeDisplayName(const char * localeId,UnicodeString & result) const687 LocaleDisplayNamesImpl::localeDisplayName(const char* localeId,
688 UnicodeString& result) const {
689 return localeDisplayName(Locale(localeId), result);
690 }
691
692 // private
693 UnicodeString&
localeIdName(const char * localeId,UnicodeString & result) const694 LocaleDisplayNamesImpl::localeIdName(const char* localeId,
695 UnicodeString& result) const {
696 return langData.getNoFallback("Languages", localeId, result);
697 }
698
699 UnicodeString&
languageDisplayName(const char * lang,UnicodeString & result) const700 LocaleDisplayNamesImpl::languageDisplayName(const char* lang,
701 UnicodeString& result) const {
702 if (uprv_strcmp("root", lang) == 0 || uprv_strchr(lang, '_') != NULL) {
703 return result = UnicodeString(lang, -1, US_INV);
704 }
705 langData.get("Languages", lang, result);
706 return adjustForUsageAndContext(kCapContextUsageLanguage, result);
707 }
708
709 UnicodeString&
scriptDisplayName(const char * script,UnicodeString & result) const710 LocaleDisplayNamesImpl::scriptDisplayName(const char* script,
711 UnicodeString& result) const {
712 langData.get("Scripts", script, result);
713 return adjustForUsageAndContext(kCapContextUsageScript, result);
714 }
715
716 UnicodeString&
scriptDisplayName(UScriptCode scriptCode,UnicodeString & result) const717 LocaleDisplayNamesImpl::scriptDisplayName(UScriptCode scriptCode,
718 UnicodeString& result) const {
719 const char* name = uscript_getName(scriptCode);
720 langData.get("Scripts", name, result);
721 return adjustForUsageAndContext(kCapContextUsageScript, result);
722 }
723
724 UnicodeString&
regionDisplayName(const char * region,UnicodeString & result) const725 LocaleDisplayNamesImpl::regionDisplayName(const char* region,
726 UnicodeString& result) const {
727 regionData.get("Countries", region, result);
728 return adjustForUsageAndContext(kCapContextUsageTerritory, result);
729 }
730
731 UnicodeString&
variantDisplayName(const char * variant,UnicodeString & result) const732 LocaleDisplayNamesImpl::variantDisplayName(const char* variant,
733 UnicodeString& result) const {
734 langData.get("Variants", variant, result);
735 return adjustForUsageAndContext(kCapContextUsageVariant, result);
736 }
737
738 UnicodeString&
keyDisplayName(const char * key,UnicodeString & result) const739 LocaleDisplayNamesImpl::keyDisplayName(const char* key,
740 UnicodeString& result) const {
741 langData.get("Keys", key, result);
742 return adjustForUsageAndContext(kCapContextUsageKey, result);
743 }
744
745 UnicodeString&
keyValueDisplayName(const char * key,const char * value,UnicodeString & result) const746 LocaleDisplayNamesImpl::keyValueDisplayName(const char* key,
747 const char* value,
748 UnicodeString& result) const {
749 langData.get("Types", key, value, result);
750 return adjustForUsageAndContext(kCapContextUsageType, result);
751 }
752
753 ////////////////////////////////////////////////////////////////////////////////////////////////////
754
755 LocaleDisplayNames*
createInstance(const Locale & locale,UDialectHandling dialectHandling)756 LocaleDisplayNames::createInstance(const Locale& locale,
757 UDialectHandling dialectHandling) {
758 return new LocaleDisplayNamesImpl(locale, dialectHandling);
759 }
760
761 LocaleDisplayNames*
createInstance(const Locale & locale,UDisplayContext * contexts,int32_t length)762 LocaleDisplayNames::createInstance(const Locale& locale,
763 UDisplayContext *contexts, int32_t length) {
764 if (contexts == NULL) {
765 length = 0;
766 }
767 return new LocaleDisplayNamesImpl(locale, contexts, length);
768 }
769
770 U_NAMESPACE_END
771
772 ////////////////////////////////////////////////////////////////////////////////////////////////////
773
774 U_NAMESPACE_USE
775
776 U_CAPI ULocaleDisplayNames * U_EXPORT2
uldn_open(const char * locale,UDialectHandling dialectHandling,UErrorCode * pErrorCode)777 uldn_open(const char * locale,
778 UDialectHandling dialectHandling,
779 UErrorCode *pErrorCode) {
780 if (U_FAILURE(*pErrorCode)) {
781 return 0;
782 }
783 if (locale == NULL) {
784 locale = uloc_getDefault();
785 }
786 return (ULocaleDisplayNames *)LocaleDisplayNames::createInstance(Locale(locale), dialectHandling);
787 }
788
789 U_CAPI ULocaleDisplayNames * U_EXPORT2
uldn_openForContext(const char * locale,UDisplayContext * contexts,int32_t length,UErrorCode * pErrorCode)790 uldn_openForContext(const char * locale,
791 UDisplayContext *contexts, int32_t length,
792 UErrorCode *pErrorCode) {
793 if (U_FAILURE(*pErrorCode)) {
794 return 0;
795 }
796 if (locale == NULL) {
797 locale = uloc_getDefault();
798 }
799 return (ULocaleDisplayNames *)LocaleDisplayNames::createInstance(Locale(locale), contexts, length);
800 }
801
802
803 U_CAPI void U_EXPORT2
uldn_close(ULocaleDisplayNames * ldn)804 uldn_close(ULocaleDisplayNames *ldn) {
805 delete (LocaleDisplayNames *)ldn;
806 }
807
808 U_CAPI const char * U_EXPORT2
uldn_getLocale(const ULocaleDisplayNames * ldn)809 uldn_getLocale(const ULocaleDisplayNames *ldn) {
810 if (ldn) {
811 return ((const LocaleDisplayNames *)ldn)->getLocale().getName();
812 }
813 return NULL;
814 }
815
816 U_CAPI UDialectHandling U_EXPORT2
uldn_getDialectHandling(const ULocaleDisplayNames * ldn)817 uldn_getDialectHandling(const ULocaleDisplayNames *ldn) {
818 if (ldn) {
819 return ((const LocaleDisplayNames *)ldn)->getDialectHandling();
820 }
821 return ULDN_STANDARD_NAMES;
822 }
823
824 U_CAPI UDisplayContext U_EXPORT2
uldn_getContext(const ULocaleDisplayNames * ldn,UDisplayContextType type,UErrorCode * pErrorCode)825 uldn_getContext(const ULocaleDisplayNames *ldn,
826 UDisplayContextType type,
827 UErrorCode *pErrorCode) {
828 if (U_FAILURE(*pErrorCode)) {
829 return (UDisplayContext)0;
830 }
831 return ((const LocaleDisplayNames *)ldn)->getContext(type);
832 }
833
834 U_CAPI int32_t U_EXPORT2
uldn_localeDisplayName(const ULocaleDisplayNames * ldn,const char * locale,UChar * result,int32_t maxResultSize,UErrorCode * pErrorCode)835 uldn_localeDisplayName(const ULocaleDisplayNames *ldn,
836 const char *locale,
837 UChar *result,
838 int32_t maxResultSize,
839 UErrorCode *pErrorCode) {
840 if (U_FAILURE(*pErrorCode)) {
841 return 0;
842 }
843 if (ldn == NULL || locale == NULL || (result == NULL && maxResultSize > 0) || maxResultSize < 0) {
844 *pErrorCode = U_ILLEGAL_ARGUMENT_ERROR;
845 return 0;
846 }
847 UnicodeString temp(result, 0, maxResultSize);
848 ((const LocaleDisplayNames *)ldn)->localeDisplayName(locale, temp);
849 return temp.extract(result, maxResultSize, *pErrorCode);
850 }
851
852 U_CAPI int32_t U_EXPORT2
uldn_languageDisplayName(const ULocaleDisplayNames * ldn,const char * lang,UChar * result,int32_t maxResultSize,UErrorCode * pErrorCode)853 uldn_languageDisplayName(const ULocaleDisplayNames *ldn,
854 const char *lang,
855 UChar *result,
856 int32_t maxResultSize,
857 UErrorCode *pErrorCode) {
858 if (U_FAILURE(*pErrorCode)) {
859 return 0;
860 }
861 if (ldn == NULL || lang == NULL || (result == NULL && maxResultSize > 0) || maxResultSize < 0) {
862 *pErrorCode = U_ILLEGAL_ARGUMENT_ERROR;
863 return 0;
864 }
865 UnicodeString temp(result, 0, maxResultSize);
866 ((const LocaleDisplayNames *)ldn)->languageDisplayName(lang, temp);
867 return temp.extract(result, maxResultSize, *pErrorCode);
868 }
869
870 U_CAPI int32_t U_EXPORT2
uldn_scriptDisplayName(const ULocaleDisplayNames * ldn,const char * script,UChar * result,int32_t maxResultSize,UErrorCode * pErrorCode)871 uldn_scriptDisplayName(const ULocaleDisplayNames *ldn,
872 const char *script,
873 UChar *result,
874 int32_t maxResultSize,
875 UErrorCode *pErrorCode) {
876 if (U_FAILURE(*pErrorCode)) {
877 return 0;
878 }
879 if (ldn == NULL || script == NULL || (result == NULL && maxResultSize > 0) || maxResultSize < 0) {
880 *pErrorCode = U_ILLEGAL_ARGUMENT_ERROR;
881 return 0;
882 }
883 UnicodeString temp(result, 0, maxResultSize);
884 ((const LocaleDisplayNames *)ldn)->scriptDisplayName(script, temp);
885 return temp.extract(result, maxResultSize, *pErrorCode);
886 }
887
888 U_CAPI int32_t U_EXPORT2
uldn_scriptCodeDisplayName(const ULocaleDisplayNames * ldn,UScriptCode scriptCode,UChar * result,int32_t maxResultSize,UErrorCode * pErrorCode)889 uldn_scriptCodeDisplayName(const ULocaleDisplayNames *ldn,
890 UScriptCode scriptCode,
891 UChar *result,
892 int32_t maxResultSize,
893 UErrorCode *pErrorCode) {
894 return uldn_scriptDisplayName(ldn, uscript_getName(scriptCode), result, maxResultSize, pErrorCode);
895 }
896
897 U_CAPI int32_t U_EXPORT2
uldn_regionDisplayName(const ULocaleDisplayNames * ldn,const char * region,UChar * result,int32_t maxResultSize,UErrorCode * pErrorCode)898 uldn_regionDisplayName(const ULocaleDisplayNames *ldn,
899 const char *region,
900 UChar *result,
901 int32_t maxResultSize,
902 UErrorCode *pErrorCode) {
903 if (U_FAILURE(*pErrorCode)) {
904 return 0;
905 }
906 if (ldn == NULL || region == NULL || (result == NULL && maxResultSize > 0) || maxResultSize < 0) {
907 *pErrorCode = U_ILLEGAL_ARGUMENT_ERROR;
908 return 0;
909 }
910 UnicodeString temp(result, 0, maxResultSize);
911 ((const LocaleDisplayNames *)ldn)->regionDisplayName(region, temp);
912 return temp.extract(result, maxResultSize, *pErrorCode);
913 }
914
915 U_CAPI int32_t U_EXPORT2
uldn_variantDisplayName(const ULocaleDisplayNames * ldn,const char * variant,UChar * result,int32_t maxResultSize,UErrorCode * pErrorCode)916 uldn_variantDisplayName(const ULocaleDisplayNames *ldn,
917 const char *variant,
918 UChar *result,
919 int32_t maxResultSize,
920 UErrorCode *pErrorCode) {
921 if (U_FAILURE(*pErrorCode)) {
922 return 0;
923 }
924 if (ldn == NULL || variant == NULL || (result == NULL && maxResultSize > 0) || maxResultSize < 0) {
925 *pErrorCode = U_ILLEGAL_ARGUMENT_ERROR;
926 return 0;
927 }
928 UnicodeString temp(result, 0, maxResultSize);
929 ((const LocaleDisplayNames *)ldn)->variantDisplayName(variant, temp);
930 return temp.extract(result, maxResultSize, *pErrorCode);
931 }
932
933 U_CAPI int32_t U_EXPORT2
uldn_keyDisplayName(const ULocaleDisplayNames * ldn,const char * key,UChar * result,int32_t maxResultSize,UErrorCode * pErrorCode)934 uldn_keyDisplayName(const ULocaleDisplayNames *ldn,
935 const char *key,
936 UChar *result,
937 int32_t maxResultSize,
938 UErrorCode *pErrorCode) {
939 if (U_FAILURE(*pErrorCode)) {
940 return 0;
941 }
942 if (ldn == NULL || key == NULL || (result == NULL && maxResultSize > 0) || maxResultSize < 0) {
943 *pErrorCode = U_ILLEGAL_ARGUMENT_ERROR;
944 return 0;
945 }
946 UnicodeString temp(result, 0, maxResultSize);
947 ((const LocaleDisplayNames *)ldn)->keyDisplayName(key, temp);
948 return temp.extract(result, maxResultSize, *pErrorCode);
949 }
950
951 U_CAPI int32_t U_EXPORT2
uldn_keyValueDisplayName(const ULocaleDisplayNames * ldn,const char * key,const char * value,UChar * result,int32_t maxResultSize,UErrorCode * pErrorCode)952 uldn_keyValueDisplayName(const ULocaleDisplayNames *ldn,
953 const char *key,
954 const char *value,
955 UChar *result,
956 int32_t maxResultSize,
957 UErrorCode *pErrorCode) {
958 if (U_FAILURE(*pErrorCode)) {
959 return 0;
960 }
961 if (ldn == NULL || key == NULL || value == NULL || (result == NULL && maxResultSize > 0)
962 || maxResultSize < 0) {
963 *pErrorCode = U_ILLEGAL_ARGUMENT_ERROR;
964 return 0;
965 }
966 UnicodeString temp(result, 0, maxResultSize);
967 ((const LocaleDisplayNames *)ldn)->keyValueDisplayName(key, value, temp);
968 return temp.extract(result, maxResultSize, *pErrorCode);
969 }
970
971 #endif
972