Compare commits
23 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
ad71dba6ad | ||
|
|
ab3d6f8405 | ||
|
|
58ee1abedb | ||
|
|
0aef81f4e9 | ||
|
|
579a5635a6 | ||
|
|
03a257b40d | ||
|
|
58875dc119 | ||
|
|
a6937bb7f0 | ||
|
|
cfe01e8aea | ||
|
|
14bad962eb | ||
|
|
ff0e5655f9 | ||
|
|
bb56f6e34c | ||
|
|
f035affef6 | ||
|
|
18a4cc0cb5 | ||
|
|
0fcff887fe | ||
|
|
9689ae5b7f | ||
|
|
5cbb657f20 | ||
|
|
e038f25d5b | ||
|
|
f26ecb0d63 | ||
|
|
3324041669 | ||
|
|
d30a02f3c2 | ||
|
|
0e3a6d74f3 | ||
|
|
d783222d37 |
@@ -1,6 +1,5 @@
|
||||
language: node_js
|
||||
node_js:
|
||||
- "0.10"
|
||||
- "0.11"
|
||||
- "node"
|
||||
before_install:
|
||||
- npm install -g grunt-cli
|
||||
|
||||
@@ -1,6 +1,6 @@
|
||||
{
|
||||
"name": "pofile",
|
||||
"version": "1.0.4",
|
||||
"version": "1.0.10",
|
||||
"authors": [
|
||||
"Ruben Vermeersch <ruben@rocketeer.be>"
|
||||
],
|
||||
|
||||
65
dist/pofile.js
vendored
65
dist/pofile.js
vendored
@@ -23,12 +23,12 @@ PO.prototype.toString = function () {
|
||||
|
||||
if (this.comments) {
|
||||
this.comments.forEach(function (comment) {
|
||||
lines.push('# ' + comment);
|
||||
lines.push(('# ' + comment).trim());
|
||||
});
|
||||
}
|
||||
if (this.extractedComments) {
|
||||
this.extractedComments.forEach(function (comment) {
|
||||
lines.push('#. ' + comment);
|
||||
lines.push(('#. ' + comment).trim());
|
||||
});
|
||||
}
|
||||
|
||||
@@ -117,7 +117,9 @@ PO.parse = function (data) {
|
||||
}
|
||||
});
|
||||
|
||||
var item = new PO.Item();
|
||||
var parsedPluralForms = PO.parsePluralForms(po.headers['Plural-Forms']);
|
||||
var nplurals = parsedPluralForms.nplurals;
|
||||
var item = new PO.Item({ nplurals: nplurals });
|
||||
var context = null;
|
||||
var plural = 0;
|
||||
var obsoleteCount = 0;
|
||||
@@ -131,7 +133,7 @@ PO.parse = function (data) {
|
||||
obsoleteCount = 0;
|
||||
noCommentLineCount = 0;
|
||||
po.items.push(item);
|
||||
item = new PO.Item();
|
||||
item = new PO.Item({ nplurals: nplurals });
|
||||
}
|
||||
}
|
||||
|
||||
@@ -213,6 +215,7 @@ PO.parse = function (data) {
|
||||
} else if (line.match(/^msgctxt/)) { // Context
|
||||
finish();
|
||||
item.msgctxt = extract(line);
|
||||
context = 'msgctxt';
|
||||
noCommentLineCount++;
|
||||
} else { // Probably multiline string or blank
|
||||
if (line.length > 0) {
|
||||
@@ -223,6 +226,8 @@ PO.parse = function (data) {
|
||||
item.msgid += extract(line);
|
||||
} else if (context === 'msgid_plural') {
|
||||
item.msgid_plural += extract(line);
|
||||
} else if (context === 'msgctxt') {
|
||||
item.msgctxt += extract(line);
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -237,7 +242,26 @@ PO.parse = function (data) {
|
||||
return po;
|
||||
};
|
||||
|
||||
PO.Item = function () {
|
||||
PO.parsePluralForms = function (pluralFormsString) {
|
||||
var results = (pluralFormsString || '')
|
||||
.split(';')
|
||||
.reduce(function (acc, keyValueString) {
|
||||
var trimmedString = keyValueString.trim();
|
||||
var equalsIndex = trimmedString.indexOf('=');
|
||||
var key = trimmedString.substring(0, equalsIndex).trim();
|
||||
var value = trimmedString.substring(equalsIndex + 1).trim();
|
||||
acc[key] = value;
|
||||
return acc;
|
||||
}, {});
|
||||
return {
|
||||
nplurals: results.nplurals,
|
||||
plural: results.plural
|
||||
};
|
||||
};
|
||||
|
||||
PO.Item = function (options) {
|
||||
var nplurals = options && options.nplurals;
|
||||
|
||||
this.msgid = '';
|
||||
this.msgctxt = null;
|
||||
this.references = [];
|
||||
@@ -247,6 +271,8 @@ PO.Item = function () {
|
||||
this.extractedComments = [];
|
||||
this.flags = {};
|
||||
this.obsolete = false;
|
||||
var npluralsNumber = Number(nplurals);
|
||||
this.nplurals = (isNaN(npluralsNumber)) ? 2 : npluralsNumber;
|
||||
};
|
||||
|
||||
PO.Item.prototype.toString = function () {
|
||||
@@ -293,6 +319,15 @@ PO.Item.prototype.toString = function () {
|
||||
return lines;
|
||||
};
|
||||
|
||||
//handle \n in single-line texts (can not be handled in _escape)
|
||||
var _processLineBreak = function (keyword, text, index) {
|
||||
var processed = _process(keyword, text, index);
|
||||
for (var i = 1; i < processed.length - 1; i++) {
|
||||
processed[i] = processed[i].slice(0, -1) + '\\n"';
|
||||
}
|
||||
return processed;
|
||||
};
|
||||
|
||||
// https://www.gnu.org/software/gettext/manual/html_node/PO-Files.html
|
||||
// says order is translator-comments, extracted-comments, references, flags
|
||||
|
||||
@@ -319,20 +354,28 @@ PO.Item.prototype.toString = function () {
|
||||
['msgctxt', 'msgid', 'msgid_plural', 'msgstr'].forEach(function (keyword) {
|
||||
var text = self[keyword];
|
||||
if (text != null) {
|
||||
var hasTranslation = false;
|
||||
if (Array.isArray(text)) {
|
||||
hasTranslation = text.some(function (text) {
|
||||
return text;
|
||||
});
|
||||
}
|
||||
|
||||
if (Array.isArray(text) && text.length > 1) {
|
||||
text.forEach(function (t, i) {
|
||||
lines = lines.concat(mkObsolete + _process(keyword, t, i));
|
||||
var processed = _processLineBreak(keyword, t, i);
|
||||
lines = lines.concat(mkObsolete + processed.join('\n' + mkObsolete));
|
||||
});
|
||||
} else if (self.msgid_plural && keyword === 'msgstr' && !hasTranslation) {
|
||||
for (var pluralIndex = 0; pluralIndex < self.nplurals; pluralIndex++) {
|
||||
lines = lines.concat(mkObsolete + _process(keyword, '', pluralIndex));
|
||||
}
|
||||
} else {
|
||||
var index = (self.msgid_plural && Array.isArray(text)) ?
|
||||
0 :
|
||||
undefined;
|
||||
text = Array.isArray(text) ? text.join() : text;
|
||||
var processed = _process(keyword, text, index);
|
||||
//handle \n in single-line texts (can not be handled in _escape)
|
||||
for (var i = 1; i < processed.length - 1; i++) {
|
||||
processed[i] = processed[i].slice(0, -1) + '\\n"';
|
||||
}
|
||||
var processed = _processLineBreak(keyword, text, index);
|
||||
lines = lines.concat(mkObsolete + processed.join('\n' + mkObsolete));
|
||||
}
|
||||
}
|
||||
|
||||
2
dist/pofile.min.js
vendored
2
dist/pofile.min.js
vendored
File diff suppressed because one or more lines are too long
65
lib/po.js
65
lib/po.js
@@ -20,12 +20,12 @@ PO.prototype.toString = function () {
|
||||
|
||||
if (this.comments) {
|
||||
this.comments.forEach(function (comment) {
|
||||
lines.push('# ' + comment);
|
||||
lines.push(('# ' + comment).trim());
|
||||
});
|
||||
}
|
||||
if (this.extractedComments) {
|
||||
this.extractedComments.forEach(function (comment) {
|
||||
lines.push('#. ' + comment);
|
||||
lines.push(('#. ' + comment).trim());
|
||||
});
|
||||
}
|
||||
|
||||
@@ -114,7 +114,9 @@ PO.parse = function (data) {
|
||||
}
|
||||
});
|
||||
|
||||
var item = new PO.Item();
|
||||
var parsedPluralForms = PO.parsePluralForms(po.headers['Plural-Forms']);
|
||||
var nplurals = parsedPluralForms.nplurals;
|
||||
var item = new PO.Item({ nplurals: nplurals });
|
||||
var context = null;
|
||||
var plural = 0;
|
||||
var obsoleteCount = 0;
|
||||
@@ -128,7 +130,7 @@ PO.parse = function (data) {
|
||||
obsoleteCount = 0;
|
||||
noCommentLineCount = 0;
|
||||
po.items.push(item);
|
||||
item = new PO.Item();
|
||||
item = new PO.Item({ nplurals: nplurals });
|
||||
}
|
||||
}
|
||||
|
||||
@@ -210,6 +212,7 @@ PO.parse = function (data) {
|
||||
} else if (line.match(/^msgctxt/)) { // Context
|
||||
finish();
|
||||
item.msgctxt = extract(line);
|
||||
context = 'msgctxt';
|
||||
noCommentLineCount++;
|
||||
} else { // Probably multiline string or blank
|
||||
if (line.length > 0) {
|
||||
@@ -220,6 +223,8 @@ PO.parse = function (data) {
|
||||
item.msgid += extract(line);
|
||||
} else if (context === 'msgid_plural') {
|
||||
item.msgid_plural += extract(line);
|
||||
} else if (context === 'msgctxt') {
|
||||
item.msgctxt += extract(line);
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -234,7 +239,26 @@ PO.parse = function (data) {
|
||||
return po;
|
||||
};
|
||||
|
||||
PO.Item = function () {
|
||||
PO.parsePluralForms = function (pluralFormsString) {
|
||||
var results = (pluralFormsString || '')
|
||||
.split(';')
|
||||
.reduce(function (acc, keyValueString) {
|
||||
var trimmedString = keyValueString.trim();
|
||||
var equalsIndex = trimmedString.indexOf('=');
|
||||
var key = trimmedString.substring(0, equalsIndex).trim();
|
||||
var value = trimmedString.substring(equalsIndex + 1).trim();
|
||||
acc[key] = value;
|
||||
return acc;
|
||||
}, {});
|
||||
return {
|
||||
nplurals: results.nplurals,
|
||||
plural: results.plural
|
||||
};
|
||||
};
|
||||
|
||||
PO.Item = function (options) {
|
||||
var nplurals = options && options.nplurals;
|
||||
|
||||
this.msgid = '';
|
||||
this.msgctxt = null;
|
||||
this.references = [];
|
||||
@@ -244,6 +268,8 @@ PO.Item = function () {
|
||||
this.extractedComments = [];
|
||||
this.flags = {};
|
||||
this.obsolete = false;
|
||||
var npluralsNumber = Number(nplurals);
|
||||
this.nplurals = (isNaN(npluralsNumber)) ? 2 : npluralsNumber;
|
||||
};
|
||||
|
||||
PO.Item.prototype.toString = function () {
|
||||
@@ -290,6 +316,15 @@ PO.Item.prototype.toString = function () {
|
||||
return lines;
|
||||
};
|
||||
|
||||
//handle \n in single-line texts (can not be handled in _escape)
|
||||
var _processLineBreak = function (keyword, text, index) {
|
||||
var processed = _process(keyword, text, index);
|
||||
for (var i = 1; i < processed.length - 1; i++) {
|
||||
processed[i] = processed[i].slice(0, -1) + '\\n"';
|
||||
}
|
||||
return processed;
|
||||
};
|
||||
|
||||
// https://www.gnu.org/software/gettext/manual/html_node/PO-Files.html
|
||||
// says order is translator-comments, extracted-comments, references, flags
|
||||
|
||||
@@ -316,20 +351,28 @@ PO.Item.prototype.toString = function () {
|
||||
['msgctxt', 'msgid', 'msgid_plural', 'msgstr'].forEach(function (keyword) {
|
||||
var text = self[keyword];
|
||||
if (text != null) {
|
||||
var hasTranslation = false;
|
||||
if (Array.isArray(text)) {
|
||||
hasTranslation = text.some(function (text) {
|
||||
return text;
|
||||
});
|
||||
}
|
||||
|
||||
if (Array.isArray(text) && text.length > 1) {
|
||||
text.forEach(function (t, i) {
|
||||
lines = lines.concat(mkObsolete + _process(keyword, t, i));
|
||||
var processed = _processLineBreak(keyword, t, i);
|
||||
lines = lines.concat(mkObsolete + processed.join('\n' + mkObsolete));
|
||||
});
|
||||
} else if (self.msgid_plural && keyword === 'msgstr' && !hasTranslation) {
|
||||
for (var pluralIndex = 0; pluralIndex < self.nplurals; pluralIndex++) {
|
||||
lines = lines.concat(mkObsolete + _process(keyword, '', pluralIndex));
|
||||
}
|
||||
} else {
|
||||
var index = (self.msgid_plural && Array.isArray(text)) ?
|
||||
0 :
|
||||
undefined;
|
||||
text = Array.isArray(text) ? text.join() : text;
|
||||
var processed = _process(keyword, text, index);
|
||||
//handle \n in single-line texts (can not be handled in _escape)
|
||||
for (var i = 1; i < processed.length - 1; i++) {
|
||||
processed[i] = processed[i].slice(0, -1) + '\\n"';
|
||||
}
|
||||
var processed = _processLineBreak(keyword, text, index);
|
||||
lines = lines.concat(mkObsolete + processed.join('\n' + mkObsolete));
|
||||
}
|
||||
}
|
||||
|
||||
@@ -1,7 +1,7 @@
|
||||
{
|
||||
"name": "pofile",
|
||||
"description": "Parse and serialize Gettext PO files.",
|
||||
"version": "1.0.4",
|
||||
"version": "1.0.10",
|
||||
"author": {
|
||||
"name": "Ruben Vermeersch",
|
||||
"email": "ruben@savanne.be",
|
||||
@@ -20,6 +20,7 @@
|
||||
"url": "http://github.com/rubenv/pofile.git"
|
||||
},
|
||||
"main": "./lib/po",
|
||||
"types": "./pofile.d.ts",
|
||||
"keywords": [
|
||||
"i18n",
|
||||
"l10n",
|
||||
|
||||
44
pofile.d.ts
vendored
Normal file
44
pofile.d.ts
vendored
Normal file
@@ -0,0 +1,44 @@
|
||||
declare module pofile {
|
||||
function parse(data: string): PO;
|
||||
function load(fileName: string, callback: (err: NodeJS.ErrnoException | null, po: PO) => void): void;
|
||||
|
||||
class PO {
|
||||
public comments: string[];
|
||||
public extractedComments: string[];
|
||||
public items: Item[];
|
||||
public headers: Partial<IHeaders>
|
||||
|
||||
public save(filename: string, callback: (err?: NodeJS.ErrnoException) => void): void;
|
||||
public toString(): string;
|
||||
}
|
||||
|
||||
interface IHeaders {
|
||||
'Project-Id-Version': string;
|
||||
'Report-Msgid-Bugs-To': string;
|
||||
'POT-Creation-Date': string;
|
||||
'PO-Revision-Date': string;
|
||||
'Last-Translator': string;
|
||||
'Language': string;
|
||||
'Language-Team': string;
|
||||
'Content-Type': string;
|
||||
'Content-Transfer-Encoding': string;
|
||||
'Plural-Forms': string;
|
||||
}
|
||||
|
||||
class Item {
|
||||
public msgid: string;
|
||||
public msgctxt?: string;
|
||||
public references: string[];
|
||||
public msgid_plural?: string;
|
||||
public msgstr: string[];
|
||||
public comments: string[];
|
||||
public extractedComments: string[];
|
||||
public flags: { [flag: string]: boolean | undefined }
|
||||
private nplurals: number;
|
||||
private obsolete: boolean;
|
||||
|
||||
public toString(): string;
|
||||
}
|
||||
}
|
||||
|
||||
export = pofile
|
||||
6
test/fixtures/big.po
vendored
6
test/fixtures/big.po
vendored
@@ -296,3 +296,9 @@ msgstr "This folder is empty."
|
||||
msgctxt "folder action"
|
||||
msgid "Empty folder"
|
||||
msgstr "Make this folder empty."
|
||||
|
||||
msgctxt ""
|
||||
"folder "
|
||||
"meta"
|
||||
msgid "Created Date"
|
||||
msgstr "Date de création"
|
||||
|
||||
14
test/fixtures/c-strings.po
vendored
14
test/fixtures/c-strings.po
vendored
@@ -57,3 +57,17 @@ msgid ""
|
||||
msgstr ""
|
||||
"\a\b\t\n"
|
||||
"\v\f\r"
|
||||
|
||||
# plural text must suport \n
|
||||
msgid ""
|
||||
"Test.\n"
|
||||
" multiline."
|
||||
msgid_plural ""
|
||||
"Test\n"
|
||||
" multiline plural."
|
||||
msgstr[0] ""
|
||||
"Test.\n"
|
||||
" mehrzeilig."
|
||||
msgstr[1] ""
|
||||
"Test.\n"
|
||||
" mehrzeilig Plural."
|
||||
|
||||
30
test/fixtures/plurals/messages.po
vendored
Normal file
30
test/fixtures/plurals/messages.po
vendored
Normal file
@@ -0,0 +1,30 @@
|
||||
# SOME DESCRIPTIVE TITLE.
|
||||
# Copyright (C) YEAR THE PACKAGE'S COPYRIGHT HOLDER
|
||||
# This file is distributed under the same license as the PACKAGE package.
|
||||
# FIRST AUTHOR <EMAIL@ADDRESS>, YEAR.
|
||||
#
|
||||
#, fuzzy
|
||||
msgid ""
|
||||
msgstr ""
|
||||
"Project-Id-Version: PACKAGE VERSION\n"
|
||||
"Report-Msgid-Bugs-To: \n"
|
||||
"POT-Creation-Date: 2017-02-13 09:59-0500\n"
|
||||
"PO-Revision-Date: YEAR-MO-DA HO:MI+ZONE\n"
|
||||
"Last-Translator: FULL NAME <EMAIL@ADDRESS>\n"
|
||||
"Language-Team: LANGUAGE <LL@li.org>\n"
|
||||
"Language: \n"
|
||||
"MIME-Version: 1.0\n"
|
||||
"Content-Type: text/plain; charset=CHARSET\n"
|
||||
"Content-Transfer-Encoding: 8bit\n"
|
||||
"Plural-Forms: nplurals=INTEGER; plural=EXPRESSION;\n"
|
||||
|
||||
# correct plurals
|
||||
msgid "1 thing"
|
||||
msgid_plural "{{$count}} things"
|
||||
msgstr[0] ""
|
||||
msgstr[1] ""
|
||||
|
||||
# incorrect plurals
|
||||
msgid "1 mistake"
|
||||
msgid_plural "{{$count}} mistakes"
|
||||
msgstr ""
|
||||
37
test/fixtures/plurals/nplurals-1.po
vendored
Normal file
37
test/fixtures/plurals/nplurals-1.po
vendored
Normal file
@@ -0,0 +1,37 @@
|
||||
# Chinese translation of Link (6.x-2.9)
|
||||
# Copyright (c) 2011 by the Chinese translation team
|
||||
#
|
||||
msgid ""
|
||||
msgstr ""
|
||||
"Project-Id-Version: Link (6.x-2.9)\n"
|
||||
"POT-Creation-Date: 2011-12-31 23:39+0000\n"
|
||||
"PO-Revision-Date: 2013-12-17 14:59+0100\n"
|
||||
"Language-Team: Chinese\n"
|
||||
"MIME-Version: 1.0\n"
|
||||
"Content-Type: text/plain; charset=UTF-8\n"
|
||||
"Content-Transfer-Encoding: 8bit\n"
|
||||
"Plural-Forms: nplurals=1; plural=0;\n"
|
||||
"Last-Translator: Ruben Vermeersch <ruben@rocketeer.be>\n"
|
||||
"Language: zh_CN\n"
|
||||
"X-Generator: Poedit 1.6.2\n"
|
||||
|
||||
# correct plurals, with translation
|
||||
msgid "1 thing"
|
||||
msgid_plural "{{$count}} things"
|
||||
msgstr[0] "{{$count}} thing"
|
||||
|
||||
# correct plurals, with no translation
|
||||
msgid "1 other thing"
|
||||
msgid_plural "{{$count}} other things"
|
||||
msgstr[0] ""
|
||||
|
||||
# incorrect plurals, with translation
|
||||
msgid "1 mistake"
|
||||
msgid_plural "{{$count}} mistakes"
|
||||
msgstr[0] "1 mistake"
|
||||
msgstr[1] "{{$count}} mistakes"
|
||||
|
||||
# incorrect plurals, with no translation
|
||||
msgid "1 other mistake"
|
||||
msgid_plural "{{$count}} other mistakes"
|
||||
msgstr ""
|
||||
@@ -16,17 +16,23 @@ msgstr ""
|
||||
"X-Generator: Poedit 1.6.2\n"
|
||||
|
||||
# correct plurals, with translation
|
||||
msgid "1 source"
|
||||
msgid_plural "{{$count}} sources"
|
||||
msgstr[0] "1 source"
|
||||
msgstr[1] "{{$count}} sources"
|
||||
msgid "1 thing"
|
||||
msgid_plural "{{$count}} things"
|
||||
msgstr[0] "1 thing"
|
||||
msgstr[1] "{{$count}} things"
|
||||
|
||||
# correct plurals, with no translation
|
||||
msgid "1 destination"
|
||||
msgid_plural "{{$count}} destinations"
|
||||
msgid "1 other thing"
|
||||
msgid_plural "{{$count}} other things"
|
||||
msgstr[0] ""
|
||||
msgstr[1] ""
|
||||
|
||||
# incorrect plurals, with no translation
|
||||
# incorrect plurals, with translation
|
||||
msgid "1 mistake"
|
||||
msgid_plural "{{$count}} mistakes"
|
||||
msgstr[0] "1 mistake"
|
||||
|
||||
# incorrect plurals, with no translation
|
||||
msgid "1 other mistake"
|
||||
msgid_plural "{{$count}} other mistakes"
|
||||
msgstr ""
|
||||
41
test/fixtures/plurals/nplurals-3.po
vendored
Normal file
41
test/fixtures/plurals/nplurals-3.po
vendored
Normal file
@@ -0,0 +1,41 @@
|
||||
# Polish translation of Link (6.x-2.9)
|
||||
# Copyright (c) 2011 by the Polish translation team
|
||||
#
|
||||
msgid ""
|
||||
msgstr ""
|
||||
"Project-Id-Version: Link (6.x-2.9)\n"
|
||||
"POT-Creation-Date: 2011-12-31 23:39+0000\n"
|
||||
"PO-Revision-Date: 2013-12-17 14:59+0100\n"
|
||||
"Language-Team: Polish\n"
|
||||
"MIME-Version: 1.0\n"
|
||||
"Content-Type: text/plain; charset=UTF-8\n"
|
||||
"Content-Transfer-Encoding: 8bit\n"
|
||||
"Plural-Forms: nplurals=3; plural=(n==1 ? 0 : n%10>=2 && n%10<=4 && (n%100<10 || n%100>=20) ? 1 : 2);\n"
|
||||
"Last-Translator: Ruben Vermeersch <ruben@rocketeer.be>\n"
|
||||
"Language: pl_PL\n"
|
||||
"X-Generator: Poedit 1.6.2\n"
|
||||
|
||||
# correct plurals, with translation
|
||||
msgid "1 thing"
|
||||
msgid_plural "{{$count}} things"
|
||||
msgstr[0] "1 thing"
|
||||
msgstr[1] "{{$count}} things"
|
||||
msgstr[2] "{{$count}} things"
|
||||
|
||||
# correct plurals, with no translation
|
||||
msgid "1 other thing"
|
||||
msgid_plural "{{$count}} other things"
|
||||
msgstr[0] ""
|
||||
msgstr[1] ""
|
||||
msgstr[2] ""
|
||||
|
||||
# incorrect plurals, with translation
|
||||
msgid "1 mistake"
|
||||
msgid_plural "{{$count}} mistakes"
|
||||
msgstr[0] "1 mistake"
|
||||
msgstr[1] "{{$count}} mistakes"
|
||||
|
||||
# incorrect plurals, with no translation
|
||||
msgid "1 other mistake"
|
||||
msgid_plural "{{$count}} other mistakes"
|
||||
msgstr ""
|
||||
49
test/fixtures/plurals/nplurals-6.po
vendored
Normal file
49
test/fixtures/plurals/nplurals-6.po
vendored
Normal file
@@ -0,0 +1,49 @@
|
||||
# Arabic translation of Link (6.x-2.9)
|
||||
# Copyright (c) 2011 by the Arabic translation team
|
||||
#
|
||||
msgid ""
|
||||
msgstr ""
|
||||
"Project-Id-Version: Link (6.x-2.9)\n"
|
||||
"POT-Creation-Date: 2011-12-31 23:39+0000\n"
|
||||
"PO-Revision-Date: 2013-12-17 14:59+0100\n"
|
||||
"Language-Team: Arabic\n"
|
||||
"MIME-Version: 1.0\n"
|
||||
"Content-Type: text/plain; charset=UTF-8\n"
|
||||
"Content-Transfer-Encoding: 8bit\n"
|
||||
"Plural-Forms: nplurals=6; plural=(n==0 ? 0 : n==1 ? 1 : n==2 ? 2 : n%100>=3 "
|
||||
"&& n%100<=10 ? 3 : n%100>=11 && n%100<=99 ? 4 : 5);\n"
|
||||
"Last-Translator: Ruben Vermeersch <ruben@rocketeer.be>\n"
|
||||
"Language: ar\n"
|
||||
"X-Generator: Poedit 1.6.2\n"
|
||||
|
||||
# correct plurals, with translation
|
||||
msgid "1 thing"
|
||||
msgid_plural "{{$count}} things"
|
||||
msgstr[0] "1 thing"
|
||||
msgstr[1] "{{$count}} things"
|
||||
msgstr[2] "{{$count}} things"
|
||||
msgstr[3] "{{$count}} things"
|
||||
msgstr[4] "{{$count}} things"
|
||||
msgstr[5] "{{$count}} things"
|
||||
|
||||
# correct plurals, with no translation
|
||||
msgid "1 other thing"
|
||||
msgid_plural "{{$count}} other things"
|
||||
msgstr[0] ""
|
||||
msgstr[1] ""
|
||||
msgstr[2] ""
|
||||
msgstr[3] ""
|
||||
msgstr[4] ""
|
||||
msgstr[5] ""
|
||||
|
||||
# incorrect plurals, with translation
|
||||
msgid "1 mistake"
|
||||
msgid_plural "{{$count}} mistakes"
|
||||
msgstr[0] "1 mistake"
|
||||
msgstr[1] "{{$count}} mistakes"
|
||||
msgstr[2] "{{$count}} mistakes"
|
||||
|
||||
# incorrect plurals, with no translation
|
||||
msgid "1 other mistake"
|
||||
msgid_plural "{{$count}} other mistakes"
|
||||
msgstr ""
|
||||
37
test/fixtures/plurals/nplurals-missing.po
vendored
Normal file
37
test/fixtures/plurals/nplurals-missing.po
vendored
Normal file
@@ -0,0 +1,37 @@
|
||||
# Chinese translation of Link (6.x-2.9)
|
||||
# Copyright (c) 2011 by the Chinese translation team
|
||||
#
|
||||
msgid ""
|
||||
msgstr ""
|
||||
"Project-Id-Version: Link (6.x-2.9)\n"
|
||||
"POT-Creation-Date: 2011-12-31 23:39+0000\n"
|
||||
"PO-Revision-Date: 2013-12-17 14:59+0100\n"
|
||||
"Language-Team: Chinese\n"
|
||||
"MIME-Version: 1.0\n"
|
||||
"Content-Type: text/plain; charset=UTF-8\n"
|
||||
"Content-Transfer-Encoding: 8bit\n"
|
||||
"Last-Translator: Ruben Vermeersch <ruben@rocketeer.be>\n"
|
||||
"Language: zh_CN\n"
|
||||
"X-Generator: Poedit 1.6.2\n"
|
||||
|
||||
# correct plurals, with translation
|
||||
msgid "1 thing"
|
||||
msgid_plural "{{$count}} things"
|
||||
msgstr[0] "1 thing"
|
||||
msgstr[1] "{{$count}} things"
|
||||
|
||||
# correct plurals, with no translation
|
||||
msgid "1 other thing"
|
||||
msgid_plural "{{$count}} other things"
|
||||
msgstr[0] ""
|
||||
msgstr[1] ""
|
||||
|
||||
# incorrect plurals, with translation
|
||||
msgid "1 mistake"
|
||||
msgid_plural "{{$count}} mistakes"
|
||||
msgstr[0] "1 mistake"
|
||||
|
||||
# incorrect plurals, with no translation
|
||||
msgid "1 other mistake"
|
||||
msgid_plural "{{$count}} other mistakes"
|
||||
msgstr ""
|
||||
52
test/parse-plural-forms.js
Normal file
52
test/parse-plural-forms.js
Normal file
@@ -0,0 +1,52 @@
|
||||
var assert = require('assert');
|
||||
var fs = require('fs');
|
||||
var PO = require('..');
|
||||
|
||||
describe('.parsePluralForms()', function () {
|
||||
it('should return an object with empty nplurals and plural expression when there is no plural forms header', function () {
|
||||
var expected = {
|
||||
nplurals: undefined,
|
||||
plural: undefined
|
||||
};
|
||||
assert.deepEqual(PO.parsePluralForms(), expected);
|
||||
assert.deepEqual(PO.parsePluralForms(null), expected);
|
||||
assert.deepEqual(PO.parsePluralForms(''), expected);
|
||||
});
|
||||
|
||||
it('should return an object with nplurals and plural set to xgettext\'s default output', function () {
|
||||
var pluralForms = 'nplurals=INTEGER; plural=EXPRESSION;';
|
||||
|
||||
var expected = {
|
||||
nplurals: 'INTEGER',
|
||||
plural: 'EXPRESSION'
|
||||
};
|
||||
var actual = PO.parsePluralForms(pluralForms);
|
||||
assert.deepEqual(actual, expected);
|
||||
});
|
||||
|
||||
it('should return an object with nplurals and plural set to typical string', function () {
|
||||
var pluralForms = 'nplurals=3; plural=(n==1 ? 0 : n%10>=2 && n%10<=4 && (n%100<10 || n%100>=20) ? 1 : 2);';
|
||||
|
||||
var expected = {
|
||||
nplurals: '3',
|
||||
plural: '(n==1 ? 0 : n%10>=2 && n%10<=4 && (n%100<10 || n%100>=20) ? 1 : 2)'
|
||||
};
|
||||
var actual = PO.parsePluralForms(pluralForms);
|
||||
assert.deepEqual(actual, expected);
|
||||
});
|
||||
|
||||
// node-gettext stores plural forms strings with spaces. They don't appear
|
||||
// to write PO files at all, but it seems prudent to handle this case
|
||||
// anyway. See
|
||||
// https://github.com/alexanderwallin/node-gettext/blob/v1.1.0/lib/plurals.js#L14
|
||||
it('should handle spaces around assignments in plural forms string', function () {
|
||||
var pluralForms = 'nplurals = 3; plural = (n==1 ? 0 : n%10>=2 && n%10<=4 && (n%100<10 || n%100>=20) ? 1 : 2);';
|
||||
|
||||
var expected = {
|
||||
nplurals: '3',
|
||||
plural: '(n==1 ? 0 : n%10>=2 && n%10<=4 && (n%100<10 || n%100>=20) ? 1 : 2)'
|
||||
};
|
||||
var actual = PO.parsePluralForms(pluralForms);
|
||||
assert.deepEqual(actual, expected);
|
||||
});
|
||||
});
|
||||
@@ -6,7 +6,7 @@ describe('Parse', function () {
|
||||
it('Parses the big po file', function () {
|
||||
var po = PO.parse(fs.readFileSync(__dirname + '/fixtures/big.po', 'utf8'));
|
||||
assert.notEqual(po, null);
|
||||
assert.equal(po.items.length, 69);
|
||||
assert.equal(po.items.length, 70);
|
||||
|
||||
var item = po.items[0];
|
||||
assert.equal(item.msgid, 'Title');
|
||||
@@ -121,6 +121,17 @@ describe('Parse', function () {
|
||||
assert.equal(ambiguousItems[1].msgctxt, 'folder action');
|
||||
});
|
||||
|
||||
it('Parses item multiline context', function () {
|
||||
var po = PO.parse(fs.readFileSync(__dirname + '/fixtures/big.po', 'utf8'));
|
||||
|
||||
var item = po.items.find(function (item) {
|
||||
return item.msgid === 'Created Date' && item.msgctxt === 'folder meta';
|
||||
});
|
||||
|
||||
assert.notEqual(item, undefined);
|
||||
assert.equal(item.msgctxt, 'folder meta');
|
||||
});
|
||||
|
||||
it('Handles obsolete items', function () {
|
||||
var po = PO.parse(fs.readFileSync(__dirname + '/fixtures/commented.po', 'utf8'));
|
||||
|
||||
|
||||
349
test/write.js
349
test/write.js
@@ -2,12 +2,13 @@ var assert = require('assert');
|
||||
var fs = require('fs');
|
||||
var PO = require('..');
|
||||
|
||||
function assertHasLine(str, line) {
|
||||
function assertHasLine(str, line, doNotTrim) {
|
||||
var lines = str.split('\n');
|
||||
var found = false;
|
||||
|
||||
for (var i = 0; i < lines.length; i++) {
|
||||
if (lines[i].trim() === line) {
|
||||
var lineToCompare = doNotTrim ? lines[i] : lines[i].trim();
|
||||
if (lineToCompare === line) {
|
||||
found = true;
|
||||
break;
|
||||
}
|
||||
@@ -53,6 +54,13 @@ describe('Write', function () {
|
||||
assertHasLine(str, '#, fuzzy');
|
||||
});
|
||||
|
||||
it('write empty comment without an unecessary space', function () {
|
||||
var input = fs.readFileSync(__dirname + '/fixtures/fuzzy.po', 'utf8');
|
||||
var po = PO.parse(input);
|
||||
var str = po.toString();
|
||||
assertHasLine(str, '#', true);
|
||||
});
|
||||
|
||||
it('write flags only when true', function () {
|
||||
var input = fs.readFileSync(__dirname + '/fixtures/fuzzy.po', 'utf8');
|
||||
var po = PO.parse(input);
|
||||
@@ -119,34 +127,323 @@ describe('Write', function () {
|
||||
});
|
||||
|
||||
describe('plurals', function () {
|
||||
it('should write multiple msgstrs', function () {
|
||||
var input = fs.readFileSync(__dirname + '/fixtures/plural.po', 'utf8');
|
||||
var po = PO.parse(input);
|
||||
var str = po.toString();
|
||||
assertHasContiguousLines(str, [
|
||||
'msgstr[0] "1 source"',
|
||||
'msgstr[1] "{{$count}} sources"'
|
||||
]);
|
||||
describe('nplurals INTEGER', function () {
|
||||
it('should write 2 msgstrs when formatted correctly', function () {
|
||||
var input = fs.readFileSync(
|
||||
__dirname + '/fixtures/plurals/messages.po', 'utf8'
|
||||
);
|
||||
var po = PO.parse(input);
|
||||
var str = po.toString();
|
||||
assertHasContiguousLines(str, [
|
||||
'msgid_plural "{{$count}} things"',
|
||||
'msgstr[0] ""',
|
||||
'msgstr[1] ""'
|
||||
]);
|
||||
});
|
||||
|
||||
it('should write 2 msgstrs when formatted incorrectly', function () {
|
||||
var input = fs.readFileSync(
|
||||
__dirname + '/fixtures/plurals/messages.po', 'utf8'
|
||||
);
|
||||
var po = PO.parse(input);
|
||||
var str = po.toString();
|
||||
assertHasContiguousLines(str, [
|
||||
'msgid_plural "{{$count}} mistakes"',
|
||||
'msgstr[0] ""',
|
||||
'msgstr[1] ""'
|
||||
]);
|
||||
});
|
||||
});
|
||||
|
||||
it('should write msgstr[0] when there is no translation', function () {
|
||||
var input = fs.readFileSync(__dirname + '/fixtures/plural.po', 'utf8');
|
||||
var po = PO.parse(input);
|
||||
var str = po.toString();
|
||||
assertHasContiguousLines(str, [
|
||||
'msgid_plural "{{$count}} destinations"',
|
||||
'msgstr[0] ""'
|
||||
]);
|
||||
describe('nplurals missing', function () {
|
||||
it('should write 2 msgstrs when formatted correctly with translation', function () {
|
||||
var input = fs.readFileSync(
|
||||
__dirname + '/fixtures/plurals/nplurals-missing.po', 'utf8'
|
||||
);
|
||||
var po = PO.parse(input);
|
||||
var str = po.toString();
|
||||
assertHasContiguousLines(str, [
|
||||
'msgid_plural "{{$count}} things"',
|
||||
'msgstr[0] "1 thing"',
|
||||
'msgstr[1] "{{$count}} things"'
|
||||
]);
|
||||
});
|
||||
|
||||
it('should write 2 msgstrs when formatted correctly with no translation', function () {
|
||||
var input = fs.readFileSync(
|
||||
__dirname + '/fixtures/plurals/nplurals-missing.po', 'utf8'
|
||||
);
|
||||
var po = PO.parse(input);
|
||||
var str = po.toString();
|
||||
assertHasContiguousLines(str, [
|
||||
'msgid_plural "{{$count}} other things"',
|
||||
'msgstr[0] ""',
|
||||
'msgstr[1] ""',
|
||||
]);
|
||||
});
|
||||
|
||||
it('should keep same number of msgstrs when formatted incorrectly with translation', function () {
|
||||
var input = fs.readFileSync(
|
||||
__dirname + '/fixtures/plurals/nplurals-missing.po', 'utf8'
|
||||
);
|
||||
var po = PO.parse(input);
|
||||
var str = po.toString();
|
||||
assertHasContiguousLines(str, [
|
||||
'msgid_plural "{{$count}} mistakes"',
|
||||
'msgstr[0] "1 mistake"',
|
||||
'',
|
||||
'# incorrect plurals, with no translation'
|
||||
]);
|
||||
});
|
||||
|
||||
it('should write 2 msgstrs when formatted incorrectly with no translation', function () {
|
||||
var input = fs.readFileSync(
|
||||
__dirname + '/fixtures/plurals/nplurals-missing.po', 'utf8'
|
||||
);
|
||||
var po = PO.parse(input);
|
||||
var str = po.toString();
|
||||
assertHasContiguousLines(str, [
|
||||
'msgid_plural "{{$count}} other mistakes"',
|
||||
'msgstr[0] ""',
|
||||
'msgstr[1] ""'
|
||||
]);
|
||||
});
|
||||
});
|
||||
|
||||
it('should write msgstr[0] when there is no translation and empty plural translation is stored in msgstr ""', function () {
|
||||
var input = fs.readFileSync(__dirname + '/fixtures/plural.po', 'utf8');
|
||||
var po = PO.parse(input);
|
||||
var str = po.toString();
|
||||
assertHasContiguousLines(str, [
|
||||
'msgid_plural "{{$count}} mistakes"',
|
||||
'msgstr[0] ""'
|
||||
]);
|
||||
describe('nplurals=1', function () {
|
||||
it('should write 1 msgstr when formatted correctly with translation', function () {
|
||||
var input = fs.readFileSync(
|
||||
__dirname + '/fixtures/plurals/nplurals-1.po', 'utf8'
|
||||
);
|
||||
var po = PO.parse(input);
|
||||
var str = po.toString();
|
||||
assertHasContiguousLines(str, [
|
||||
'msgid_plural "{{$count}} things"',
|
||||
'msgstr[0] "{{$count}} thing"'
|
||||
]);
|
||||
});
|
||||
|
||||
it('should write 1 msgstr when formatted correctly with no translation', function () {
|
||||
var input = fs.readFileSync(
|
||||
__dirname + '/fixtures/plurals/nplurals-1.po', 'utf8'
|
||||
);
|
||||
var po = PO.parse(input);
|
||||
var str = po.toString();
|
||||
assertHasContiguousLines(str, [
|
||||
'msgid_plural "{{$count}} other things"',
|
||||
'msgstr[0] ""',
|
||||
'',
|
||||
'# incorrect plurals, with translation'
|
||||
]);
|
||||
});
|
||||
|
||||
it('should keep same number of msgstrs when formatted incorrectly with translation', function () {
|
||||
var input = fs.readFileSync(
|
||||
__dirname + '/fixtures/plurals/nplurals-1.po', 'utf8'
|
||||
);
|
||||
var po = PO.parse(input);
|
||||
var str = po.toString();
|
||||
assertHasContiguousLines(str, [
|
||||
'msgid_plural "{{$count}} mistakes"',
|
||||
'msgstr[0] "1 mistake"',
|
||||
'msgstr[1] "{{$count}} mistakes"'
|
||||
]);
|
||||
});
|
||||
|
||||
it('should write 1 msgstr when formatted incorrectly with no translation', function () {
|
||||
var input = fs.readFileSync(
|
||||
__dirname + '/fixtures/plurals/nplurals-1.po', 'utf8'
|
||||
);
|
||||
var po = PO.parse(input);
|
||||
var str = po.toString();
|
||||
assertHasContiguousLines(str, [
|
||||
'msgid_plural "{{$count}} other mistakes"',
|
||||
'msgstr[0] ""'
|
||||
]);
|
||||
});
|
||||
});
|
||||
|
||||
describe('nplurals=2', function () {
|
||||
it('should write 2 msgstrs when formatted correctly with translation', function () {
|
||||
var input = fs.readFileSync(
|
||||
__dirname + '/fixtures/plurals/nplurals-2.po', 'utf8'
|
||||
);
|
||||
var po = PO.parse(input);
|
||||
var str = po.toString();
|
||||
assertHasContiguousLines(str, [
|
||||
'msgid_plural "{{$count}} things"',
|
||||
'msgstr[0] "1 thing"',
|
||||
'msgstr[1] "{{$count}} things"'
|
||||
]);
|
||||
});
|
||||
|
||||
it('should write 2 msgstrs when formatted correctly with no translation', function () {
|
||||
var input = fs.readFileSync(
|
||||
__dirname + '/fixtures/plurals/nplurals-2.po', 'utf8'
|
||||
);
|
||||
var po = PO.parse(input);
|
||||
var str = po.toString();
|
||||
assertHasContiguousLines(str, [
|
||||
'msgid_plural "{{$count}} other things"',
|
||||
'msgstr[0] ""',
|
||||
'msgstr[1] ""',
|
||||
]);
|
||||
});
|
||||
|
||||
it('should keep same number of msgstrs when formatted incorrectly with translation', function () {
|
||||
var input = fs.readFileSync(
|
||||
__dirname + '/fixtures/plurals/nplurals-2.po', 'utf8'
|
||||
);
|
||||
var po = PO.parse(input);
|
||||
var str = po.toString();
|
||||
assertHasContiguousLines(str, [
|
||||
'msgid_plural "{{$count}} mistakes"',
|
||||
'msgstr[0] "1 mistake"',
|
||||
'',
|
||||
'# incorrect plurals, with no translation'
|
||||
]);
|
||||
});
|
||||
|
||||
it('should write 2 msgstrs when formatted incorrectly with no translation', function () {
|
||||
var input = fs.readFileSync(
|
||||
__dirname + '/fixtures/plurals/nplurals-2.po', 'utf8'
|
||||
);
|
||||
var po = PO.parse(input);
|
||||
var str = po.toString();
|
||||
assertHasContiguousLines(str, [
|
||||
'msgid_plural "{{$count}} other mistakes"',
|
||||
'msgstr[0] ""',
|
||||
'msgstr[1] ""'
|
||||
]);
|
||||
});
|
||||
});
|
||||
|
||||
describe('nplurals=3', function () {
|
||||
it('should write 3 msgstrs when formatted correctly with translation', function () {
|
||||
var input = fs.readFileSync(
|
||||
__dirname + '/fixtures/plurals/nplurals-3.po', 'utf8'
|
||||
);
|
||||
var po = PO.parse(input);
|
||||
var str = po.toString();
|
||||
assertHasContiguousLines(str, [
|
||||
'msgid_plural "{{$count}} things"',
|
||||
'msgstr[0] "1 thing"',
|
||||
'msgstr[1] "{{$count}} things"',
|
||||
'msgstr[2] "{{$count}} things"'
|
||||
]);
|
||||
});
|
||||
|
||||
it('should write 3 msgstrs when formatted correctly with no translation', function () {
|
||||
var input = fs.readFileSync(
|
||||
__dirname + '/fixtures/plurals/nplurals-3.po', 'utf8'
|
||||
);
|
||||
var po = PO.parse(input);
|
||||
var str = po.toString();
|
||||
assertHasContiguousLines(str, [
|
||||
'msgid_plural "{{$count}} other things"',
|
||||
'msgstr[0] ""',
|
||||
'msgstr[1] ""',
|
||||
'msgstr[2] ""'
|
||||
]);
|
||||
});
|
||||
|
||||
it('should keep same number of msgstrs when formatted incorrectly with translation', function () {
|
||||
var input = fs.readFileSync(
|
||||
__dirname + '/fixtures/plurals/nplurals-3.po', 'utf8'
|
||||
);
|
||||
var po = PO.parse(input);
|
||||
var str = po.toString();
|
||||
assertHasContiguousLines(str, [
|
||||
'msgid_plural "{{$count}} mistakes"',
|
||||
'msgstr[0] "1 mistake"',
|
||||
'msgstr[1] "{{$count}} mistakes"',
|
||||
'',
|
||||
'# incorrect plurals, with no translation'
|
||||
]);
|
||||
});
|
||||
|
||||
it('should write 3 msgstrs when formatted incorrectly with no translation', function () {
|
||||
var input = fs.readFileSync(
|
||||
__dirname + '/fixtures/plurals/nplurals-3.po', 'utf8'
|
||||
);
|
||||
var po = PO.parse(input);
|
||||
var str = po.toString();
|
||||
assertHasContiguousLines(str, [
|
||||
'msgid_plural "{{$count}} other mistakes"',
|
||||
'msgstr[0] ""',
|
||||
'msgstr[1] ""',
|
||||
'msgstr[2] ""'
|
||||
]);
|
||||
});
|
||||
});
|
||||
|
||||
describe('nplurals=6', function () {
|
||||
it('should write 6 msgstrs when formatted correctly with translation', function () {
|
||||
var input = fs.readFileSync(
|
||||
__dirname + '/fixtures/plurals/nplurals-6.po', 'utf8'
|
||||
);
|
||||
var po = PO.parse(input);
|
||||
var str = po.toString();
|
||||
assertHasContiguousLines(str, [
|
||||
'msgid_plural "{{$count}} things"',
|
||||
'msgstr[0] "1 thing"',
|
||||
'msgstr[1] "{{$count}} things"',
|
||||
'msgstr[2] "{{$count}} things"',
|
||||
'msgstr[3] "{{$count}} things"',
|
||||
'msgstr[4] "{{$count}} things"',
|
||||
'msgstr[5] "{{$count}} things"'
|
||||
]);
|
||||
});
|
||||
|
||||
it('should write 6 msgstrs when formatted correctly with no translation', function () {
|
||||
var input = fs.readFileSync(
|
||||
__dirname + '/fixtures/plurals/nplurals-6.po', 'utf8'
|
||||
);
|
||||
var po = PO.parse(input);
|
||||
var str = po.toString();
|
||||
assertHasContiguousLines(str, [
|
||||
'msgid_plural "{{$count}} other things"',
|
||||
'msgstr[0] ""',
|
||||
'msgstr[1] ""',
|
||||
'msgstr[2] ""',
|
||||
'msgstr[3] ""',
|
||||
'msgstr[4] ""',
|
||||
'msgstr[5] ""'
|
||||
]);
|
||||
});
|
||||
|
||||
it('should keep same number of msgstrs when formatted incorrectly with translation', function () {
|
||||
var input = fs.readFileSync(
|
||||
__dirname + '/fixtures/plurals/nplurals-6.po', 'utf8'
|
||||
);
|
||||
var po = PO.parse(input);
|
||||
var str = po.toString();
|
||||
assertHasContiguousLines(str, [
|
||||
'msgid_plural "{{$count}} mistakes"',
|
||||
'msgstr[0] "1 mistake"',
|
||||
'msgstr[1] "{{$count}} mistakes"',
|
||||
'msgstr[2] "{{$count}} mistakes"',
|
||||
'',
|
||||
'# incorrect plurals, with no translation'
|
||||
]);
|
||||
});
|
||||
|
||||
it('should write 6 msgstrs when formatted incorrectly with no translation', function () {
|
||||
var input = fs.readFileSync(
|
||||
__dirname + '/fixtures/plurals/nplurals-6.po', 'utf8'
|
||||
);
|
||||
var po = PO.parse(input);
|
||||
var str = po.toString();
|
||||
assertHasContiguousLines(str, [
|
||||
'msgid_plural "{{$count}} other mistakes"',
|
||||
'msgstr[0] ""',
|
||||
'msgstr[1] ""',
|
||||
'msgstr[2] ""',
|
||||
'msgstr[3] ""',
|
||||
'msgstr[4] ""',
|
||||
'msgstr[5] ""'
|
||||
]);
|
||||
});
|
||||
});
|
||||
});
|
||||
|
||||
|
||||
Reference in New Issue
Block a user