fx115: Bundle PluralForm.jsm from older Firefox
This commit is contained in:
parent
33db3c28fc
commit
5a8a7ebccc
2 changed files with 315 additions and 1 deletions
|
@ -53,7 +53,7 @@ Zotero.Intl = new function () {
|
|||
}
|
||||
}
|
||||
|
||||
Components.utils.import("resource://gre/modules/PluralForm.jsm");
|
||||
Components.utils.import("resource://zotero/PluralForm.jsm");
|
||||
|
||||
// Exposed for tests
|
||||
this._bundle = bundle = Services.strings.createBundle('chrome://zotero/locale/zotero.properties');
|
||||
|
|
314
resource/PluralForm.jsm
Normal file
314
resource/PluralForm.jsm
Normal file
|
@ -0,0 +1,314 @@
|
|||
/* This Source Code Form is subject to the terms of the Mozilla Public
|
||||
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
||||
|
||||
var EXPORTED_SYMBOLS = ["PluralForm"];
|
||||
|
||||
const { Services } = ChromeUtils.import("resource://gre/modules/Services.jsm");
|
||||
|
||||
/**
|
||||
* This module provides the PluralForm object which contains a method to figure
|
||||
* out which plural form of a word to use for a given number based on the
|
||||
* current localization. There is also a makeGetter method that creates a get
|
||||
* function for the desired plural rule. This is useful for extensions that
|
||||
* specify their own plural rule instead of relying on the browser default.
|
||||
* (I.e., the extension hasn't been localized to the browser's locale.)
|
||||
*
|
||||
* See: http://developer.mozilla.org/en/docs/Localization_and_Plurals
|
||||
*
|
||||
* NOTE: any change to these plural forms need to be reflected in
|
||||
* compare-locales:
|
||||
* https://hg.mozilla.org/l10n/compare-locales/file/default/compare_locales/plurals.py
|
||||
*
|
||||
* List of methods:
|
||||
*
|
||||
* string pluralForm
|
||||
* get(int aNum, string aWords)
|
||||
*
|
||||
* int numForms
|
||||
* numForms()
|
||||
*
|
||||
* [string pluralForm get(int aNum, string aWords), int numForms numForms()]
|
||||
* makeGetter(int aRuleNum)
|
||||
* Note: Basically, makeGetter returns 2 functions that do "get" and "numForm"
|
||||
*/
|
||||
|
||||
const kIntlProperties = "chrome://global/locale/intl.properties";
|
||||
|
||||
// These are the available plural functions that give the appropriate index
|
||||
// based on the plural rule number specified. The first element is the number
|
||||
// of plural forms and the second is the function to figure out the index.
|
||||
/* eslint-disable no-nested-ternary */
|
||||
var gFunctions = [
|
||||
// 0: Chinese
|
||||
[1, n => 0],
|
||||
// 1: English
|
||||
[2, n => (n != 1 ? 1 : 0)],
|
||||
// 2: French
|
||||
[2, n => (n > 1 ? 1 : 0)],
|
||||
// 3: Latvian
|
||||
[3, n => (n % 10 == 1 && n % 100 != 11 ? 1 : n % 10 == 0 ? 0 : 2)],
|
||||
// 4: Scottish Gaelic
|
||||
[
|
||||
4,
|
||||
n =>
|
||||
n == 1 || n == 11 ? 0 : n == 2 || n == 12 ? 1 : n > 0 && n < 20 ? 2 : 3,
|
||||
],
|
||||
// 5: Romanian
|
||||
[3, n => (n == 1 ? 0 : n == 0 || (n % 100 > 0 && n % 100 < 20) ? 1 : 2)],
|
||||
// 6: Lithuanian
|
||||
[
|
||||
3,
|
||||
n =>
|
||||
n % 10 == 1 && n % 100 != 11
|
||||
? 0
|
||||
: n % 10 >= 2 && (n % 100 < 10 || n % 100 >= 20)
|
||||
? 2
|
||||
: 1,
|
||||
],
|
||||
// 7: Russian
|
||||
[
|
||||
3,
|
||||
n =>
|
||||
n % 10 == 1 && n % 100 != 11
|
||||
? 0
|
||||
: n % 10 >= 2 && n % 10 <= 4 && (n % 100 < 10 || n % 100 >= 20)
|
||||
? 1
|
||||
: 2,
|
||||
],
|
||||
// 8: Slovak
|
||||
[3, n => (n == 1 ? 0 : n >= 2 && n <= 4 ? 1 : 2)],
|
||||
// 9: Polish
|
||||
[
|
||||
3,
|
||||
n =>
|
||||
n == 1
|
||||
? 0
|
||||
: n % 10 >= 2 && n % 10 <= 4 && (n % 100 < 10 || n % 100 >= 20)
|
||||
? 1
|
||||
: 2,
|
||||
],
|
||||
// 10: Slovenian
|
||||
[
|
||||
4,
|
||||
n =>
|
||||
n % 100 == 1
|
||||
? 0
|
||||
: n % 100 == 2
|
||||
? 1
|
||||
: n % 100 == 3 || n % 100 == 4
|
||||
? 2
|
||||
: 3,
|
||||
],
|
||||
// 11: Irish Gaeilge
|
||||
[
|
||||
5,
|
||||
n =>
|
||||
n == 1
|
||||
? 0
|
||||
: n == 2
|
||||
? 1
|
||||
: n >= 3 && n <= 6
|
||||
? 2
|
||||
: n >= 7 && n <= 10
|
||||
? 3
|
||||
: 4,
|
||||
],
|
||||
// 12: Arabic
|
||||
[
|
||||
6,
|
||||
n =>
|
||||
n == 0
|
||||
? 5
|
||||
: n == 1
|
||||
? 0
|
||||
: n == 2
|
||||
? 1
|
||||
: n % 100 >= 3 && n % 100 <= 10
|
||||
? 2
|
||||
: n % 100 >= 11 && n % 100 <= 99
|
||||
? 3
|
||||
: 4,
|
||||
],
|
||||
// 13: Maltese
|
||||
[
|
||||
4,
|
||||
n =>
|
||||
n == 1
|
||||
? 0
|
||||
: n == 0 || (n % 100 > 0 && n % 100 <= 10)
|
||||
? 1
|
||||
: n % 100 > 10 && n % 100 < 20
|
||||
? 2
|
||||
: 3,
|
||||
],
|
||||
// 14: Unused
|
||||
[3, n => (n % 10 == 1 ? 0 : n % 10 == 2 ? 1 : 2)],
|
||||
// 15: Icelandic, Macedonian
|
||||
[2, n => (n % 10 == 1 && n % 100 != 11 ? 0 : 1)],
|
||||
// 16: Breton
|
||||
[
|
||||
5,
|
||||
n =>
|
||||
n % 10 == 1 && n % 100 != 11 && n % 100 != 71 && n % 100 != 91
|
||||
? 0
|
||||
: n % 10 == 2 && n % 100 != 12 && n % 100 != 72 && n % 100 != 92
|
||||
? 1
|
||||
: (n % 10 == 3 || n % 10 == 4 || n % 10 == 9) &&
|
||||
n % 100 != 13 &&
|
||||
n % 100 != 14 &&
|
||||
n % 100 != 19 &&
|
||||
n % 100 != 73 &&
|
||||
n % 100 != 74 &&
|
||||
n % 100 != 79 &&
|
||||
n % 100 != 93 &&
|
||||
n % 100 != 94 &&
|
||||
n % 100 != 99
|
||||
? 2
|
||||
: n % 1000000 == 0 && n != 0
|
||||
? 3
|
||||
: 4,
|
||||
],
|
||||
// 17: Shuar
|
||||
[2, n => (n != 0 ? 1 : 0)],
|
||||
// 18: Welsh
|
||||
[
|
||||
6,
|
||||
n => (n == 0 ? 0 : n == 1 ? 1 : n == 2 ? 2 : n == 3 ? 3 : n == 6 ? 4 : 5),
|
||||
],
|
||||
// 19: Slavic languages (bs, hr, sr). Same as rule 7, but resulting in different CLDR categories
|
||||
[
|
||||
3,
|
||||
n =>
|
||||
n % 10 == 1 && n % 100 != 11
|
||||
? 0
|
||||
: n % 10 >= 2 && n % 10 <= 4 && (n % 100 < 10 || n % 100 >= 20)
|
||||
? 1
|
||||
: 2,
|
||||
],
|
||||
];
|
||||
/* eslint-enable no-nested-ternary */
|
||||
|
||||
var PluralForm = {
|
||||
/**
|
||||
* Get the correct plural form of a word based on the number
|
||||
*
|
||||
* @param aNum
|
||||
* The number to decide which plural form to use
|
||||
* @param aWords
|
||||
* A semi-colon (;) separated string of words to pick the plural form
|
||||
* @return The appropriate plural form of the word
|
||||
*/
|
||||
get get() {
|
||||
// This method will lazily load to avoid perf when it is first needed and
|
||||
// creates getPluralForm function. The function it creates is based on the
|
||||
// value of pluralRule specified in the intl stringbundle.
|
||||
// See: http://developer.mozilla.org/en/docs/Localization_and_Plurals
|
||||
|
||||
// Delete the getters to be overwritten
|
||||
delete PluralForm.numForms;
|
||||
delete PluralForm.get;
|
||||
|
||||
// Make the plural form get function and set it as the default get
|
||||
[PluralForm.get, PluralForm.numForms] = PluralForm.makeGetter(
|
||||
PluralForm.ruleNum
|
||||
);
|
||||
return PluralForm.get;
|
||||
},
|
||||
|
||||
/**
|
||||
* Create a pair of plural form functions for the given plural rule number.
|
||||
*
|
||||
* @param aRuleNum
|
||||
* The plural rule number to create functions
|
||||
* @return A pair: [function that gets the right plural form,
|
||||
* function that returns the number of plural forms]
|
||||
*/
|
||||
makeGetter(aRuleNum) {
|
||||
// Default to "all plural" if the value is out of bounds or invalid
|
||||
if (aRuleNum < 0 || aRuleNum >= gFunctions.length || isNaN(aRuleNum)) {
|
||||
log(["Invalid rule number: ", aRuleNum, " -- defaulting to 0"]);
|
||||
aRuleNum = 0;
|
||||
}
|
||||
|
||||
// Get the desired pluralRule function
|
||||
let [numForms, pluralFunc] = gFunctions[aRuleNum];
|
||||
|
||||
// Return functions that give 1) the number of forms and 2) gets the right
|
||||
// plural form
|
||||
return [
|
||||
function(aNum, aWords) {
|
||||
// Figure out which index to use for the semi-colon separated words
|
||||
let index = pluralFunc(aNum ? Number(aNum) : 0);
|
||||
let words = aWords ? aWords.split(/;/) : [""];
|
||||
|
||||
// Explicitly check bounds to avoid strict warnings
|
||||
let ret = index < words.length ? words[index] : undefined;
|
||||
|
||||
// Check for array out of bounds or empty strings
|
||||
if (ret == undefined || ret == "") {
|
||||
// Report the caller to help figure out who is causing badness
|
||||
let caller = Components.stack.caller
|
||||
? Components.stack.caller.name
|
||||
: "top";
|
||||
|
||||
// Display a message in the error console
|
||||
log([
|
||||
"Index #",
|
||||
index,
|
||||
" of '",
|
||||
aWords,
|
||||
"' for value ",
|
||||
aNum,
|
||||
" is invalid -- plural rule #",
|
||||
aRuleNum,
|
||||
"; called by ",
|
||||
caller,
|
||||
]);
|
||||
|
||||
// Default to the first entry (which might be empty, but not undefined)
|
||||
ret = words[0];
|
||||
}
|
||||
|
||||
return ret;
|
||||
},
|
||||
() => numForms,
|
||||
];
|
||||
},
|
||||
|
||||
/**
|
||||
* Get the number of forms for the current plural rule
|
||||
*
|
||||
* @return The number of forms
|
||||
*/
|
||||
get numForms() {
|
||||
// We lazily load numForms, so trigger the init logic with get()
|
||||
PluralForm.get();
|
||||
return PluralForm.numForms;
|
||||
},
|
||||
|
||||
/**
|
||||
* Get the plural rule number from the intl stringbundle
|
||||
*
|
||||
* @return The plural rule number
|
||||
*/
|
||||
get ruleNum() {
|
||||
return Number(
|
||||
Services.strings
|
||||
.createBundle(kIntlProperties)
|
||||
.GetStringFromName("pluralRule")
|
||||
);
|
||||
},
|
||||
};
|
||||
|
||||
/**
|
||||
* Private helper function to log errors to the error console and command line
|
||||
*
|
||||
* @param aMsg
|
||||
* Error message to log or an array of strings to concat
|
||||
*/
|
||||
function log(aMsg) {
|
||||
let msg = "PluralForm.jsm: " + (aMsg.join ? aMsg.join("") : aMsg);
|
||||
Services.console.logStringMessage(msg);
|
||||
dump(msg + "\n");
|
||||
}
|
Loading…
Reference in a new issue