Coverage Report

Created: 2026-06-13 06:44

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/icu/source/common/unicode/localematcher.h
Line
Count
Source
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
0
        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
0
        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
0
        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
0
        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
0
                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
0
        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
0
        Builder &setDirection(ULocMatchDirection matchDirection) {
471
0
            if (U_SUCCESS(errorCode_)) {
472
0
                direction_ = matchDirection;
473
0
            }
474
0
            return *this;
475
0
        }
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__