tor-browser

The Tor Browser
git clone https://git.dasho.dev/tor-browser.git
Log | Files | Refs | README | LICENSE

localematcher.h (27504B)


      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         */
    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         */
    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         */
    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         */
    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:
    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         */
    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>
    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>
    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         */
    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__