blob: d6174f3142c9e9bb4fa07080444e02bb9faf2df2 [file] [log] [blame]
 // Copyright (c) 2016, the Dart project authors. Please see the AUTHORS file // for details. All rights reserved. Use of this source code is governed by a // BSD-style license that can be found in the LICENSE file. /// Provides locale-specific plural rules. Based on pluralrules.js from Closure. /// /// Each function does the calculation for one or more locales. These are done in terms of /// various values used by the CLDR syntax and defined by UTS #35 /// http://unicode.org/reports/tr35/tr35-numbers.html#Plural_rules_syntax /// /// * n - absolute value of the source number (integer and decimals). /// * i - integer digits of n. /// * v - number of visible fraction digits in n, with trailing zeros. /// * w - number of visible fraction digits in n, without trailing zeros. /// * f - visible fractional digits in n, with trailing zeros. /// * t - visible fractional digits in n, without trailing zeros. library plural_rules; typedef PluralCase PluralRule(); /// The possible cases used in a plural rule. enum PluralCase { ZERO, ONE, TWO, FEW, MANY, OTHER } /// The default rule in case we don't have anything more specific for a locale. PluralCase _default_rule() => OTHER; /// This must be called before evaluating a new rule, because we're using /// library-global state to both keep the rules terse and minimize space. startRuleEvaluation(int howMany) { _n = howMany; } /// The number whose [PluralCase] we are trying to find. /// // This is library-global state, along with the other variables. This allows us // to avoid calculating parameters that the functions don't need and also // not introduce a subclass per locale or have instance tear-offs which // we can't cache. This is fine as long as these methods aren't async, which // they should never be. int _n; /// The integer part of [_n] - since we only support integers, it's the same as /// [_n]. int get _i => _n; int opt_precision; // Not currently used. /// Number of visible fraction digits. Always zero since we only support int. int get _v => 0; /// Number of visible fraction digits without trailing zeros. Always zero /// since we only support int. //int get _w => 0; /// The visible fraction digits in n, with trailing zeros. Always zero since /// we only support int. int get _f => 0; /// The visible fraction digits in n, without trailing zeros. Always zero since /// we only support int. int get _t => 0; PluralCase get ZERO => PluralCase.ZERO; PluralCase get ONE => PluralCase.ONE; PluralCase get TWO => PluralCase.TWO; PluralCase get FEW => PluralCase.FEW; PluralCase get MANY => PluralCase.MANY; PluralCase get OTHER => PluralCase.OTHER; PluralCase _fil_rule() { if (_v == 0 && (_i == 1 || _i == 2 || _i == 3) || _v == 0 && _i % 10 != 4 && _i % 10 != 6 && _i % 10 != 9 || _v != 0 && _f % 10 != 4 && _f % 10 != 6 && _f % 10 != 9) { return ONE; } return OTHER; } PluralCase _pt_PT_rule() { if (_n == 1 && _v == 0) { return ONE; } return OTHER; } PluralCase _br_rule() { if (_n % 10 == 1 && _n % 100 != 11 && _n % 100 != 71 && _n % 100 != 91) { return ONE; } if (_n % 10 == 2 && _n % 100 != 12 && _n % 100 != 72 && _n % 100 != 92) { return TWO; } if ((_n % 10 >= 3 && _n % 10 <= 4 || _n % 10 == 9) && (_n % 100 < 10 || _n % 100 > 19) && (_n % 100 < 70 || _n % 100 > 79) && (_n % 100 < 90 || _n % 100 > 99)) { return FEW; } if (_n != 0 && _n % 1000000 == 0) { return MANY; } return OTHER; } PluralCase _sr_rule() { if (_v == 0 && _i % 10 == 1 && _i % 100 != 11 || _f % 10 == 1 && _f % 100 != 11) { return ONE; } if (_v == 0 && _i % 10 >= 2 && _i % 10 <= 4 && (_i % 100 < 12 || _i % 100 > 14) || _f % 10 >= 2 && _f % 10 <= 4 && (_f % 100 < 12 || _f % 100 > 14)) { return FEW; } return OTHER; } PluralCase _ro_rule() { if (_i == 1 && _v == 0) { return ONE; } if (_v != 0 || _n == 0 || _n != 1 && _n % 100 >= 1 && _n % 100 <= 19) { return FEW; } return OTHER; } PluralCase _hi_rule() { if (_i == 0 || _n == 1) { return ONE; } return OTHER; } PluralCase _fr_rule() { if (_i == 0 || _i == 1) { return ONE; } return OTHER; } PluralCase _cs_rule() { if (_i == 1 && _v == 0) { return ONE; } if (_i >= 2 && _i <= 4 && _v == 0) { return FEW; } if (_v != 0) { return MANY; } return OTHER; } PluralCase _pl_rule() { if (_i == 1 && _v == 0) { return ONE; } if (_v == 0 && _i % 10 >= 2 && _i % 10 <= 4 && (_i % 100 < 12 || _i % 100 > 14)) { return FEW; } if (_v == 0 && _i != 1 && _i % 10 >= 0 && _i % 10 <= 1 || _v == 0 && _i % 10 >= 5 && _i % 10 <= 9 || _v == 0 && _i % 100 >= 12 && _i % 100 <= 14) { return MANY; } return OTHER; } PluralCase _lv_rule() { if (_n % 10 == 0 || _n % 100 >= 11 && _n % 100 <= 19 || _v == 2 && _f % 100 >= 11 && _f % 100 <= 19) { return ZERO; } if (_n % 10 == 1 && _n % 100 != 11 || _v == 2 && _f % 10 == 1 && _f % 100 != 11 || _v != 2 && _f % 10 == 1) { return ONE; } return OTHER; } PluralCase _he_rule() { if (_i == 1 && _v == 0) { return ONE; } if (_i == 2 && _v == 0) { return TWO; } if (_v == 0 && (_n < 0 || _n > 10) && _n % 10 == 0) { return MANY; } return OTHER; } PluralCase _mt_rule() { if (_n == 1) { return ONE; } if (_n == 0 || _n % 100 >= 2 && _n % 100 <= 10) { return FEW; } if (_n % 100 >= 11 && _n % 100 <= 19) { return MANY; } return OTHER; } PluralCase _si_rule() { if ((_n == 0 || _n == 1) || _i == 0 && _f == 1) { return ONE; } return OTHER; } PluralCase _cy_rule() { if (_n == 0) { return ZERO; } if (_n == 1) { return ONE; } if (_n == 2) { return TWO; } if (_n == 3) { return FEW; } if (_n == 6) { return MANY; } return OTHER; } PluralCase _da_rule() { if (_n == 1 || _t != 0 && (_i == 0 || _i == 1)) { return ONE; } return OTHER; } PluralCase _ru_rule() { if (_v == 0 && _i % 10 == 1 && _i % 100 != 11) { return ONE; } if (_v == 0 && _i % 10 >= 2 && _i % 10 <= 4 && (_i % 100 < 12 || _i % 100 > 14)) { return FEW; } if (_v == 0 && _i % 10 == 0 || _v == 0 && _i % 10 >= 5 && _i % 10 <= 9 || _v == 0 && _i % 100 >= 11 && _i % 100 <= 14) { return MANY; } return OTHER; } PluralCase _be_rule() { if (_n % 10 == 1 && _n % 100 != 11) { return ONE; } if (_n % 10 >= 2 && _n % 10 <= 4 && (_n % 100 < 12 || _n % 100 > 14)) { return FEW; } if (_n % 10 == 0 || _n % 10 >= 5 && _n % 10 <= 9 || _n % 100 >= 11 && _n % 100 <= 14) { return MANY; } return OTHER; } PluralCase _mk_rule() { if (_v == 0 && _i % 10 == 1 || _f % 10 == 1) { return ONE; } return OTHER; } PluralCase _ga_rule() { if (_n == 1) { return ONE; } if (_n == 2) { return TWO; } if (_n >= 3 && _n <= 6) { return FEW; } if (_n >= 7 && _n <= 10) { return MANY; } return OTHER; } PluralCase _pt_rule() { if (_n >= 0 && _n <= 2 && _n != 2) { return ONE; } return OTHER; } PluralCase _es_rule() { if (_n == 1) { return ONE; } return OTHER; } PluralCase _is_rule() { if (_t == 0 && _i % 10 == 1 && _i % 100 != 11 || _t != 0) { return ONE; } return OTHER; } PluralCase _ar_rule() { if (_n == 0) { return ZERO; } if (_n == 1) { return ONE; } if (_n == 2) { return TWO; } if (_n % 100 >= 3 && _n % 100 <= 10) { return FEW; } if (_n % 100 >= 11 && _n % 100 <= 99) { return MANY; } return OTHER; } PluralCase _sl_rule() { if (_v == 0 && _i % 100 == 1) { return ONE; } if (_v == 0 && _i % 100 == 2) { return TWO; } if (_v == 0 && _i % 100 >= 3 && _i % 100 <= 4 || _v != 0) { return FEW; } return OTHER; } PluralCase _lt_rule() { if (_n % 10 == 1 && (_n % 100 < 11 || _n % 100 > 19)) { return ONE; } if (_n % 10 >= 2 && _n % 10 <= 9 && (_n % 100 < 11 || _n % 100 > 19)) { return FEW; } if (_f != 0) { return MANY; } return OTHER; } PluralCase _en_rule() { if (_i == 1 && _v == 0) { return ONE; } return OTHER; } PluralCase _ak_rule() { if (_n >= 0 && _n <= 1) { return ONE; } return OTHER; } /// Selected Plural rules by locale. final Map pluralRules = { 'af': _es_rule, 'am': _hi_rule, 'ar': _ar_rule, 'az': _es_rule, 'be': _be_rule, 'bg': _es_rule, 'bn': _hi_rule, 'br': _br_rule, 'bs': _sr_rule, 'ca': _en_rule, 'chr': _es_rule, 'cs': _cs_rule, 'cy': _cy_rule, 'da': _da_rule, 'de': _en_rule, 'de_AT': _en_rule, 'de_CH': _en_rule, 'el': _es_rule, 'en': _en_rule, 'en_AU': _en_rule, 'en_CA': _en_rule, 'en_GB': _en_rule, 'en_IE': _en_rule, 'en_IN': _en_rule, 'en_SG': _en_rule, 'en_US': _en_rule, 'en_ZA': _en_rule, 'es': _es_rule, 'es_419': _es_rule, 'es_ES': _es_rule, 'es_MX': _es_rule, 'es_US': _es_rule, 'et': _en_rule, 'eu': _es_rule, 'fa': _hi_rule, 'fi': _en_rule, 'fil': _fil_rule, 'fr': _fr_rule, 'fr_CA': _fr_rule, 'ga': _ga_rule, 'gl': _en_rule, 'gsw': _es_rule, 'gu': _hi_rule, 'haw': _es_rule, 'he': _he_rule, 'hi': _hi_rule, 'hr': _sr_rule, 'hu': _es_rule, 'hy': _fr_rule, 'id': _default_rule, 'in': _default_rule, 'is': _is_rule, 'it': _en_rule, 'iw': _he_rule, 'ja': _default_rule, 'ka': _es_rule, 'kk': _es_rule, 'km': _default_rule, 'kn': _hi_rule, 'ko': _default_rule, 'ky': _es_rule, 'ln': _ak_rule, 'lo': _default_rule, 'lt': _lt_rule, 'lv': _lv_rule, 'mk': _mk_rule, 'ml': _es_rule, 'mn': _es_rule, 'mo': _ro_rule, 'mr': _hi_rule, 'ms': _default_rule, 'mt': _mt_rule, 'my': _default_rule, 'nb': _es_rule, 'ne': _es_rule, 'nl': _en_rule, 'no': _es_rule, 'no_NO': _es_rule, 'or': _es_rule, 'pa': _ak_rule, 'pl': _pl_rule, 'pt': _pt_rule, 'pt_BR': _pt_rule, 'pt_PT': _pt_PT_rule, 'ro': _ro_rule, 'ru': _ru_rule, 'sh': _sr_rule, 'si': _si_rule, 'sk': _cs_rule, 'sl': _sl_rule, 'sq': _es_rule, 'sr': _sr_rule, 'sr_Latn': _sr_rule, 'sv': _en_rule, 'sw': _en_rule, 'ta': _es_rule, 'te': _es_rule, 'th': _default_rule, 'tl': _fil_rule, 'tr': _es_rule, 'uk': _ru_rule, 'ur': _en_rule, 'uz': _es_rule, 'vi': _default_rule, 'zh': _default_rule, 'zh_CN': _default_rule, 'zh_HK': _default_rule, 'zh_TW': _default_rule, 'zu': _hi_rule, 'default': _default_rule }; /// Do we have plural rules specific to [locale] bool localeHasPluralRules(String locale) => pluralRules.containsKey(locale);