1 // © 2019 and later: Unicode, Inc. and others. 2 // License & terms of use: http://www.unicode.org/copyright.html 3 4 // localematcher.h 5 // created: 2019may08 Markus W. Scherer 6 7 #ifndef __LOCALEMATCHER_H__ 8 #define __LOCALEMATCHER_H__ 9 10 #include "unicode/utypes.h" 11 12 #if U_SHOW_CPLUSPLUS_API 13 14 #include <optional> 15 16 #include "unicode/locid.h" 17 #include "unicode/stringpiece.h" 18 #include "unicode/uobject.h" 19 20 /** 21 * \file 22 * \brief C++ API: Locale matcher: User's desired locales vs. application's supported locales. 23 */ 24 25 /** 26 * Builder option for whether the language subtag or the script subtag is most important. 27 * 28 * @see LocaleMatcher::Builder#setFavorSubtag(ULocMatchFavorSubtag) 29 * @stable ICU 65 30 */ 31 enum ULocMatchFavorSubtag { 32 /** 33 * Language differences are most important, then script differences, then region differences. 34 * (This is the default behavior.) 35 * 36 * @stable ICU 65 37 */ 38 ULOCMATCH_FAVOR_LANGUAGE, 39 /** 40 * Makes script differences matter relatively more than language differences. 41 * 42 * @stable ICU 65 43 */ 44 ULOCMATCH_FAVOR_SCRIPT 45 }; 46 #ifndef U_IN_DOXYGEN 47 typedef enum ULocMatchFavorSubtag ULocMatchFavorSubtag; 48 #endif 49 50 /** 51 * Builder option for whether all desired locales are treated equally or 52 * earlier ones are preferred. 53 * 54 * @see LocaleMatcher::Builder#setDemotionPerDesiredLocale(ULocMatchDemotion) 55 * @stable ICU 65 56 */ 57 enum ULocMatchDemotion { 58 /** 59 * All desired locales are treated equally. 60 * 61 * @stable ICU 65 62 */ 63 ULOCMATCH_DEMOTION_NONE, 64 /** 65 * Earlier desired locales are preferred. 66 * 67 * <p>From each desired locale to the next, 68 * the distance to any supported locale is increased by an additional amount 69 * which is at least as large as most region mismatches. 70 * A later desired locale has to have a better match with some supported locale 71 * due to more than merely having the same region subtag. 72 * 73 * <p>For example: <code>Supported={en, sv} desired=[en-GB, sv]</code> 74 * yields <code>Result(en-GB, en)</code> because 75 * with the demotion of sv its perfect match is no better than 76 * the region distance between the earlier desired locale en-GB and en=en-US. 77 * 78 * <p>Notes: 79 * <ul> 80 * <li>In some cases, language and/or script differences can be as small as 81 * the typical region difference. (Example: sr-Latn vs. sr-Cyrl) 82 * <li>It is possible for certain region differences to be larger than usual, 83 * and larger than the demotion. 84 * (As of CLDR 35 there is no such case, but 85 * this is possible in future versions of the data.) 86 * </ul> 87 * 88 * @stable ICU 65 89 */ 90 ULOCMATCH_DEMOTION_REGION 91 }; 92 #ifndef U_IN_DOXYGEN 93 typedef enum ULocMatchDemotion ULocMatchDemotion; 94 #endif 95 96 /** 97 * Builder option for whether to include or ignore one-way (fallback) match data. 98 * The LocaleMatcher uses CLDR languageMatch data which includes fallback (oneway=true) entries. 99 * Sometimes it is desirable to ignore those. 100 * 101 * <p>For example, consider a web application with the UI in a given language, 102 * with a link to another, related web app. 103 * The link should include the UI language, and the target server may also use 104 * the client’s Accept-Language header data. 105 * The target server has its own list of supported languages. 106 * One may want to favor UI language consistency, that is, 107 * if there is a decent match for the original UI language, we want to use it, 108 * but not if it is merely a fallback. 109 * 110 * @see LocaleMatcher::Builder#setDirection(ULocMatchDirection) 111 * @stable ICU 67 112 */ 113 enum ULocMatchDirection { 114 /** 115 * Locale matching includes one-way matches such as Breton→French. (default) 116 * 117 * @stable ICU 67 118 */ 119 ULOCMATCH_DIRECTION_WITH_ONE_WAY, 120 /** 121 * Locale matching limited to two-way matches including e.g. Danish↔Norwegian 122 * but ignoring one-way matches. 123 * 124 * @stable ICU 67 125 */ 126 ULOCMATCH_DIRECTION_ONLY_TWO_WAY 127 }; 128 #ifndef U_IN_DOXYGEN 129 typedef enum ULocMatchDirection ULocMatchDirection; 130 #endif 131 132 struct UHashtable; 133 134 U_NAMESPACE_BEGIN 135 136 struct LSR; 137 138 class LikelySubtags; 139 class LocaleDistance; 140 class LocaleLsrIterator; 141 class UVector; 142 143 /** 144 * Immutable class that picks the best match between a user's desired locales and 145 * an application's supported locales. 146 * Movable but not copyable. 147 * 148 * <p>Example: 149 * <pre> 150 * UErrorCode errorCode = U_ZERO_ERROR; 151 * LocaleMatcher matcher = LocaleMatcher::Builder().setSupportedLocales("fr, en-GB, en").build(errorCode); 152 * Locale *bestSupported = matcher.getBestLocale(Locale.US, errorCode); // "en" 153 * </pre> 154 * 155 * <p>A matcher takes into account when languages are close to one another, 156 * such as Danish and Norwegian, 157 * and when regional variants are close, like en-GB and en-AU as opposed to en-US. 158 * 159 * <p>If there are multiple supported locales with the same (language, script, region) 160 * likely subtags, then the current implementation returns the first of those locales. 161 * It ignores variant subtags (except for pseudolocale variants) and extensions. 162 * This may change in future versions. 163 * 164 * <p>For example, the current implementation does not distinguish between 165 * de, de-DE, de-Latn, de-1901, de-u-co-phonebk. 166 * 167 * <p>If you prefer one equivalent locale over another, then provide only the preferred one, 168 * or place it earlier in the list of supported locales. 169 * 170 * <p>Otherwise, the order of supported locales may have no effect on the best-match results. 171 * The current implementation compares each desired locale with supported locales 172 * in the following order: 173 * 1. Default locale, if supported; 174 * 2. CLDR "paradigm locales" like en-GB and es-419; 175 * 3. other supported locales. 176 * This may change in future versions. 177 * 178 * <p>Often a product will just need one matcher instance, built with the languages 179 * that it supports. However, it may want multiple instances with different 180 * default languages based on additional information, such as the domain. 181 * 182 * <p>This class is not intended for public subclassing. 183 * 184 * @stable ICU 65 185 */ 186 class U_COMMON_API LocaleMatcher : public UMemory { 187 public: 188 /** 189 * Data for the best-matching pair of a desired and a supported locale. 190 * Movable but not copyable. 191 * 192 * @stable ICU 65 193 */ 194 class U_COMMON_API Result : public UMemory { 195 public: 196 /** 197 * Move constructor; might modify the source. 198 * This object will have the same contents that the source object had. 199 * 200 * @param src Result to move contents from. 201 * @stable ICU 65 202 */ 203 Result(Result &&src) noexcept; 204 205 /** 206 * Destructor. 207 * 208 * @stable ICU 65 209 */ 210 ~Result(); 211 212 /** 213 * Move assignment; might modify the source. 214 * This object will have the same contents that the source object had. 215 * 216 * @param src Result to move contents from. 217 * @stable ICU 65 218 */ 219 Result &operator=(Result &&src) noexcept; 220 221 /** 222 * Returns the best-matching desired locale. 223 * nullptr if the list of desired locales is empty or if none matched well enough. 224 * 225 * @return the best-matching desired locale, or nullptr. 226 * @stable ICU 65 227 */ getDesiredLocale()228 inline const Locale *getDesiredLocale() const { return desiredLocale; } 229 230 /** 231 * Returns the best-matching supported locale. 232 * If none matched well enough, this is the default locale. 233 * The default locale is nullptr if Builder::setNoDefaultLocale() was called, 234 * or if the list of supported locales is empty and no explicit default locale is set. 235 * 236 * @return the best-matching supported locale, or nullptr. 237 * @stable ICU 65 238 */ getSupportedLocale()239 inline const Locale *getSupportedLocale() const { return supportedLocale; } 240 241 /** 242 * Returns the index of the best-matching desired locale in the input Iterable order. 243 * -1 if the list of desired locales is empty or if none matched well enough. 244 * 245 * @return the index of the best-matching desired locale, or -1. 246 * @stable ICU 65 247 */ getDesiredIndex()248 inline int32_t getDesiredIndex() const { return desiredIndex; } 249 250 /** 251 * Returns the index of the best-matching supported locale in the 252 * constructor’s or builder’s input order (“set” Collection plus “added” locales). 253 * If the matcher was built from a locale list string, then the iteration order is that 254 * of a LocalePriorityList built from the same string. 255 * -1 if the list of supported locales is empty or if none matched well enough. 256 * 257 * @return the index of the best-matching supported locale, or -1. 258 * @stable ICU 65 259 */ getSupportedIndex()260 inline int32_t getSupportedIndex() const { return supportedIndex; } 261 262 /** 263 * Takes the best-matching supported locale and adds relevant fields of the 264 * best-matching desired locale, such as the -t- and -u- extensions. 265 * May replace some fields of the supported locale. 266 * The result is the locale that should be used for date and number formatting, collation, etc. 267 * Returns the root locale if getSupportedLocale() returns nullptr. 268 * 269 * <p>Example: desired=ar-SA-u-nu-latn, supported=ar-EG, resolved locale=ar-SA-u-nu-latn 270 * 271 * @return a locale combining the best-matching desired and supported locales. 272 * @stable ICU 65 273 */ 274 Locale makeResolvedLocale(UErrorCode &errorCode) const; 275 276 private: Result(const Locale * desired,const Locale * supported,int32_t desIndex,int32_t suppIndex,UBool owned)277 Result(const Locale *desired, const Locale *supported, 278 int32_t desIndex, int32_t suppIndex, UBool owned) : 279 desiredLocale(desired), supportedLocale(supported), 280 desiredIndex(desIndex), supportedIndex(suppIndex), 281 desiredIsOwned(owned) {} 282 283 Result(const Result &other) = delete; 284 Result &operator=(const Result &other) = delete; 285 286 const Locale *desiredLocale; 287 const Locale *supportedLocale; 288 int32_t desiredIndex; 289 int32_t supportedIndex; 290 UBool desiredIsOwned; 291 292 friend class LocaleMatcher; 293 }; 294 295 /** 296 * LocaleMatcher builder. 297 * Movable but not copyable. 298 * 299 * @stable ICU 65 300 */ 301 class U_COMMON_API Builder : public UMemory { 302 public: 303 /** 304 * Constructs a builder used in chaining parameters for building a LocaleMatcher. 305 * 306 * @return a new Builder object 307 * @stable ICU 65 308 */ Builder()309 Builder() {} 310 311 /** 312 * Move constructor; might modify the source. 313 * This builder will have the same contents that the source builder had. 314 * 315 * @param src Builder to move contents from. 316 * @stable ICU 65 317 */ 318 Builder(Builder &&src) noexcept; 319 320 /** 321 * Destructor. 322 * 323 * @stable ICU 65 324 */ 325 ~Builder(); 326 327 /** 328 * Move assignment; might modify the source. 329 * This builder will have the same contents that the source builder had. 330 * 331 * @param src Builder to move contents from. 332 * @stable ICU 65 333 */ 334 Builder &operator=(Builder &&src) noexcept; 335 336 /** 337 * Parses an Accept-Language string 338 * (<a href="https://tools.ietf.org/html/rfc2616#section-14.4">RFC 2616 Section 14.4</a>), 339 * such as "af, en, fr;q=0.9", and sets the supported locales accordingly. 340 * Allows whitespace in more places but does not allow "*". 341 * Clears any previously set/added supported locales first. 342 * 343 * @param locales the Accept-Language string of locales to set 344 * @return this Builder object 345 * @stable ICU 65 346 */ 347 Builder &setSupportedLocalesFromListString(StringPiece locales); 348 349 /** 350 * Copies the supported locales, preserving iteration order. 351 * Clears any previously set/added supported locales first. 352 * Duplicates are allowed, and are not removed. 353 * 354 * @param locales the list of locale 355 * @return this Builder object 356 * @stable ICU 65 357 */ 358 Builder &setSupportedLocales(Locale::Iterator &locales); 359 360 /** 361 * Copies the supported locales from the begin/end range, preserving iteration order. 362 * Clears any previously set/added supported locales first. 363 * Duplicates are allowed, and are not removed. 364 * 365 * Each of the iterator parameter values must be an 366 * input iterator whose value is convertible to const Locale &. 367 * 368 * @param begin Start of range. 369 * @param end Exclusive end of range. 370 * @return this Builder object 371 * @stable ICU 65 372 */ 373 template<typename Iter> setSupportedLocales(Iter begin,Iter end)374 Builder &setSupportedLocales(Iter begin, Iter end) { 375 if (U_FAILURE(errorCode_)) { return *this; } 376 clearSupportedLocales(); 377 while (begin != end) { 378 addSupportedLocale(*begin++); 379 } 380 return *this; 381 } 382 383 /** 384 * Copies the supported locales from the begin/end range, preserving iteration order. 385 * Calls the converter to convert each *begin to a Locale or const Locale &. 386 * Clears any previously set/added supported locales first. 387 * Duplicates are allowed, and are not removed. 388 * 389 * Each of the iterator parameter values must be an 390 * input iterator whose value is convertible to const Locale &. 391 * 392 * @param begin Start of range. 393 * @param end Exclusive end of range. 394 * @param converter Converter from *begin to const Locale & or compatible. 395 * @return this Builder object 396 * @stable ICU 65 397 */ 398 template<typename Iter, typename Conv> setSupportedLocalesViaConverter(Iter begin,Iter end,Conv converter)399 Builder &setSupportedLocalesViaConverter(Iter begin, Iter end, Conv converter) { 400 if (U_FAILURE(errorCode_)) { return *this; } 401 clearSupportedLocales(); 402 while (begin != end) { 403 addSupportedLocale(converter(*begin++)); 404 } 405 return *this; 406 } 407 408 /** 409 * Adds another supported locale. 410 * Duplicates are allowed, and are not removed. 411 * 412 * @param locale another locale 413 * @return this Builder object 414 * @stable ICU 65 415 */ 416 Builder &addSupportedLocale(const Locale &locale); 417 418 /** 419 * Sets no default locale. 420 * There will be no explicit or implicit default locale. 421 * If there is no good match, then the matcher will return nullptr for the 422 * best supported locale. 423 * 424 * @stable ICU 68 425 */ 426 Builder &setNoDefaultLocale(); 427 428 /** 429 * Sets the default locale; if nullptr, or if it is not set explicitly, 430 * then the first supported locale is used as the default locale. 431 * There is no default locale at all (nullptr will be returned instead) 432 * if setNoDefaultLocale() is called. 433 * 434 * @param defaultLocale the default locale (will be copied) 435 * @return this Builder object 436 * @stable ICU 65 437 */ 438 Builder &setDefaultLocale(const Locale *defaultLocale); 439 440 /** 441 * If ULOCMATCH_FAVOR_SCRIPT, then the language differences are smaller than script 442 * differences. 443 * This is used in situations (such as maps) where 444 * it is better to fall back to the same script than a similar language. 445 * 446 * @param subtag the subtag to favor 447 * @return this Builder object 448 * @stable ICU 65 449 */ 450 Builder &setFavorSubtag(ULocMatchFavorSubtag subtag); 451 452 /** 453 * Option for whether all desired locales are treated equally or 454 * earlier ones are preferred (this is the default). 455 * 456 * @param demotion the demotion per desired locale to set. 457 * @return this Builder object 458 * @stable ICU 65 459 */ 460 Builder &setDemotionPerDesiredLocale(ULocMatchDemotion demotion); 461 462 /** 463 * Option for whether to include or ignore one-way (fallback) match data. 464 * By default, they are included. 465 * 466 * @param matchDirection the match direction to set. 467 * @return this Builder object 468 * @stable ICU 67 469 */ setDirection(ULocMatchDirection matchDirection)470 Builder &setDirection(ULocMatchDirection matchDirection) { 471 if (U_SUCCESS(errorCode_)) { 472 direction_ = matchDirection; 473 } 474 return *this; 475 } 476 477 /** 478 * Sets the maximum distance for an acceptable match. 479 * The matcher will return a match for a pair of locales only if 480 * they match at least as well as the pair given here. 481 * 482 * For example, setMaxDistance(en-US, en-GB) limits matches to ones where the 483 * (desired, support) locales have a distance no greater than a region subtag difference. 484 * This is much stricter than the CLDR default. 485 * 486 * The details of locale matching are subject to changes in 487 * CLDR data and in the algorithm. 488 * Specifying a maximum distance in relative terms via a sample pair of locales 489 * insulates from changes that affect all distance metrics similarly, 490 * but some changes will necessarily affect relative distances between 491 * different pairs of locales. 492 * 493 * @param desired the desired locale for distance comparison. 494 * @param supported the supported locale for distance comparison. 495 * @return this Builder object 496 * @stable ICU 68 497 */ 498 Builder &setMaxDistance(const Locale &desired, const Locale &supported); 499 500 /** 501 * Sets the UErrorCode if an error occurred while setting parameters. 502 * Preserves older error codes in the outErrorCode. 503 * 504 * @param outErrorCode Set to an error code if it does not contain one already 505 * and an error occurred while setting parameters. 506 * Otherwise unchanged. 507 * @return true if U_FAILURE(outErrorCode) 508 * @stable ICU 65 509 */ 510 UBool copyErrorTo(UErrorCode &outErrorCode) const; 511 512 /** 513 * Builds and returns a new locale matcher. 514 * This builder can continue to be used. 515 * 516 * @param errorCode ICU error code. Its input value must pass the U_SUCCESS() test, 517 * or else the function returns immediately. Check for U_FAILURE() 518 * on output or use with function chaining. (See User Guide for details.) 519 * @return LocaleMatcher 520 * @stable ICU 65 521 */ 522 LocaleMatcher build(UErrorCode &errorCode) const; 523 524 private: 525 friend class LocaleMatcher; 526 527 Builder(const Builder &other) = delete; 528 Builder &operator=(const Builder &other) = delete; 529 530 void clearSupportedLocales(); 531 bool ensureSupportedLocaleVector(); 532 533 UErrorCode errorCode_ = U_ZERO_ERROR; 534 UVector *supportedLocales_ = nullptr; 535 int32_t thresholdDistance_ = -1; 536 ULocMatchDemotion demotion_ = ULOCMATCH_DEMOTION_REGION; 537 Locale *defaultLocale_ = nullptr; 538 bool withDefault_ = true; 539 ULocMatchFavorSubtag favor_ = ULOCMATCH_FAVOR_LANGUAGE; 540 ULocMatchDirection direction_ = ULOCMATCH_DIRECTION_WITH_ONE_WAY; 541 Locale *maxDistanceDesired_ = nullptr; 542 Locale *maxDistanceSupported_ = nullptr; 543 }; 544 545 // FYI No public LocaleMatcher constructors in C++; use the Builder. 546 547 /** 548 * Move copy constructor; might modify the source. 549 * This matcher will have the same settings that the source matcher had. 550 * @param src source matcher 551 * @stable ICU 65 552 */ 553 LocaleMatcher(LocaleMatcher &&src) noexcept; 554 555 /** 556 * Destructor. 557 * @stable ICU 65 558 */ 559 ~LocaleMatcher(); 560 561 /** 562 * Move assignment operator; might modify the source. 563 * This matcher will have the same settings that the source matcher had. 564 * The behavior is undefined if *this and src are the same object. 565 * @param src source matcher 566 * @return *this 567 * @stable ICU 65 568 */ 569 LocaleMatcher &operator=(LocaleMatcher &&src) noexcept; 570 571 /** 572 * Returns the supported locale which best matches the desired locale. 573 * 574 * @param desiredLocale Typically a user's language. 575 * @param errorCode ICU error code. Its input value must pass the U_SUCCESS() test, 576 * or else the function returns immediately. Check for U_FAILURE() 577 * on output or use with function chaining. (See User Guide for details.) 578 * @return the best-matching supported locale. 579 * @stable ICU 65 580 */ 581 const Locale *getBestMatch(const Locale &desiredLocale, UErrorCode &errorCode) const; 582 583 /** 584 * Returns the supported locale which best matches one of the desired locales. 585 * 586 * @param desiredLocales Typically a user's languages, in order of preference (descending). 587 * @param errorCode ICU error code. Its input value must pass the U_SUCCESS() test, 588 * or else the function returns immediately. Check for U_FAILURE() 589 * on output or use with function chaining. (See User Guide for details.) 590 * @return the best-matching supported locale. 591 * @stable ICU 65 592 */ 593 const Locale *getBestMatch(Locale::Iterator &desiredLocales, UErrorCode &errorCode) const; 594 595 /** 596 * Parses an Accept-Language string 597 * (<a href="https://tools.ietf.org/html/rfc2616#section-14.4">RFC 2616 Section 14.4</a>), 598 * such as "af, en, fr;q=0.9", 599 * and returns the supported locale which best matches one of the desired locales. 600 * Allows whitespace in more places but does not allow "*". 601 * 602 * @param desiredLocaleList Typically a user's languages, as an Accept-Language string. 603 * @param errorCode ICU error code. Its input value must pass the U_SUCCESS() test, 604 * or else the function returns immediately. Check for U_FAILURE() 605 * on output or use with function chaining. (See User Guide for details.) 606 * @return the best-matching supported locale. 607 * @stable ICU 65 608 */ 609 const Locale *getBestMatchForListString(StringPiece desiredLocaleList, UErrorCode &errorCode) const; 610 611 /** 612 * Returns the best match between the desired locale and the supported locales. 613 * If the result's desired locale is not nullptr, then it is the address of the input locale. 614 * It has not been cloned. 615 * 616 * @param desiredLocale Typically a user's language. 617 * @param errorCode ICU error code. Its input value must pass the U_SUCCESS() test, 618 * or else the function returns immediately. Check for U_FAILURE() 619 * on output or use with function chaining. (See User Guide for details.) 620 * @return the best-matching pair of the desired and a supported locale. 621 * @stable ICU 65 622 */ 623 Result getBestMatchResult(const Locale &desiredLocale, UErrorCode &errorCode) const; 624 625 /** 626 * Returns the best match between the desired and supported locales. 627 * If the result's desired locale is not nullptr, then it is a clone of 628 * the best-matching desired locale. The Result object owns the clone. 629 * 630 * @param desiredLocales Typically a user's languages, in order of preference (descending). 631 * @param errorCode ICU error code. Its input value must pass the U_SUCCESS() test, 632 * or else the function returns immediately. Check for U_FAILURE() 633 * on output or use with function chaining. (See User Guide for details.) 634 * @return the best-matching pair of a desired and a supported locale. 635 * @stable ICU 65 636 */ 637 Result getBestMatchResult(Locale::Iterator &desiredLocales, UErrorCode &errorCode) const; 638 639 /** 640 * Returns true if the pair of locales matches acceptably. 641 * This is influenced by Builder options such as setDirection(), setFavorSubtag(), 642 * and setMaxDistance(). 643 * 644 * @param desired The desired locale. 645 * @param supported The supported locale. 646 * @param errorCode ICU error code. Its input value must pass the U_SUCCESS() test, 647 * or else the function returns immediately. Check for U_FAILURE() 648 * on output or use with function chaining. (See User Guide for details.) 649 * @return true if the pair of locales matches acceptably. 650 * @stable ICU 68 651 */ 652 UBool isMatch(const Locale &desired, const Locale &supported, UErrorCode &errorCode) const; 653 654 #ifndef U_HIDE_INTERNAL_API 655 /** 656 * Returns a fraction between 0 and 1, where 1 means that the languages are a 657 * perfect match, and 0 means that they are completely different. 658 * 659 * <p>This is mostly an implementation detail, and the precise values may change over time. 660 * The implementation may use either the maximized forms or the others ones, or both. 661 * The implementation may or may not rely on the forms to be consistent with each other. 662 * 663 * <p>Callers should construct and use a matcher rather than match pairs of locales directly. 664 * 665 * @param desired Desired locale. 666 * @param supported Supported locale. 667 * @param errorCode ICU error code. Its input value must pass the U_SUCCESS() test, 668 * or else the function returns immediately. Check for U_FAILURE() 669 * on output or use with function chaining. (See User Guide for details.) 670 * @return value between 0 and 1, inclusive. 671 * @internal (has a known user) 672 */ 673 double internalMatch(const Locale &desired, const Locale &supported, UErrorCode &errorCode) const; 674 #endif // U_HIDE_INTERNAL_API 675 676 private: 677 LocaleMatcher(const Builder &builder, UErrorCode &errorCode); 678 LocaleMatcher(const LocaleMatcher &other) = delete; 679 LocaleMatcher &operator=(const LocaleMatcher &other) = delete; 680 681 int32_t putIfAbsent(const LSR &lsr, int32_t i, int32_t suppLength, UErrorCode &errorCode); 682 683 std::optional<int32_t> getBestSuppIndex(LSR desiredLSR, LocaleLsrIterator *remainingIter, UErrorCode &errorCode) const; 684 685 const LikelySubtags &likelySubtags; 686 const LocaleDistance &localeDistance; 687 int32_t thresholdDistance; 688 int32_t demotionPerDesiredLocale; 689 ULocMatchFavorSubtag favorSubtag; 690 ULocMatchDirection direction; 691 692 // These are in input order. 693 const Locale ** supportedLocales; 694 LSR *lsrs; 695 int32_t supportedLocalesLength; 696 // These are in preference order: 1. Default locale 2. paradigm locales 3. others. 697 UHashtable *supportedLsrToIndex; // Map<LSR, Integer> 698 // Array versions of the supportedLsrToIndex keys and values. 699 // The distance lookup loops over the supportedLSRs and returns the index of the best match. 700 const LSR **supportedLSRs; 701 int32_t *supportedIndexes; 702 int32_t supportedLSRsLength; 703 Locale *ownedDefaultLocale; 704 const Locale *defaultLocale; 705 }; 706 707 U_NAMESPACE_END 708 709 #endif // U_SHOW_CPLUSPLUS_API 710 #endif // __LOCALEMATCHER_H__ 711