Compare commits
87 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
ff1b888af1 | ||
|
|
3535e60695 | ||
|
|
11b1affe5b | ||
|
|
29d40186c9 | ||
|
|
d6937a7da2 | ||
|
|
c62b82a98d | ||
|
|
fe23027f32 | ||
|
|
d92e1f9e89 | ||
|
|
0359aa12a4 | ||
|
|
6357bf3edd | ||
|
|
3b28b3ed08 | ||
|
|
207308a1ac | ||
|
|
b8676a4fe6 | ||
|
|
bf879ca5ee | ||
|
|
8bd7810703 | ||
|
|
ad71dba6ad | ||
|
|
ab3d6f8405 | ||
|
|
58ee1abedb | ||
|
|
0aef81f4e9 | ||
|
|
579a5635a6 | ||
|
|
03a257b40d | ||
|
|
58875dc119 | ||
|
|
a6937bb7f0 | ||
|
|
cfe01e8aea | ||
|
|
14bad962eb | ||
|
|
ff0e5655f9 | ||
|
|
bb56f6e34c | ||
|
|
f035affef6 | ||
|
|
18a4cc0cb5 | ||
|
|
0fcff887fe | ||
|
|
9689ae5b7f | ||
|
|
5cbb657f20 | ||
|
|
e038f25d5b | ||
|
|
f26ecb0d63 | ||
|
|
3324041669 | ||
|
|
d30a02f3c2 | ||
|
|
0e3a6d74f3 | ||
|
|
d783222d37 | ||
|
|
36f5d05828 | ||
|
|
8fe9cd7bf8 | ||
|
|
48fa2c39f3 | ||
|
|
037edff739 | ||
|
|
16f423879a | ||
|
|
52c11f77e9 | ||
|
|
115459160e | ||
|
|
79407fcf5e | ||
|
|
e2330bd433 | ||
|
|
853823f214 | ||
|
|
832962d30d | ||
|
|
f6119a57af | ||
|
|
6c09df93a2 | ||
|
|
1bf498ecf7 | ||
|
|
71bb04f046 | ||
|
|
ba9a2db453 | ||
|
|
fbe773c636 | ||
|
|
7ceda82794 | ||
|
|
d1be0f51b0 | ||
|
|
869f763d80 | ||
|
|
b9394176b1 | ||
|
|
9060221403 | ||
|
|
63c4209cd5 | ||
|
|
104d114d5d | ||
|
|
db5e540824 | ||
|
|
cfc9b2ae82 | ||
|
|
2e1640d847 | ||
|
|
b499b7f449 | ||
|
|
e42dc28fd2 | ||
|
|
e1742e66a6 | ||
|
|
4cfebdee80 | ||
|
|
d8fc514359 | ||
|
|
d202e39a60 | ||
|
|
f5056bc57f | ||
|
|
af94d8ff5e | ||
|
|
8e49417916 | ||
|
|
b9534d20fe | ||
|
|
1b5668b5c7 | ||
|
|
08e7db58b3 | ||
|
|
eeb1382dfb | ||
|
|
656bfd0b8d | ||
|
|
03572f9711 | ||
|
|
6af1cf741f | ||
|
|
e164fcfe9d | ||
|
|
8d40e1b3c4 | ||
|
|
04fe915389 | ||
|
|
2fb07fb1d0 | ||
|
|
06571c89fc | ||
|
|
73b267b3e8 |
34
.jscs.json
Normal file
34
.jscs.json
Normal file
@@ -0,0 +1,34 @@
|
|||||||
|
{
|
||||||
|
"requireCurlyBraces": ["if", "else", "for", "while", "do", "try", "catch"],
|
||||||
|
"requireSpaceAfterKeywords": ["if", "else", "for", "while", "do", "switch", "return", "try", "catch"],
|
||||||
|
"requireParenthesesAroundIIFE": true,
|
||||||
|
"requireSpacesInFunctionExpression": {
|
||||||
|
"beforeOpeningCurlyBrace": true
|
||||||
|
},
|
||||||
|
"requireSpacesInAnonymousFunctionExpression": {
|
||||||
|
"beforeOpeningRoundBrace": true
|
||||||
|
},
|
||||||
|
"disallowSpacesInNamedFunctionExpression": {
|
||||||
|
"beforeOpeningRoundBrace": true
|
||||||
|
},
|
||||||
|
"disallowSpacesInFunctionDeclaration": {
|
||||||
|
"beforeOpeningRoundBrace": true
|
||||||
|
},
|
||||||
|
"disallowMultipleVarDecl": true,
|
||||||
|
"requireSpacesInsideObjectBrackets": "all",
|
||||||
|
"disallowSpaceAfterObjectKeys": true,
|
||||||
|
"requireCommaBeforeLineBreak": true,
|
||||||
|
"disallowSpaceBeforeBinaryOperators": [","],
|
||||||
|
"disallowSpaceAfterPrefixUnaryOperators": ["++", "--", "+", "-", "~", "!"],
|
||||||
|
"disallowSpaceBeforePostfixUnaryOperators": ["++", "--"],
|
||||||
|
"requireSpaceBeforeBinaryOperators": ["+", "-", "/", "*", "=", "==", "===", "!=", "!==", ">", ">=", "<", "<="],
|
||||||
|
"requireSpaceAfterBinaryOperators": [",", "+", "-", "/", "*", "=", "==", "===", "!=", "!==", ">", ">=", "<", "<="],
|
||||||
|
"validateQuoteMarks": true,
|
||||||
|
"validateIndentation": 4,
|
||||||
|
"disallowTrailingWhitespace": true,
|
||||||
|
"disallowKeywordsOnNewLine": ["else"],
|
||||||
|
"requireCapitalizedConstructors": true,
|
||||||
|
"safeContextKeyword": "self",
|
||||||
|
"requireDotNotation": true,
|
||||||
|
"disallowYodaConditions": true
|
||||||
|
}
|
||||||
@@ -1,7 +1,5 @@
|
|||||||
language: node_js
|
language: node_js
|
||||||
node_js:
|
node_js:
|
||||||
- "0.8"
|
- "node"
|
||||||
- "0.10"
|
|
||||||
- "0.11"
|
|
||||||
before_install:
|
before_install:
|
||||||
- npm install -g grunt-cli
|
- npm install -g grunt-cli
|
||||||
|
|||||||
@@ -5,6 +5,7 @@ module.exports = (grunt) ->
|
|||||||
@loadNpmTasks('grunt-contrib-jshint')
|
@loadNpmTasks('grunt-contrib-jshint')
|
||||||
@loadNpmTasks('grunt-contrib-uglify')
|
@loadNpmTasks('grunt-contrib-uglify')
|
||||||
@loadNpmTasks('grunt-contrib-watch')
|
@loadNpmTasks('grunt-contrib-watch')
|
||||||
|
@loadNpmTasks('grunt-jscs')
|
||||||
@loadNpmTasks('grunt-mocha-cli')
|
@loadNpmTasks('grunt-mocha-cli')
|
||||||
|
|
||||||
@initConfig
|
@initConfig
|
||||||
@@ -16,6 +17,13 @@ module.exports = (grunt) ->
|
|||||||
options:
|
options:
|
||||||
jshintrc: '.jshintrc'
|
jshintrc: '.jshintrc'
|
||||||
|
|
||||||
|
jscs:
|
||||||
|
src:
|
||||||
|
options:
|
||||||
|
config: '.jscs.json'
|
||||||
|
files:
|
||||||
|
src: [ 'lib/*.js', 'test/*.js' ]
|
||||||
|
|
||||||
watch:
|
watch:
|
||||||
all:
|
all:
|
||||||
options:
|
options:
|
||||||
@@ -35,7 +43,8 @@ module.exports = (grunt) ->
|
|||||||
files:
|
files:
|
||||||
'dist/pofile.js': ['lib/po.js']
|
'dist/pofile.js': ['lib/po.js']
|
||||||
options:
|
options:
|
||||||
alias: 'lib/po.js:pofile'
|
alias:
|
||||||
|
pofile: './lib/po.js'
|
||||||
|
|
||||||
uglify:
|
uglify:
|
||||||
dist:
|
dist:
|
||||||
@@ -49,5 +58,5 @@ module.exports = (grunt) ->
|
|||||||
pushTo: 'origin'
|
pushTo: 'origin'
|
||||||
|
|
||||||
@registerTask 'default', ['test']
|
@registerTask 'default', ['test']
|
||||||
@registerTask 'build', ['clean', 'jshint', 'browserify', 'uglify']
|
@registerTask 'build', ['clean', 'jshint', 'jscs', 'browserify', 'uglify']
|
||||||
@registerTask 'test', ['build', 'mochacli']
|
@registerTask 'test', ['build', 'mochacli']
|
||||||
|
|||||||
2
LICENSE
2
LICENSE
@@ -1,4 +1,4 @@
|
|||||||
Copyright (C) 2013-2014 Ruben Vermeersch
|
Copyright (C) 2013-2017 Ruben Vermeersch
|
||||||
Copyright (C) 2012 Michael Holly
|
Copyright (C) 2012 Michael Holly
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy of
|
Permission is hereby granted, free of charge, to any person obtaining a copy of
|
||||||
|
|||||||
@@ -81,7 +81,7 @@ po.save('out.po', function (err) {
|
|||||||
|
|
||||||
### The PO.Item class
|
### The PO.Item class
|
||||||
|
|
||||||
The `PO` class exposes the following members:
|
The `PO.Item` class exposes the following members:
|
||||||
|
|
||||||
* `msgid`: The message id.
|
* `msgid`: The message id.
|
||||||
* `msgid_plural`: The plural message id (null if absent).
|
* `msgid_plural`: The plural message id (null if absent).
|
||||||
@@ -143,7 +143,7 @@ PO.load('text.po', function (err, po) {
|
|||||||
|
|
||||||
(The MIT License)
|
(The MIT License)
|
||||||
|
|
||||||
Copyright (C) 2013-2014 by Ruben Vermeersch <ruben@rocketeer.be>
|
Copyright (C) 2013-2017 by Ruben Vermeersch <ruben@rocketeer.be>
|
||||||
Copyright (C) 2012 by Michael Holly
|
Copyright (C) 2012 by Michael Holly
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "pofile",
|
"name": "pofile",
|
||||||
"version": "0.2.7",
|
"version": "1.1.1",
|
||||||
"authors": [
|
"authors": [
|
||||||
"Ruben Vermeersch <ruben@rocketeer.be>"
|
"Ruben Vermeersch <ruben@rocketeer.be>"
|
||||||
],
|
],
|
||||||
|
|||||||
393
dist/pofile.js
vendored
393
dist/pofile.js
vendored
@@ -1,37 +1,58 @@
|
|||||||
require=(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require=="function"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);throw new Error("Cannot find module '"+o+"'")}var f=n[o]={exports:{}};t[o][0].call(f.exports,function(e){var n=t[o][1][e];return s(n?n:e)},f,f.exports,e,t,n,r)}return n[o].exports}var i=typeof require=="function"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})({"W8CkM0":[function(require,module,exports){
|
require=(function(){function r(e,n,t){function o(i,f){if(!n[i]){if(!e[i]){var c="function"==typeof require&&require;if(!f&&c)return c(i,!0);if(u)return u(i,!0);var a=new Error("Cannot find module '"+i+"'");throw a.code="MODULE_NOT_FOUND",a}var p=n[i]={exports:{}};e[i][0].call(p.exports,function(r){var n=e[i][1][r];return o(n||r)},p,p.exports,r,e,n,t)}return n[i].exports}for(var u="function"==typeof require&&require,i=0;i<t.length;i++)o(t[i]);return o}return r})()({1:[function(require,module,exports){
|
||||||
var fs = require('fs'),
|
|
||||||
isArray = require('lodash.isarray');
|
|
||||||
|
|
||||||
|
},{}],"pofile":[function(require,module,exports){
|
||||||
function trim(string) {
|
function trim(string) {
|
||||||
return string.replace(/^\s+|\s+$/g, '');
|
return string.replace(/^\s+|\s+$/g, '');
|
||||||
}
|
}
|
||||||
|
|
||||||
var PO = function () {
|
var PO = function () {
|
||||||
this.comments = [];
|
this.comments = [];
|
||||||
|
this.extractedComments = [];
|
||||||
this.headers = {};
|
this.headers = {};
|
||||||
|
this.headerOrder = [];
|
||||||
this.items = [];
|
this.items = [];
|
||||||
};
|
};
|
||||||
|
|
||||||
PO.prototype.save = function (filename, callback) {
|
PO.prototype.save = function (filename, callback) {
|
||||||
fs.writeFile(filename, this.toString(), callback);
|
require('fs').writeFile(filename, this.toString(), callback);
|
||||||
};
|
};
|
||||||
|
|
||||||
PO.prototype.toString = function () {
|
PO.prototype.toString = function () {
|
||||||
var lines = [],
|
var lines = [];
|
||||||
that = this;
|
|
||||||
|
|
||||||
if (this.comments) {
|
if (this.comments) {
|
||||||
this.comments.forEach(function (comment) {
|
this.comments.forEach(function (comment) {
|
||||||
lines.push('# ' + comment);
|
lines.push(('# ' + comment).trim());
|
||||||
|
});
|
||||||
|
}
|
||||||
|
if (this.extractedComments) {
|
||||||
|
this.extractedComments.forEach(function (comment) {
|
||||||
|
lines.push(('#. ' + comment).trim());
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
lines.push('msgid ""');
|
lines.push('msgid ""');
|
||||||
lines.push('msgstr ""');
|
lines.push('msgstr ""');
|
||||||
|
|
||||||
|
var self = this;
|
||||||
|
var headerOrder = [];
|
||||||
|
|
||||||
|
this.headerOrder.forEach(function (key) {
|
||||||
|
if (key in self.headers) {
|
||||||
|
headerOrder.push(key);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
var keys = Object.keys(this.headers);
|
var keys = Object.keys(this.headers);
|
||||||
|
|
||||||
keys.forEach(function (key) {
|
keys.forEach(function (key) {
|
||||||
lines.push('"' + key + ': ' + that.headers[key] + '\\n"');
|
if (headerOrder.indexOf(key) === -1) {
|
||||||
|
headerOrder.push(key);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
headerOrder.forEach(function (key) {
|
||||||
|
lines.push('"' + key + ': ' + self.headers[key] + '\\n"');
|
||||||
});
|
});
|
||||||
|
|
||||||
lines.push('');
|
lines.push('');
|
||||||
@@ -41,11 +62,11 @@ PO.prototype.toString = function () {
|
|||||||
lines.push('');
|
lines.push('');
|
||||||
});
|
});
|
||||||
|
|
||||||
return lines.join("\n");
|
return lines.join('\n');
|
||||||
};
|
};
|
||||||
|
|
||||||
PO.load = function (filename, callback) {
|
PO.load = function (filename, callback) {
|
||||||
fs.readFile(filename, 'utf-8', function (err, data) {
|
require('fs').readFile(filename, 'utf-8', function (err, data) {
|
||||||
if (err) {
|
if (err) {
|
||||||
return callback(err);
|
return callback(err);
|
||||||
}
|
}
|
||||||
@@ -57,10 +78,20 @@ PO.load = function (filename, callback) {
|
|||||||
PO.parse = function (data) {
|
PO.parse = function (data) {
|
||||||
//support both unix and windows newline formats.
|
//support both unix and windows newline formats.
|
||||||
data = data.replace(/\r\n/g, '\n');
|
data = data.replace(/\r\n/g, '\n');
|
||||||
var po = new PO(),
|
var po = new PO();
|
||||||
sections = data.split(/\n\n/),
|
var sections = data.split(/\n\n/);
|
||||||
headers = sections.shift(),
|
var headers = [];
|
||||||
lines = sections.join("\n").split(/\n/);
|
//everything until the first 'msgid ""' is considered header
|
||||||
|
while (sections[0] && (headers.length === 0 || headers[headers.length - 1].indexOf('msgid ""') < 0)) {
|
||||||
|
if (sections[0].match(/msgid "[^"]/)) {
|
||||||
|
//found first real string, adding a dummy header item
|
||||||
|
headers.push('msgid ""');
|
||||||
|
} else {
|
||||||
|
headers.push(sections.shift());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
headers = headers.join('\n');
|
||||||
|
var lines = sections.join('\n').split(/\n/);
|
||||||
|
|
||||||
po.headers = {
|
po.headers = {
|
||||||
'Project-Id-Version': '',
|
'Project-Id-Version': '',
|
||||||
@@ -74,111 +105,179 @@ PO.parse = function (data) {
|
|||||||
'Content-Transfer-Encoding': '',
|
'Content-Transfer-Encoding': '',
|
||||||
'Plural-Forms': '',
|
'Plural-Forms': '',
|
||||||
};
|
};
|
||||||
|
po.headerOrder = [];
|
||||||
|
|
||||||
headers.split(/\n/).forEach(function (header) {
|
headers.split(/\n/).reduce(function (acc, line) {
|
||||||
if (header.match(/^#/)) {
|
if (acc.merge) {
|
||||||
po.comments.push(header.replace(/^#\s*/, ''));
|
//join lines, remove last resp. first "
|
||||||
|
line = acc.pop().slice(0, -1) + line.slice(1);
|
||||||
|
delete acc.merge;
|
||||||
}
|
}
|
||||||
if (header.match(/^"/)) {
|
if (/^".*"$/.test(line) && !/^".*\\n"$/.test(line)) {
|
||||||
|
acc.merge = true;
|
||||||
|
}
|
||||||
|
acc.push(line);
|
||||||
|
return acc;
|
||||||
|
}, []).forEach(function (header) {
|
||||||
|
if (header.match(/^#\./)) {
|
||||||
|
po.extractedComments.push(header.replace(/^#\.\s*/, ''));
|
||||||
|
} else if (header.match(/^#/)) {
|
||||||
|
po.comments.push(header.replace(/^#\s*/, ''));
|
||||||
|
} else if (header.match(/^"/)) {
|
||||||
header = header.trim().replace(/^"/, '').replace(/\\n"$/, '');
|
header = header.trim().replace(/^"/, '').replace(/\\n"$/, '');
|
||||||
var p = header.split(/:/),
|
var p = header.split(/:/);
|
||||||
name = p.shift().trim(),
|
var name = p.shift().trim();
|
||||||
value = p.join(':').trim();
|
var value = p.join(':').trim();
|
||||||
po.headers[name] = value;
|
po.headers[name] = value;
|
||||||
|
po.headerOrder.push(name);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
var item = new PO.Item(),
|
var parsedPluralForms = PO.parsePluralForms(po.headers['Plural-Forms']);
|
||||||
context = null,
|
var nplurals = parsedPluralForms.nplurals;
|
||||||
plural = 0,
|
var item = new PO.Item({ nplurals: nplurals });
|
||||||
obsolete = false;
|
var context = null;
|
||||||
|
var plural = 0;
|
||||||
|
var obsoleteCount = 0;
|
||||||
|
var noCommentLineCount = 0;
|
||||||
|
|
||||||
function finish() {
|
function finish() {
|
||||||
if (item.msgid.length > 0) {
|
if (item.msgid.length > 0) {
|
||||||
|
if (obsoleteCount >= noCommentLineCount) {
|
||||||
|
item.obsolete = true;
|
||||||
|
}
|
||||||
|
obsoleteCount = 0;
|
||||||
|
noCommentLineCount = 0;
|
||||||
po.items.push(item);
|
po.items.push(item);
|
||||||
item = new PO.Item();
|
item = new PO.Item({ nplurals: nplurals });
|
||||||
item.obsolete = obsolete;
|
|
||||||
obsolete = false;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function extract(string) {
|
function extract(string) {
|
||||||
string = trim(string);
|
string = trim(string);
|
||||||
string = string.replace(/^[^"]*"|"$/g, '');
|
string = string.replace(/^[^"]*"|"$/g, '');
|
||||||
string = string.replace(/\\"/g, '"');
|
string = string.replace(/\\([abtnvfr'"\\?]|([0-7]{3})|x([0-9a-fA-F]{2}))/g, function (match, esc, oct, hex) {
|
||||||
string = string.replace(/\\\\/g, '\\');
|
if (oct) {
|
||||||
|
return String.fromCharCode(parseInt(oct, 8));
|
||||||
|
}
|
||||||
|
if (hex) {
|
||||||
|
return String.fromCharCode(parseInt(hex, 16));
|
||||||
|
}
|
||||||
|
switch (esc) {
|
||||||
|
case 'a':
|
||||||
|
return '\x07';
|
||||||
|
case 'b':
|
||||||
|
return '\b';
|
||||||
|
case 't':
|
||||||
|
return '\t';
|
||||||
|
case 'n':
|
||||||
|
return '\n';
|
||||||
|
case 'v':
|
||||||
|
return '\v';
|
||||||
|
case 'f':
|
||||||
|
return '\f';
|
||||||
|
case 'r':
|
||||||
|
return '\r';
|
||||||
|
default:
|
||||||
|
return esc;
|
||||||
|
}
|
||||||
|
});
|
||||||
return string;
|
return string;
|
||||||
}
|
}
|
||||||
|
|
||||||
while (lines.length > 0) {
|
while (lines.length > 0) {
|
||||||
var line = trim(lines.shift()),
|
var line = trim(lines.shift());
|
||||||
add = false;
|
var lineObsolete = false;
|
||||||
|
var add = false;
|
||||||
|
|
||||||
if (line.match(/^#\~/)) { // Obsolete item
|
if (line.match(/^#\~/)) { // Obsolete item
|
||||||
obsolete = true;
|
//only remove the obsolte comment mark, here
|
||||||
|
//might be, this is a new item, so
|
||||||
|
//only remember, this line is marked obsolete, count after line is parsed
|
||||||
line = trim(line.substring(2));
|
line = trim(line.substring(2));
|
||||||
} else {
|
lineObsolete = true;
|
||||||
obsolete = false;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (line.match(/^#:/)) { // Reference
|
if (line.match(/^#:/)) { // Reference
|
||||||
finish();
|
finish();
|
||||||
item.references.push(trim(line.replace(/^#:/, '')));
|
item.references.push(trim(line.replace(/^#:/, '')));
|
||||||
}
|
} else if (line.match(/^#,/)) { // Flags
|
||||||
else if (line.match(/^#,/)) { // Flags
|
|
||||||
finish();
|
finish();
|
||||||
var flags = trim(line.replace(/^#,/, '')).split(",");
|
var flags = trim(line.replace(/^#,/, '')).split(',');
|
||||||
for (var i = 0; i < flags.length; i++) {
|
for (var i = 0; i < flags.length; i++) {
|
||||||
item.flags[flags[i]] = true;
|
item.flags[flags[i]] = true;
|
||||||
}
|
}
|
||||||
}
|
} else if (line.match(/^#($|\s+)/)) { // Translator comment
|
||||||
else if (line.match(/^#\s+/)) { // Translator comment
|
|
||||||
finish();
|
finish();
|
||||||
item.comments.push(trim(line.replace(/^#\s+/, '')));
|
item.comments.push(trim(line.replace(/^#($|\s+)/, '')));
|
||||||
}
|
} else if (line.match(/^#\./)) { // Extracted comment
|
||||||
else if (line.match(/^#\./)) { // Extracted comment
|
|
||||||
finish();
|
finish();
|
||||||
item.extractedComments.push(trim(line.replace(/^#\./, '')));
|
item.extractedComments.push(trim(line.replace(/^#\./, '')));
|
||||||
}
|
} else if (line.match(/^msgid_plural/)) { // Plural form
|
||||||
else if (line.match(/^msgid_plural/)) { // Plural form
|
|
||||||
item.msgid_plural = extract(line);
|
item.msgid_plural = extract(line);
|
||||||
context = 'msgid_plural';
|
context = 'msgid_plural';
|
||||||
}
|
noCommentLineCount++;
|
||||||
else if (line.match(/^msgid/)) { // Original
|
} else if (line.match(/^msgid/)) { // Original
|
||||||
finish();
|
finish();
|
||||||
item.msgid = extract(line);
|
item.msgid = extract(line);
|
||||||
context = 'msgid';
|
context = 'msgid';
|
||||||
}
|
noCommentLineCount++;
|
||||||
else if (line.match(/^msgstr/)) { // Translation
|
} else if (line.match(/^msgstr/)) { // Translation
|
||||||
var m = line.match(/^msgstr\[(\d+)\]/);
|
var m = line.match(/^msgstr\[(\d+)\]/);
|
||||||
plural = m && m[1] ? parseInt(m[1]) : 0;
|
plural = m && m[1] ? parseInt(m[1]) : 0;
|
||||||
item.msgstr[plural] = extract(line);
|
item.msgstr[plural] = extract(line);
|
||||||
context = 'msgstr';
|
context = 'msgstr';
|
||||||
}
|
noCommentLineCount++;
|
||||||
else if (line.match(/^msgctxt/)) { // Context
|
} else if (line.match(/^msgctxt/)) { // Context
|
||||||
finish();
|
finish();
|
||||||
item.msgctxt = extract(line);
|
item.msgctxt = extract(line);
|
||||||
}
|
context = 'msgctxt';
|
||||||
else { // Probably multiline string or blank
|
noCommentLineCount++;
|
||||||
|
} else { // Probably multiline string or blank
|
||||||
if (line.length > 0) {
|
if (line.length > 0) {
|
||||||
|
noCommentLineCount++;
|
||||||
if (context === 'msgstr') {
|
if (context === 'msgstr') {
|
||||||
item.msgstr[plural] += extract(line);
|
item.msgstr[plural] += extract(line);
|
||||||
}
|
} else if (context === 'msgid') {
|
||||||
else if (context === 'msgid') {
|
|
||||||
item.msgid += extract(line);
|
item.msgid += extract(line);
|
||||||
}
|
} else if (context === 'msgid_plural') {
|
||||||
else if (context === 'msgid_plural') {
|
|
||||||
item.msgid_plural += extract(line);
|
item.msgid_plural += extract(line);
|
||||||
|
} else if (context === 'msgctxt') {
|
||||||
|
item.msgctxt += extract(line);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (lineObsolete) {
|
||||||
|
// Count obsolete lines for this item
|
||||||
|
obsoleteCount++;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
finish();
|
finish();
|
||||||
|
|
||||||
return po;
|
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.msgid = '';
|
||||||
this.msgctxt = null;
|
this.msgctxt = null;
|
||||||
this.references = [];
|
this.references = [];
|
||||||
@@ -188,34 +287,63 @@ PO.Item = function () {
|
|||||||
this.extractedComments = [];
|
this.extractedComments = [];
|
||||||
this.flags = {};
|
this.flags = {};
|
||||||
this.obsolete = false;
|
this.obsolete = false;
|
||||||
|
var npluralsNumber = Number(nplurals);
|
||||||
|
this.nplurals = (isNaN(npluralsNumber)) ? 2 : npluralsNumber;
|
||||||
};
|
};
|
||||||
|
|
||||||
PO.Item.prototype.toString = function () {
|
PO.Item.prototype.toString = function () {
|
||||||
var lines = [],
|
var lines = [];
|
||||||
that = this;
|
var self = this;
|
||||||
|
|
||||||
// reverse what extract(string) method during PO.parse does
|
// reverse what extract(string) method during PO.parse does
|
||||||
var _escape = function (string) {
|
var _escape = function (string) {
|
||||||
string = string.replace(/\\/g, '\\\\');
|
// don't unescape \n, since string can never contain it
|
||||||
return string.replace(/"/g, '\\"');
|
// since split('\n') is called on it
|
||||||
|
string = string.replace(/[\x07\b\t\v\f\r"\\]/g, function (match) {
|
||||||
|
switch (match) {
|
||||||
|
case '\x07':
|
||||||
|
return '\\a';
|
||||||
|
case '\b':
|
||||||
|
return '\\b';
|
||||||
|
case '\t':
|
||||||
|
return '\\t';
|
||||||
|
case '\v':
|
||||||
|
return '\\v';
|
||||||
|
case '\f':
|
||||||
|
return '\\f';
|
||||||
|
case '\r':
|
||||||
|
return '\\r';
|
||||||
|
default:
|
||||||
|
return '\\' + match;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
return string;
|
||||||
};
|
};
|
||||||
|
|
||||||
var _process = function (keyword, text, i) {
|
var _process = function (keyword, text, i) {
|
||||||
var lines = [],
|
var lines = [];
|
||||||
parts = text.split(/\n/),
|
var parts = text.split(/\n/);
|
||||||
index = typeof i !== 'undefined' ? '[' + i + ']' : '';
|
var index = typeof i !== 'undefined' ? '[' + i + ']' : '';
|
||||||
if (parts.length > 1) {
|
if (parts.length > 1) {
|
||||||
lines.push(keyword + index + ' ""');
|
lines.push(keyword + index + ' ""');
|
||||||
parts.forEach(function (part) {
|
parts.forEach(function (part) {
|
||||||
lines.push('"' + _escape(part) + '"');
|
lines.push('"' + _escape(part) + '"');
|
||||||
});
|
});
|
||||||
}
|
} else {
|
||||||
else {
|
|
||||||
lines.push(keyword + index + ' "' + _escape(text) + '"');
|
lines.push(keyword + index + ' "' + _escape(text) + '"');
|
||||||
}
|
}
|
||||||
return lines;
|
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
|
// https://www.gnu.org/software/gettext/manual/html_node/PO-Files.html
|
||||||
// says order is translator-comments, extracted-comments, references, flags
|
// says order is translator-comments, extracted-comments, references, flags
|
||||||
|
|
||||||
@@ -231,120 +359,47 @@ PO.Item.prototype.toString = function () {
|
|||||||
lines.push('#: ' + ref);
|
lines.push('#: ' + ref);
|
||||||
});
|
});
|
||||||
|
|
||||||
var flags = Object.keys(this.flags);
|
var flags = Object.keys(this.flags).filter(function (flag) {
|
||||||
|
return !!this.flags[flag];
|
||||||
|
}, this);
|
||||||
if (flags.length > 0) {
|
if (flags.length > 0) {
|
||||||
lines.push('#, ' + flags.join(","));
|
lines.push('#, ' + flags.join(','));
|
||||||
}
|
}
|
||||||
|
var mkObsolete = this.obsolete ? '#~ ' : '';
|
||||||
|
|
||||||
['msgctxt', 'msgid', 'msgid_plural', 'msgstr'].forEach(function (keyword) {
|
['msgctxt', 'msgid', 'msgid_plural', 'msgstr'].forEach(function (keyword) {
|
||||||
var text = that[keyword];
|
var text = self[keyword];
|
||||||
if (text != null) {
|
if (text != null) {
|
||||||
if (isArray(text) && text.length > 1) {
|
var hasTranslation = false;
|
||||||
text.forEach(function (t, i) {
|
if (Array.isArray(text)) {
|
||||||
lines = lines.concat(_process(keyword, t, i));
|
hasTranslation = text.some(function (text) {
|
||||||
|
return text;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
else {
|
|
||||||
text = isArray(text) ? text.join() : text;
|
if (Array.isArray(text) && text.length > 1) {
|
||||||
lines = lines.concat(_process(keyword, text));
|
text.forEach(function (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 = _processLineBreak(keyword, text, index);
|
||||||
|
lines = lines.concat(mkObsolete + processed.join('\n' + mkObsolete));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
if (this.obsolete) {
|
return lines.join('\n');
|
||||||
return "#~ " + lines.join("\n#~ ");
|
|
||||||
} else {
|
|
||||||
return lines.join("\n");
|
|
||||||
}
|
|
||||||
};
|
};
|
||||||
|
|
||||||
module.exports = PO;
|
module.exports = PO;
|
||||||
|
|
||||||
},{"fs":3,"lodash.isarray":4}],"pofile":[function(require,module,exports){
|
},{"fs":1}]},{},["pofile"]);
|
||||||
module.exports=require('W8CkM0');
|
|
||||||
},{}],3:[function(require,module,exports){
|
|
||||||
|
|
||||||
},{}],4:[function(require,module,exports){
|
|
||||||
/**
|
|
||||||
* Lo-Dash 2.4.1 (Custom Build) <http://lodash.com/>
|
|
||||||
* Build: `lodash modularize modern exports="npm" -o ./npm/`
|
|
||||||
* Copyright 2012-2013 The Dojo Foundation <http://dojofoundation.org/>
|
|
||||||
* Based on Underscore.js 1.5.2 <http://underscorejs.org/LICENSE>
|
|
||||||
* Copyright 2009-2013 Jeremy Ashkenas, DocumentCloud and Investigative Reporters & Editors
|
|
||||||
* Available under MIT license <http://lodash.com/license>
|
|
||||||
*/
|
|
||||||
var isNative = require('lodash._isnative');
|
|
||||||
|
|
||||||
/** `Object#toString` result shortcuts */
|
|
||||||
var arrayClass = '[object Array]';
|
|
||||||
|
|
||||||
/** Used for native method references */
|
|
||||||
var objectProto = Object.prototype;
|
|
||||||
|
|
||||||
/** Used to resolve the internal [[Class]] of values */
|
|
||||||
var toString = objectProto.toString;
|
|
||||||
|
|
||||||
/* Native method shortcuts for methods with the same name as other `lodash` methods */
|
|
||||||
var nativeIsArray = isNative(nativeIsArray = Array.isArray) && nativeIsArray;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Checks if `value` is an array.
|
|
||||||
*
|
|
||||||
* @static
|
|
||||||
* @memberOf _
|
|
||||||
* @type Function
|
|
||||||
* @category Objects
|
|
||||||
* @param {*} value The value to check.
|
|
||||||
* @returns {boolean} Returns `true` if the `value` is an array, else `false`.
|
|
||||||
* @example
|
|
||||||
*
|
|
||||||
* (function() { return _.isArray(arguments); })();
|
|
||||||
* // => false
|
|
||||||
*
|
|
||||||
* _.isArray([1, 2, 3]);
|
|
||||||
* // => true
|
|
||||||
*/
|
|
||||||
var isArray = nativeIsArray || function(value) {
|
|
||||||
return value && typeof value == 'object' && typeof value.length == 'number' &&
|
|
||||||
toString.call(value) == arrayClass || false;
|
|
||||||
};
|
|
||||||
|
|
||||||
module.exports = isArray;
|
|
||||||
|
|
||||||
},{"lodash._isnative":5}],5:[function(require,module,exports){
|
|
||||||
/**
|
|
||||||
* Lo-Dash 2.4.1 (Custom Build) <http://lodash.com/>
|
|
||||||
* Build: `lodash modularize modern exports="npm" -o ./npm/`
|
|
||||||
* Copyright 2012-2013 The Dojo Foundation <http://dojofoundation.org/>
|
|
||||||
* Based on Underscore.js 1.5.2 <http://underscorejs.org/LICENSE>
|
|
||||||
* Copyright 2009-2013 Jeremy Ashkenas, DocumentCloud and Investigative Reporters & Editors
|
|
||||||
* Available under MIT license <http://lodash.com/license>
|
|
||||||
*/
|
|
||||||
|
|
||||||
/** Used for native method references */
|
|
||||||
var objectProto = Object.prototype;
|
|
||||||
|
|
||||||
/** Used to resolve the internal [[Class]] of values */
|
|
||||||
var toString = objectProto.toString;
|
|
||||||
|
|
||||||
/** Used to detect if a method is native */
|
|
||||||
var reNative = RegExp('^' +
|
|
||||||
String(toString)
|
|
||||||
.replace(/[.*+?^${}()|[\]\\]/g, '\\$&')
|
|
||||||
.replace(/toString| for [^\]]+/g, '.*?') + '$'
|
|
||||||
);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Checks if `value` is a native function.
|
|
||||||
*
|
|
||||||
* @private
|
|
||||||
* @param {*} value The value to check.
|
|
||||||
* @returns {boolean} Returns `true` if the `value` is a native function, else `false`.
|
|
||||||
*/
|
|
||||||
function isNative(value) {
|
|
||||||
return typeof value == 'function' && reNative.test(value);
|
|
||||||
}
|
|
||||||
|
|
||||||
module.exports = isNative;
|
|
||||||
|
|
||||||
},{}]},{},["W8CkM0"])
|
|
||||||
|
|||||||
2
dist/pofile.min.js
vendored
2
dist/pofile.min.js
vendored
File diff suppressed because one or more lines are too long
302
lib/po.js
302
lib/po.js
@@ -1,36 +1,55 @@
|
|||||||
var fs = require('fs'),
|
|
||||||
isArray = require('lodash.isarray');
|
|
||||||
|
|
||||||
function trim(string) {
|
function trim(string) {
|
||||||
return string.replace(/^\s+|\s+$/g, '');
|
return string.replace(/^\s+|\s+$/g, '');
|
||||||
}
|
}
|
||||||
|
|
||||||
var PO = function () {
|
var PO = function () {
|
||||||
this.comments = [];
|
this.comments = [];
|
||||||
|
this.extractedComments = [];
|
||||||
this.headers = {};
|
this.headers = {};
|
||||||
|
this.headerOrder = [];
|
||||||
this.items = [];
|
this.items = [];
|
||||||
};
|
};
|
||||||
|
|
||||||
PO.prototype.save = function (filename, callback) {
|
PO.prototype.save = function (filename, callback) {
|
||||||
fs.writeFile(filename, this.toString(), callback);
|
require('fs').writeFile(filename, this.toString(), callback);
|
||||||
};
|
};
|
||||||
|
|
||||||
PO.prototype.toString = function () {
|
PO.prototype.toString = function () {
|
||||||
var lines = [],
|
var lines = [];
|
||||||
that = this;
|
|
||||||
|
|
||||||
if (this.comments) {
|
if (this.comments) {
|
||||||
this.comments.forEach(function (comment) {
|
this.comments.forEach(function (comment) {
|
||||||
lines.push('# ' + comment);
|
lines.push(('# ' + comment).trim());
|
||||||
|
});
|
||||||
|
}
|
||||||
|
if (this.extractedComments) {
|
||||||
|
this.extractedComments.forEach(function (comment) {
|
||||||
|
lines.push(('#. ' + comment).trim());
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
lines.push('msgid ""');
|
lines.push('msgid ""');
|
||||||
lines.push('msgstr ""');
|
lines.push('msgstr ""');
|
||||||
|
|
||||||
|
var self = this;
|
||||||
|
var headerOrder = [];
|
||||||
|
|
||||||
|
this.headerOrder.forEach(function (key) {
|
||||||
|
if (key in self.headers) {
|
||||||
|
headerOrder.push(key);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
var keys = Object.keys(this.headers);
|
var keys = Object.keys(this.headers);
|
||||||
|
|
||||||
keys.forEach(function (key) {
|
keys.forEach(function (key) {
|
||||||
lines.push('"' + key + ': ' + that.headers[key] + '\\n"');
|
if (headerOrder.indexOf(key) === -1) {
|
||||||
|
headerOrder.push(key);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
headerOrder.forEach(function (key) {
|
||||||
|
lines.push('"' + key + ': ' + self.headers[key] + '\\n"');
|
||||||
});
|
});
|
||||||
|
|
||||||
lines.push('');
|
lines.push('');
|
||||||
@@ -40,11 +59,11 @@ PO.prototype.toString = function () {
|
|||||||
lines.push('');
|
lines.push('');
|
||||||
});
|
});
|
||||||
|
|
||||||
return lines.join("\n");
|
return lines.join('\n');
|
||||||
};
|
};
|
||||||
|
|
||||||
PO.load = function (filename, callback) {
|
PO.load = function (filename, callback) {
|
||||||
fs.readFile(filename, 'utf-8', function (err, data) {
|
require('fs').readFile(filename, 'utf-8', function (err, data) {
|
||||||
if (err) {
|
if (err) {
|
||||||
return callback(err);
|
return callback(err);
|
||||||
}
|
}
|
||||||
@@ -56,10 +75,20 @@ PO.load = function (filename, callback) {
|
|||||||
PO.parse = function (data) {
|
PO.parse = function (data) {
|
||||||
//support both unix and windows newline formats.
|
//support both unix and windows newline formats.
|
||||||
data = data.replace(/\r\n/g, '\n');
|
data = data.replace(/\r\n/g, '\n');
|
||||||
var po = new PO(),
|
var po = new PO();
|
||||||
sections = data.split(/\n\n/),
|
var sections = data.split(/\n\n/);
|
||||||
headers = sections.shift(),
|
var headers = [];
|
||||||
lines = sections.join("\n").split(/\n/);
|
//everything until the first 'msgid ""' is considered header
|
||||||
|
while (sections[0] && (headers.length === 0 || headers[headers.length - 1].indexOf('msgid ""') < 0)) {
|
||||||
|
if (sections[0].match(/msgid "[^"]/)) {
|
||||||
|
//found first real string, adding a dummy header item
|
||||||
|
headers.push('msgid ""');
|
||||||
|
} else {
|
||||||
|
headers.push(sections.shift());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
headers = headers.join('\n');
|
||||||
|
var lines = sections.join('\n').split(/\n/);
|
||||||
|
|
||||||
po.headers = {
|
po.headers = {
|
||||||
'Project-Id-Version': '',
|
'Project-Id-Version': '',
|
||||||
@@ -73,111 +102,179 @@ PO.parse = function (data) {
|
|||||||
'Content-Transfer-Encoding': '',
|
'Content-Transfer-Encoding': '',
|
||||||
'Plural-Forms': '',
|
'Plural-Forms': '',
|
||||||
};
|
};
|
||||||
|
po.headerOrder = [];
|
||||||
|
|
||||||
headers.split(/\n/).forEach(function (header) {
|
headers.split(/\n/).reduce(function (acc, line) {
|
||||||
if (header.match(/^#/)) {
|
if (acc.merge) {
|
||||||
po.comments.push(header.replace(/^#\s*/, ''));
|
//join lines, remove last resp. first "
|
||||||
|
line = acc.pop().slice(0, -1) + line.slice(1);
|
||||||
|
delete acc.merge;
|
||||||
}
|
}
|
||||||
if (header.match(/^"/)) {
|
if (/^".*"$/.test(line) && !/^".*\\n"$/.test(line)) {
|
||||||
|
acc.merge = true;
|
||||||
|
}
|
||||||
|
acc.push(line);
|
||||||
|
return acc;
|
||||||
|
}, []).forEach(function (header) {
|
||||||
|
if (header.match(/^#\./)) {
|
||||||
|
po.extractedComments.push(header.replace(/^#\.\s*/, ''));
|
||||||
|
} else if (header.match(/^#/)) {
|
||||||
|
po.comments.push(header.replace(/^#\s*/, ''));
|
||||||
|
} else if (header.match(/^"/)) {
|
||||||
header = header.trim().replace(/^"/, '').replace(/\\n"$/, '');
|
header = header.trim().replace(/^"/, '').replace(/\\n"$/, '');
|
||||||
var p = header.split(/:/),
|
var p = header.split(/:/);
|
||||||
name = p.shift().trim(),
|
var name = p.shift().trim();
|
||||||
value = p.join(':').trim();
|
var value = p.join(':').trim();
|
||||||
po.headers[name] = value;
|
po.headers[name] = value;
|
||||||
|
po.headerOrder.push(name);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
var item = new PO.Item(),
|
var parsedPluralForms = PO.parsePluralForms(po.headers['Plural-Forms']);
|
||||||
context = null,
|
var nplurals = parsedPluralForms.nplurals;
|
||||||
plural = 0,
|
var item = new PO.Item({ nplurals: nplurals });
|
||||||
obsolete = false;
|
var context = null;
|
||||||
|
var plural = 0;
|
||||||
|
var obsoleteCount = 0;
|
||||||
|
var noCommentLineCount = 0;
|
||||||
|
|
||||||
function finish() {
|
function finish() {
|
||||||
if (item.msgid.length > 0) {
|
if (item.msgid.length > 0) {
|
||||||
|
if (obsoleteCount >= noCommentLineCount) {
|
||||||
|
item.obsolete = true;
|
||||||
|
}
|
||||||
|
obsoleteCount = 0;
|
||||||
|
noCommentLineCount = 0;
|
||||||
po.items.push(item);
|
po.items.push(item);
|
||||||
item = new PO.Item();
|
item = new PO.Item({ nplurals: nplurals });
|
||||||
item.obsolete = obsolete;
|
|
||||||
obsolete = false;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function extract(string) {
|
function extract(string) {
|
||||||
string = trim(string);
|
string = trim(string);
|
||||||
string = string.replace(/^[^"]*"|"$/g, '');
|
string = string.replace(/^[^"]*"|"$/g, '');
|
||||||
string = string.replace(/\\"/g, '"');
|
string = string.replace(/\\([abtnvfr'"\\?]|([0-7]{3})|x([0-9a-fA-F]{2}))/g, function (match, esc, oct, hex) {
|
||||||
string = string.replace(/\\\\/g, '\\');
|
if (oct) {
|
||||||
|
return String.fromCharCode(parseInt(oct, 8));
|
||||||
|
}
|
||||||
|
if (hex) {
|
||||||
|
return String.fromCharCode(parseInt(hex, 16));
|
||||||
|
}
|
||||||
|
switch (esc) {
|
||||||
|
case 'a':
|
||||||
|
return '\x07';
|
||||||
|
case 'b':
|
||||||
|
return '\b';
|
||||||
|
case 't':
|
||||||
|
return '\t';
|
||||||
|
case 'n':
|
||||||
|
return '\n';
|
||||||
|
case 'v':
|
||||||
|
return '\v';
|
||||||
|
case 'f':
|
||||||
|
return '\f';
|
||||||
|
case 'r':
|
||||||
|
return '\r';
|
||||||
|
default:
|
||||||
|
return esc;
|
||||||
|
}
|
||||||
|
});
|
||||||
return string;
|
return string;
|
||||||
}
|
}
|
||||||
|
|
||||||
while (lines.length > 0) {
|
while (lines.length > 0) {
|
||||||
var line = trim(lines.shift()),
|
var line = trim(lines.shift());
|
||||||
add = false;
|
var lineObsolete = false;
|
||||||
|
var add = false;
|
||||||
|
|
||||||
if (line.match(/^#\~/)) { // Obsolete item
|
if (line.match(/^#\~/)) { // Obsolete item
|
||||||
obsolete = true;
|
//only remove the obsolte comment mark, here
|
||||||
|
//might be, this is a new item, so
|
||||||
|
//only remember, this line is marked obsolete, count after line is parsed
|
||||||
line = trim(line.substring(2));
|
line = trim(line.substring(2));
|
||||||
} else {
|
lineObsolete = true;
|
||||||
obsolete = false;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (line.match(/^#:/)) { // Reference
|
if (line.match(/^#:/)) { // Reference
|
||||||
finish();
|
finish();
|
||||||
item.references.push(trim(line.replace(/^#:/, '')));
|
item.references.push(trim(line.replace(/^#:/, '')));
|
||||||
}
|
} else if (line.match(/^#,/)) { // Flags
|
||||||
else if (line.match(/^#,/)) { // Flags
|
|
||||||
finish();
|
finish();
|
||||||
var flags = trim(line.replace(/^#,/, '')).split(",");
|
var flags = trim(line.replace(/^#,/, '')).split(',');
|
||||||
for (var i = 0; i < flags.length; i++) {
|
for (var i = 0; i < flags.length; i++) {
|
||||||
item.flags[flags[i]] = true;
|
item.flags[flags[i]] = true;
|
||||||
}
|
}
|
||||||
}
|
} else if (line.match(/^#($|\s+)/)) { // Translator comment
|
||||||
else if (line.match(/^#\s+/)) { // Translator comment
|
|
||||||
finish();
|
finish();
|
||||||
item.comments.push(trim(line.replace(/^#\s+/, '')));
|
item.comments.push(trim(line.replace(/^#($|\s+)/, '')));
|
||||||
}
|
} else if (line.match(/^#\./)) { // Extracted comment
|
||||||
else if (line.match(/^#\./)) { // Extracted comment
|
|
||||||
finish();
|
finish();
|
||||||
item.extractedComments.push(trim(line.replace(/^#\./, '')));
|
item.extractedComments.push(trim(line.replace(/^#\./, '')));
|
||||||
}
|
} else if (line.match(/^msgid_plural/)) { // Plural form
|
||||||
else if (line.match(/^msgid_plural/)) { // Plural form
|
|
||||||
item.msgid_plural = extract(line);
|
item.msgid_plural = extract(line);
|
||||||
context = 'msgid_plural';
|
context = 'msgid_plural';
|
||||||
}
|
noCommentLineCount++;
|
||||||
else if (line.match(/^msgid/)) { // Original
|
} else if (line.match(/^msgid/)) { // Original
|
||||||
finish();
|
finish();
|
||||||
item.msgid = extract(line);
|
item.msgid = extract(line);
|
||||||
context = 'msgid';
|
context = 'msgid';
|
||||||
}
|
noCommentLineCount++;
|
||||||
else if (line.match(/^msgstr/)) { // Translation
|
} else if (line.match(/^msgstr/)) { // Translation
|
||||||
var m = line.match(/^msgstr\[(\d+)\]/);
|
var m = line.match(/^msgstr\[(\d+)\]/);
|
||||||
plural = m && m[1] ? parseInt(m[1]) : 0;
|
plural = m && m[1] ? parseInt(m[1]) : 0;
|
||||||
item.msgstr[plural] = extract(line);
|
item.msgstr[plural] = extract(line);
|
||||||
context = 'msgstr';
|
context = 'msgstr';
|
||||||
}
|
noCommentLineCount++;
|
||||||
else if (line.match(/^msgctxt/)) { // Context
|
} else if (line.match(/^msgctxt/)) { // Context
|
||||||
finish();
|
finish();
|
||||||
item.msgctxt = extract(line);
|
item.msgctxt = extract(line);
|
||||||
}
|
context = 'msgctxt';
|
||||||
else { // Probably multiline string or blank
|
noCommentLineCount++;
|
||||||
|
} else { // Probably multiline string or blank
|
||||||
if (line.length > 0) {
|
if (line.length > 0) {
|
||||||
|
noCommentLineCount++;
|
||||||
if (context === 'msgstr') {
|
if (context === 'msgstr') {
|
||||||
item.msgstr[plural] += extract(line);
|
item.msgstr[plural] += extract(line);
|
||||||
}
|
} else if (context === 'msgid') {
|
||||||
else if (context === 'msgid') {
|
|
||||||
item.msgid += extract(line);
|
item.msgid += extract(line);
|
||||||
}
|
} else if (context === 'msgid_plural') {
|
||||||
else if (context === 'msgid_plural') {
|
|
||||||
item.msgid_plural += extract(line);
|
item.msgid_plural += extract(line);
|
||||||
|
} else if (context === 'msgctxt') {
|
||||||
|
item.msgctxt += extract(line);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (lineObsolete) {
|
||||||
|
// Count obsolete lines for this item
|
||||||
|
obsoleteCount++;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
finish();
|
finish();
|
||||||
|
|
||||||
return po;
|
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.msgid = '';
|
||||||
this.msgctxt = null;
|
this.msgctxt = null;
|
||||||
this.references = [];
|
this.references = [];
|
||||||
@@ -187,34 +284,63 @@ PO.Item = function () {
|
|||||||
this.extractedComments = [];
|
this.extractedComments = [];
|
||||||
this.flags = {};
|
this.flags = {};
|
||||||
this.obsolete = false;
|
this.obsolete = false;
|
||||||
|
var npluralsNumber = Number(nplurals);
|
||||||
|
this.nplurals = (isNaN(npluralsNumber)) ? 2 : npluralsNumber;
|
||||||
};
|
};
|
||||||
|
|
||||||
PO.Item.prototype.toString = function () {
|
PO.Item.prototype.toString = function () {
|
||||||
var lines = [],
|
var lines = [];
|
||||||
that = this;
|
var self = this;
|
||||||
|
|
||||||
// reverse what extract(string) method during PO.parse does
|
// reverse what extract(string) method during PO.parse does
|
||||||
var _escape = function (string) {
|
var _escape = function (string) {
|
||||||
string = string.replace(/\\/g, '\\\\');
|
// don't unescape \n, since string can never contain it
|
||||||
return string.replace(/"/g, '\\"');
|
// since split('\n') is called on it
|
||||||
|
string = string.replace(/[\x07\b\t\v\f\r"\\]/g, function (match) {
|
||||||
|
switch (match) {
|
||||||
|
case '\x07':
|
||||||
|
return '\\a';
|
||||||
|
case '\b':
|
||||||
|
return '\\b';
|
||||||
|
case '\t':
|
||||||
|
return '\\t';
|
||||||
|
case '\v':
|
||||||
|
return '\\v';
|
||||||
|
case '\f':
|
||||||
|
return '\\f';
|
||||||
|
case '\r':
|
||||||
|
return '\\r';
|
||||||
|
default:
|
||||||
|
return '\\' + match;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
return string;
|
||||||
};
|
};
|
||||||
|
|
||||||
var _process = function (keyword, text, i) {
|
var _process = function (keyword, text, i) {
|
||||||
var lines = [],
|
var lines = [];
|
||||||
parts = text.split(/\n/),
|
var parts = text.split(/\n/);
|
||||||
index = typeof i !== 'undefined' ? '[' + i + ']' : '';
|
var index = typeof i !== 'undefined' ? '[' + i + ']' : '';
|
||||||
if (parts.length > 1) {
|
if (parts.length > 1) {
|
||||||
lines.push(keyword + index + ' ""');
|
lines.push(keyword + index + ' ""');
|
||||||
parts.forEach(function (part) {
|
parts.forEach(function (part) {
|
||||||
lines.push('"' + _escape(part) + '"');
|
lines.push('"' + _escape(part) + '"');
|
||||||
});
|
});
|
||||||
}
|
} else {
|
||||||
else {
|
|
||||||
lines.push(keyword + index + ' "' + _escape(text) + '"');
|
lines.push(keyword + index + ' "' + _escape(text) + '"');
|
||||||
}
|
}
|
||||||
return lines;
|
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
|
// https://www.gnu.org/software/gettext/manual/html_node/PO-Files.html
|
||||||
// says order is translator-comments, extracted-comments, references, flags
|
// says order is translator-comments, extracted-comments, references, flags
|
||||||
|
|
||||||
@@ -230,31 +356,45 @@ PO.Item.prototype.toString = function () {
|
|||||||
lines.push('#: ' + ref);
|
lines.push('#: ' + ref);
|
||||||
});
|
});
|
||||||
|
|
||||||
var flags = Object.keys(this.flags);
|
var flags = Object.keys(this.flags).filter(function (flag) {
|
||||||
|
return !!this.flags[flag];
|
||||||
|
}, this);
|
||||||
if (flags.length > 0) {
|
if (flags.length > 0) {
|
||||||
lines.push('#, ' + flags.join(","));
|
lines.push('#, ' + flags.join(','));
|
||||||
}
|
}
|
||||||
|
var mkObsolete = this.obsolete ? '#~ ' : '';
|
||||||
|
|
||||||
['msgctxt', 'msgid', 'msgid_plural', 'msgstr'].forEach(function (keyword) {
|
['msgctxt', 'msgid', 'msgid_plural', 'msgstr'].forEach(function (keyword) {
|
||||||
var text = that[keyword];
|
var text = self[keyword];
|
||||||
if (text != null) {
|
if (text != null) {
|
||||||
if (isArray(text) && text.length > 1) {
|
var hasTranslation = false;
|
||||||
text.forEach(function (t, i) {
|
if (Array.isArray(text)) {
|
||||||
lines = lines.concat(_process(keyword, t, i));
|
hasTranslation = text.some(function (text) {
|
||||||
|
return text;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
else {
|
|
||||||
text = isArray(text) ? text.join() : text;
|
if (Array.isArray(text) && text.length > 1) {
|
||||||
lines = lines.concat(_process(keyword, text));
|
text.forEach(function (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 = _processLineBreak(keyword, text, index);
|
||||||
|
lines = lines.concat(mkObsolete + processed.join('\n' + mkObsolete));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
if (this.obsolete) {
|
return lines.join('\n');
|
||||||
return "#~ " + lines.join("\n#~ ");
|
|
||||||
} else {
|
|
||||||
return lines.join("\n");
|
|
||||||
}
|
|
||||||
};
|
};
|
||||||
|
|
||||||
module.exports = PO;
|
module.exports = PO;
|
||||||
|
|||||||
26
package.json
26
package.json
@@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"name": "pofile",
|
"name": "pofile",
|
||||||
"description": "Parse and serialize Gettext PO files.",
|
"description": "Parse and serialize Gettext PO files.",
|
||||||
"version": "0.2.7",
|
"version": "1.1.1",
|
||||||
"author": {
|
"author": {
|
||||||
"name": "Ruben Vermeersch",
|
"name": "Ruben Vermeersch",
|
||||||
"email": "ruben@savanne.be",
|
"email": "ruben@savanne.be",
|
||||||
@@ -20,6 +20,7 @@
|
|||||||
"url": "http://github.com/rubenv/pofile.git"
|
"url": "http://github.com/rubenv/pofile.git"
|
||||||
},
|
},
|
||||||
"main": "./lib/po",
|
"main": "./lib/po",
|
||||||
|
"types": "./pofile.d.ts",
|
||||||
"keywords": [
|
"keywords": [
|
||||||
"i18n",
|
"i18n",
|
||||||
"l10n",
|
"l10n",
|
||||||
@@ -35,17 +36,16 @@
|
|||||||
"test": "test"
|
"test": "test"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"grunt": "~0.4.2",
|
"browserify": "~14.0.0",
|
||||||
"grunt-contrib-watch": "~0.5.3",
|
"grunt": "~1.0.1",
|
||||||
"grunt-contrib-jshint": "~0.7.2",
|
"grunt-browserify": "~5.0.0",
|
||||||
"grunt-mocha-cli": "~1.4.0",
|
"grunt-bump": "0.8.0",
|
||||||
"grunt-contrib-uglify": "~0.2.7",
|
"grunt-contrib-clean": "~1.0.0",
|
||||||
"browserify": "~3.11.1",
|
"grunt-contrib-jshint": "~1.1.0",
|
||||||
"grunt-browserify": "~1.3.0",
|
"grunt-contrib-uglify": "~2.1.0",
|
||||||
"grunt-contrib-clean": "~0.5.0",
|
"grunt-contrib-watch": "~1.0.0",
|
||||||
"grunt-bump": "0.0.13"
|
"grunt-jscs": "~3.0.1",
|
||||||
|
"grunt-mocha-cli": "~3.0.0"
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {}
|
||||||
"lodash.isarray": "~2.4.1"
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
45
pofile.d.ts
vendored
Normal file
45
pofile.d.ts
vendored
Normal file
@@ -0,0 +1,45 @@
|
|||||||
|
declare 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;
|
||||||
|
[name: string]: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
declare 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: Record<string, boolean | undefined>;
|
||||||
|
private nplurals: number;
|
||||||
|
private obsolete: boolean;
|
||||||
|
|
||||||
|
public toString(): string;
|
||||||
|
}
|
||||||
|
|
||||||
|
declare class PO {
|
||||||
|
public comments: string[];
|
||||||
|
public extractedComments: string[];
|
||||||
|
public items: Item[];
|
||||||
|
public headers: Partial<IHeaders>
|
||||||
|
|
||||||
|
public static parse(data: string): PO;
|
||||||
|
public static parsePluralForms(forms: string): PO;
|
||||||
|
public static load(fileName: string, callback: (err: NodeJS.ErrnoException, po: PO) => void): void;
|
||||||
|
public static Item: typeof Item;
|
||||||
|
|
||||||
|
public save(fileName: string, callback: (err: NodeJS.ErrnoException) => void): void;
|
||||||
|
public toString(): string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export = PO
|
||||||
7
test/fixtures/big.po
vendored
7
test/fixtures/big.po
vendored
@@ -1,4 +1,5 @@
|
|||||||
# French translation of Link (6.x-2.9)
|
# French translation of Link (6.x-2.9)
|
||||||
|
|
||||||
# Copyright (c) 2011 by the French translation team
|
# Copyright (c) 2011 by the French translation team
|
||||||
#
|
#
|
||||||
msgid ""
|
msgid ""
|
||||||
@@ -295,3 +296,9 @@ msgstr "This folder is empty."
|
|||||||
msgctxt "folder action"
|
msgctxt "folder action"
|
||||||
msgid "Empty folder"
|
msgid "Empty folder"
|
||||||
msgstr "Make this folder empty."
|
msgstr "Make this folder empty."
|
||||||
|
|
||||||
|
msgctxt ""
|
||||||
|
"folder "
|
||||||
|
"meta"
|
||||||
|
msgid "Created Date"
|
||||||
|
msgstr "Date de création"
|
||||||
|
|||||||
54
test/fixtures/c-strings.po
vendored
54
test/fixtures/c-strings.po
vendored
@@ -17,3 +17,57 @@ msgstr ""
|
|||||||
|
|
||||||
msgid "The name field must not contain characters like \" or \\"
|
msgid "The name field must not contain characters like \" or \\"
|
||||||
msgstr ""
|
msgstr ""
|
||||||
|
|
||||||
|
# possibility to reorder items depending on locale
|
||||||
|
#. Format of addresses
|
||||||
|
#. %1$s is the street
|
||||||
|
#. %2$s is the postal code
|
||||||
|
#. %3$s is the city
|
||||||
|
#. %4$s is the state
|
||||||
|
#. %5$s is the country
|
||||||
|
msgid ""
|
||||||
|
"%1$s\n"
|
||||||
|
"%2$s %3$s\n"
|
||||||
|
"%4$s\n"
|
||||||
|
"%5$s"
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
# "i18"ned code
|
||||||
|
#. used in <pre> environment, so don't remove any control sequences
|
||||||
|
msgid ""
|
||||||
|
"define('some/test/module', function () {\n"
|
||||||
|
"\t'use strict';\n"
|
||||||
|
"\treturn {};\n"
|
||||||
|
"});\n"
|
||||||
|
""
|
||||||
|
msgstr ""
|
||||||
|
"define('random/test/file', function () {\n"
|
||||||
|
"\t'use strict';\n"
|
||||||
|
"\treturn {};\n"
|
||||||
|
"});\n"
|
||||||
|
""
|
||||||
|
|
||||||
|
# all one-letter escape characters
|
||||||
|
# be aware, that \a, \b, \v, \f and \r should not be used
|
||||||
|
# in i18ned messages (according to gettext tools)
|
||||||
|
# however, they should be properly parsed, anyway
|
||||||
|
msgid ""
|
||||||
|
"\a\b\t\n"
|
||||||
|
"\v\f\r"
|
||||||
|
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."
|
||||||
|
|||||||
9
test/fixtures/comment.po
vendored
9
test/fixtures/comment.po
vendored
@@ -1,6 +1,7 @@
|
|||||||
# French translation of Link (6.x-2.9)
|
# French translation of Link (6.x-2.9)
|
||||||
# Copyright (c) 2011 by the French translation team
|
# Copyright (c) 2011 by the French translation team
|
||||||
#
|
#
|
||||||
|
#. extracted from test
|
||||||
msgid ""
|
msgid ""
|
||||||
msgstr ""
|
msgstr ""
|
||||||
"Project-Id-Version: Link (6.x-2.9)\n"
|
"Project-Id-Version: Link (6.x-2.9)\n"
|
||||||
@@ -19,3 +20,11 @@ msgstr ""
|
|||||||
#. Extracted comment
|
#. Extracted comment
|
||||||
msgid "Title, as plain text"
|
msgid "Title, as plain text"
|
||||||
msgstr "Attribut title, en tant que texte brut"
|
msgstr "Attribut title, en tant que texte brut"
|
||||||
|
|
||||||
|
#
|
||||||
|
#.
|
||||||
|
#:
|
||||||
|
#, fuzzy
|
||||||
|
msgid "Empty comment"
|
||||||
|
msgstr "Empty"
|
||||||
|
|
||||||
|
|||||||
10
test/fixtures/commented.po
vendored
10
test/fixtures/commented.po
vendored
@@ -21,3 +21,13 @@ msgstr "{{dataLoader.data.length}} resultaten"
|
|||||||
|
|
||||||
#~ msgid "Add order"
|
#~ msgid "Add order"
|
||||||
#~ msgstr "Order toevoegen"
|
#~ msgstr "Order toevoegen"
|
||||||
|
|
||||||
|
#~ # commented obsolete item
|
||||||
|
#~ #, fuzzy
|
||||||
|
#~ msgid "Commented item"
|
||||||
|
#~ msgstr "not sure"
|
||||||
|
|
||||||
|
# commented obsolete item
|
||||||
|
#, fuzzy
|
||||||
|
#~ msgid "Second commented item"
|
||||||
|
#~ msgstr "also not sure"
|
||||||
|
|||||||
5
test/fixtures/multi-line.po
vendored
5
test/fixtures/multi-line.po
vendored
@@ -1,6 +1,8 @@
|
|||||||
# French translation of Link (6.x-2.9)
|
# French translation of Link (6.x-2.9)
|
||||||
# Copyright (c) 2011 by the French translation team
|
# Copyright (c) 2011 by the French translation team
|
||||||
#
|
#
|
||||||
|
## Plural-Forms by polish translation team to demonstrate multi-line ##
|
||||||
|
#
|
||||||
msgid ""
|
msgid ""
|
||||||
msgstr ""
|
msgstr ""
|
||||||
"Project-Id-Version: Link (6.x-2.9)\n"
|
"Project-Id-Version: Link (6.x-2.9)\n"
|
||||||
@@ -10,7 +12,8 @@ msgstr ""
|
|||||||
"MIME-Version: 1.0\n"
|
"MIME-Version: 1.0\n"
|
||||||
"Content-Type: text/plain; charset=UTF-8\n"
|
"Content-Type: text/plain; charset=UTF-8\n"
|
||||||
"Content-Transfer-Encoding: 8bit\n"
|
"Content-Transfer-Encoding: 8bit\n"
|
||||||
"Plural-Forms: nplurals=2; plural=(n > 1);\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"
|
"Last-Translator: Ruben Vermeersch <ruben@rocketeer.be>\n"
|
||||||
"Language: fr\n"
|
"Language: fr\n"
|
||||||
"X-Generator: Poedit 1.6.2\n"
|
"X-Generator: Poedit 1.6.2\n"
|
||||||
|
|||||||
8
test/fixtures/no_header.po
vendored
Normal file
8
test/fixtures/no_header.po
vendored
Normal file
@@ -0,0 +1,8 @@
|
|||||||
|
# some comment
|
||||||
|
|
||||||
|
msgid "First id, no header"
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
|
msgid "A second string"
|
||||||
|
msgstr ""
|
||||||
|
|
||||||
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 ""
|
||||||
38
test/fixtures/plurals/nplurals-2.po
vendored
Normal file
38
test/fixtures/plurals/nplurals-2.po
vendored
Normal file
@@ -0,0 +1,38 @@
|
|||||||
|
# French translation of Link (6.x-2.9)
|
||||||
|
# Copyright (c) 2011 by the French 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: French\n"
|
||||||
|
"MIME-Version: 1.0\n"
|
||||||
|
"Content-Type: text/plain; charset=UTF-8\n"
|
||||||
|
"Content-Transfer-Encoding: 8bit\n"
|
||||||
|
"Plural-Forms: nplurals=2; plural=(n > 1);\n"
|
||||||
|
"Last-Translator: Ruben Vermeersch <ruben@rocketeer.be>\n"
|
||||||
|
"Language: fr\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 ""
|
||||||
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 ""
|
||||||
4
test/fixtures/reference.po
vendored
4
test/fixtures/reference.po
vendored
@@ -24,3 +24,7 @@ msgstr "Attribut title, en tant que texte brut"
|
|||||||
#: b
|
#: b
|
||||||
msgid "X"
|
msgid "X"
|
||||||
msgstr "Y"
|
msgstr "Y"
|
||||||
|
|
||||||
|
#: standard input:12 standard input:17
|
||||||
|
msgid "Z"
|
||||||
|
msgstr "ZZ"
|
||||||
|
|||||||
@@ -28,3 +28,46 @@ describe('Headers', function () {
|
|||||||
assert.equal(Object.keys(po.headers).length, 12);
|
assert.equal(Object.keys(po.headers).length, 12);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
describe('PO files with no headers', function () {
|
||||||
|
|
||||||
|
it('Parses an empty string', function () {
|
||||||
|
var po = PO.parse('');
|
||||||
|
assert.notEqual(po, null);
|
||||||
|
// all headers should be empty
|
||||||
|
for (var key in po.headers) {
|
||||||
|
assert.equal(po.headers[key], '');
|
||||||
|
}
|
||||||
|
assert.equal(po.items.length, 0);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Parses a minimal example', function () {
|
||||||
|
var po = PO.parse('msgid "minimal PO"\nmsgstr ""');
|
||||||
|
assert.notEqual(po, null);
|
||||||
|
// all headers should be empty
|
||||||
|
for (var key in po.headers) {
|
||||||
|
assert.equal(po.headers[key], '');
|
||||||
|
}
|
||||||
|
assert.equal(po.items.length, 1);
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('advanced example', function () {
|
||||||
|
var po;
|
||||||
|
|
||||||
|
before(function (done) {
|
||||||
|
PO.load(__dirname + '/fixtures/no_header.po', function (err, result) {
|
||||||
|
assert.equal(err, null);
|
||||||
|
po = result;
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Parses the po file', function () {
|
||||||
|
assert.notEqual(po, null);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Finds all items', function () {
|
||||||
|
assert.equal(po.items.length, 2);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|||||||
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);
|
||||||
|
});
|
||||||
|
});
|
||||||
147
test/parse.js
147
test/parse.js
@@ -6,11 +6,11 @@ describe('Parse', function () {
|
|||||||
it('Parses the big po file', function () {
|
it('Parses the big po file', function () {
|
||||||
var po = PO.parse(fs.readFileSync(__dirname + '/fixtures/big.po', 'utf8'));
|
var po = PO.parse(fs.readFileSync(__dirname + '/fixtures/big.po', 'utf8'));
|
||||||
assert.notEqual(po, null);
|
assert.notEqual(po, null);
|
||||||
assert.equal(po.items.length, 69);
|
assert.equal(po.items.length, 70);
|
||||||
|
|
||||||
var item = po.items[0];
|
var item = po.items[0];
|
||||||
assert.equal(item.msgid, "Title");
|
assert.equal(item.msgid, 'Title');
|
||||||
assert.equal(item.msgstr, "Titre");
|
assert.equal(item.msgstr, 'Titre');
|
||||||
});
|
});
|
||||||
|
|
||||||
it('Handles multi-line strings', function () {
|
it('Handles multi-line strings', function () {
|
||||||
@@ -19,47 +19,83 @@ describe('Parse', function () {
|
|||||||
assert.equal(po.items.length, 1);
|
assert.equal(po.items.length, 1);
|
||||||
|
|
||||||
var item = po.items[0];
|
var item = po.items[0];
|
||||||
assert.equal(item.msgid, "The following placeholder tokens can be used in both paths and titles. When used in a path or title, they will be replaced with the appropriate values.");
|
assert.equal(item.msgid, 'The following placeholder tokens can be used in both paths and titles. When used in a path or title, they will be replaced with the appropriate values.');
|
||||||
assert.equal(item.msgstr, "Les ébauches de jetons suivantes peuvent être utilisées à la fois dans les chemins et dans les titres. Lorsqu'elles sont utilisées dans un chemin ou un titre, elles seront remplacées par les valeurs appropriées.");
|
assert.equal(item.msgstr, 'Les ébauches de jetons suivantes peuvent être utilisées à la fois dans les chemins et dans les titres. Lorsqu\'elles sont utilisées dans un chemin ou un titre, elles seront remplacées par les valeurs appropriées.');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Handles multi-line headers', function () {
|
||||||
|
var po = PO.parse(fs.readFileSync(__dirname + '/fixtures/multi-line.po', 'utf8'));
|
||||||
|
assert.notEqual(po, null);
|
||||||
|
assert.equal(po.items.length, 1);
|
||||||
|
|
||||||
|
assert.equal(po.headers['Plural-Forms'], 'nplurals=3; plural=n==1 ? 0 : n%10>=2 && n%10<=4 && (n%100<10 || n%100>=20) ? 1 : 2;');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Handle empty comments', function (done) {
|
||||||
|
PO.load(__dirname + '/fixtures/comment.po', function (err, po) {
|
||||||
|
assert.equal(err, null);
|
||||||
|
|
||||||
|
var item = po.items[1];
|
||||||
|
assert.equal(item.msgid, 'Empty comment');
|
||||||
|
assert.equal(item.msgstr, 'Empty');
|
||||||
|
assert.deepEqual(item.comments, ['']);
|
||||||
|
assert.deepEqual(item.extractedComments, ['']);
|
||||||
|
assert.deepEqual(item.references, ['']);
|
||||||
|
done();
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
it('Handles translator comments', function () {
|
it('Handles translator comments', function () {
|
||||||
var po = PO.parse(fs.readFileSync(__dirname + '/fixtures/comment.po', 'utf8'));
|
var po = PO.parse(fs.readFileSync(__dirname + '/fixtures/comment.po', 'utf8'));
|
||||||
assert.notEqual(po, null);
|
assert.notEqual(po, null);
|
||||||
assert.equal(po.items.length, 1);
|
assert.equal(po.items.length, 2);
|
||||||
|
|
||||||
var item = po.items[0];
|
var item = po.items[0];
|
||||||
assert.equal(item.msgid, "Title, as plain text");
|
assert.equal(item.msgid, 'Title, as plain text');
|
||||||
assert.equal(item.msgstr, "Attribut title, en tant que texte brut");
|
assert.equal(item.msgstr, 'Attribut title, en tant que texte brut');
|
||||||
assert.deepEqual(item.comments, ["Translator comment"]);
|
assert.deepEqual(item.comments, ['Translator comment']);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('Handles extracted comments', function () {
|
it('Handles extracted comments', function () {
|
||||||
var po = PO.parse(fs.readFileSync(__dirname + '/fixtures/comment.po', 'utf8'));
|
var po = PO.parse(fs.readFileSync(__dirname + '/fixtures/comment.po', 'utf8'));
|
||||||
assert.notEqual(po, null);
|
assert.notEqual(po, null);
|
||||||
assert.equal(po.items.length, 1);
|
|
||||||
|
|
||||||
var item = po.items[0];
|
|
||||||
assert.equal(item.msgid, "Title, as plain text");
|
|
||||||
assert.equal(item.msgstr, "Attribut title, en tant que texte brut");
|
|
||||||
assert.deepEqual(item.extractedComments, ["Extracted comment"]);
|
|
||||||
});
|
|
||||||
|
|
||||||
it('Handles string references', function () {
|
|
||||||
var po = PO.parse(fs.readFileSync(__dirname + '/fixtures/reference.po', 'utf8'));
|
|
||||||
assert.notEqual(po, null);
|
|
||||||
assert.equal(po.items.length, 2);
|
assert.equal(po.items.length, 2);
|
||||||
|
|
||||||
var item = po.items[0];
|
assert.equal(po.extractedComments.length, 1);
|
||||||
assert.equal(item.msgid, "Title, as plain text");
|
assert.equal(po.extractedComments[0], 'extracted from test');
|
||||||
assert.equal(item.msgstr, "Attribut title, en tant que texte brut");
|
|
||||||
assert.deepEqual(item.comments, ["Comment"]);
|
|
||||||
assert.deepEqual(item.references, [".tmp/crm/controllers/map.js"]);
|
|
||||||
|
|
||||||
item = po.items[1];
|
var item = po.items[0];
|
||||||
assert.equal(item.msgid, "X");
|
assert.equal(item.msgid, 'Title, as plain text');
|
||||||
assert.equal(item.msgstr, "Y");
|
assert.equal(item.msgstr, 'Attribut title, en tant que texte brut');
|
||||||
assert.deepEqual(item.references, ["a", "b"]);
|
assert.deepEqual(item.extractedComments, ['Extracted comment']);
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('Handles string references', function () {
|
||||||
|
var po = PO.parse(fs.readFileSync(__dirname + '/fixtures/reference.po', 'utf8'));
|
||||||
|
assert.notEqual(po, null);
|
||||||
|
assert.equal(po.items.length, 3);
|
||||||
|
|
||||||
|
it('in simple cases', function () {
|
||||||
|
var item = po.items[0];
|
||||||
|
assert.equal(item.msgid, 'Title, as plain text');
|
||||||
|
assert.equal(item.msgstr, 'Attribut title, en tant que texte brut');
|
||||||
|
assert.deepEqual(item.comments, ['Comment']);
|
||||||
|
assert.deepEqual(item.references, ['.tmp/crm/controllers/map.js']);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('with two different references', function () {
|
||||||
|
var item = po.items[1];
|
||||||
|
assert.equal(item.msgid, 'X');
|
||||||
|
assert.equal(item.msgstr, 'Y');
|
||||||
|
assert.deepEqual(item.references, ['a', 'b']);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('and does not process reference items', function () {
|
||||||
|
var item = po.items[2];
|
||||||
|
assert.equal(item.msgid, 'Z');
|
||||||
|
assert.equal(item.msgstr, 'ZZ');
|
||||||
|
assert.deepEqual(item.references, ['standard input:12 standard input:17']);
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
it('Parses flags', function () {
|
it('Parses flags', function () {
|
||||||
@@ -68,8 +104,8 @@ describe('Parse', function () {
|
|||||||
assert.equal(po.items.length, 1);
|
assert.equal(po.items.length, 1);
|
||||||
|
|
||||||
var item = po.items[0];
|
var item = po.items[0];
|
||||||
assert.equal(item.msgid, "Sources");
|
assert.equal(item.msgid, 'Sources');
|
||||||
assert.equal(item.msgstr, "Source");
|
assert.equal(item.msgstr, 'Source');
|
||||||
assert.notEqual(item.flags, null);
|
assert.notEqual(item.flags, null);
|
||||||
assert.equal(item.flags.fuzzy, true);
|
assert.equal(item.flags.fuzzy, true);
|
||||||
});
|
});
|
||||||
@@ -85,35 +121,66 @@ describe('Parse', function () {
|
|||||||
assert.equal(ambiguousItems[1].msgctxt, 'folder action');
|
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 () {
|
it('Handles obsolete items', function () {
|
||||||
var po = PO.parse(fs.readFileSync(__dirname + '/fixtures/commented.po', 'utf8'));
|
var po = PO.parse(fs.readFileSync(__dirname + '/fixtures/commented.po', 'utf8'));
|
||||||
|
|
||||||
assert.equal(po.items.length, 2);
|
assert.equal(po.items.length, 4);
|
||||||
var item = po.items[0];
|
var item = po.items[0];
|
||||||
assert.equal(item.obsolete, false);
|
assert.equal(item.obsolete, false);
|
||||||
assert.equal(item.msgid, "{{dataLoader.data.length}} results");
|
assert.equal(item.msgid, '{{dataLoader.data.length}} results');
|
||||||
assert.equal(item.msgstr, "{{dataLoader.data.length}} resultaten");
|
assert.equal(item.msgstr, '{{dataLoader.data.length}} resultaten');
|
||||||
|
|
||||||
item = po.items[1];
|
item = po.items[1];
|
||||||
assert.equal(item.obsolete, true);
|
assert.equal(item.obsolete, true);
|
||||||
assert.equal(item.msgid, "Add order");
|
assert.equal(item.msgid, 'Add order');
|
||||||
assert.equal(item.msgstr, "Order toevoegen");
|
assert.equal(item.msgstr, 'Order toevoegen');
|
||||||
|
|
||||||
|
item = po.items[2];
|
||||||
|
assert.equal(item.obsolete, true);
|
||||||
|
assert.equal(item.msgid, 'Commented item');
|
||||||
|
assert.equal(item.msgstr, 'not sure');
|
||||||
|
|
||||||
|
item = po.items[3];
|
||||||
|
assert.equal(item.obsolete, true);
|
||||||
|
assert.equal(item.msgid, 'Second commented item');
|
||||||
|
assert.equal(item.msgstr, 'also not sure');
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('C-Strings', function () {
|
describe('C-Strings', function () {
|
||||||
|
var po = PO.parse(fs.readFileSync(__dirname + '/fixtures/c-strings.po', 'utf8'));
|
||||||
it('should parse the c-strings.po file', function () {
|
it('should parse the c-strings.po file', function () {
|
||||||
var po = PO.parse(fs.readFileSync(__dirname + '/fixtures/c-strings.po', 'utf8'));
|
|
||||||
|
|
||||||
assert.notEqual(po, null);
|
assert.notEqual(po, null);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should extract strings containing " and \\ characters', function () {
|
it('should extract strings containing " and \\ characters', function () {
|
||||||
var po = PO.parse(fs.readFileSync(__dirname + '/fixtures/c-strings.po', 'utf8'));
|
|
||||||
|
|
||||||
var items = po.items.filter(function (item) {
|
var items = po.items.filter(function (item) {
|
||||||
return (/^The name field must not contain/).test(item.msgid);
|
return (/^The name field must not contain/).test(item.msgid);
|
||||||
});
|
});
|
||||||
assert.equal(items[0].msgid, 'The name field must not contain characters like " or \\');
|
assert.equal(items[0].msgid, 'The name field must not contain characters like " or \\');
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('should handle \\n characters', function () {
|
||||||
|
var item = po.items[1];
|
||||||
|
assert.equal(item.msgid, '%1$s\n%2$s %3$s\n%4$s\n%5$s');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should handle \\t characters', function () {
|
||||||
|
var item = po.items[2];
|
||||||
|
assert.equal(item.msgid, 'define(\'some/test/module\', function () {\n' +
|
||||||
|
'\t\'use strict\';\n' +
|
||||||
|
'\treturn {};\n' +
|
||||||
|
'});\n');
|
||||||
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|||||||
431
test/write.js
431
test/write.js
@@ -2,8 +2,38 @@ var assert = require('assert');
|
|||||||
var fs = require('fs');
|
var fs = require('fs');
|
||||||
var PO = require('..');
|
var PO = require('..');
|
||||||
|
|
||||||
function assertHasLine(str, line) {
|
function assertHasLine(str, line, doNotTrim) {
|
||||||
var lines = str.split("\n");
|
var lines = str.split('\n');
|
||||||
|
var found = false;
|
||||||
|
|
||||||
|
for (var i = 0; i < lines.length; i++) {
|
||||||
|
var lineToCompare = doNotTrim ? lines[i] : lines[i].trim();
|
||||||
|
if (lineToCompare === line) {
|
||||||
|
found = true;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
assert(found, 'Could not find line: ' + line);
|
||||||
|
}
|
||||||
|
|
||||||
|
function assertHasContiguousLines(str, assertedLines) {
|
||||||
|
var assertedLinesJoined = assertedLines.join('\n');
|
||||||
|
|
||||||
|
var trimmedStr = str
|
||||||
|
.split('\n')
|
||||||
|
.map(function (line) {
|
||||||
|
return line.trim();
|
||||||
|
})
|
||||||
|
.join('\n');
|
||||||
|
|
||||||
|
var found = trimmedStr.indexOf(assertedLinesJoined) !== -1;
|
||||||
|
|
||||||
|
assert(found, 'Could not find lines: \n' + assertedLinesJoined);
|
||||||
|
}
|
||||||
|
|
||||||
|
function assertDoesntHaveLine(str, line) {
|
||||||
|
var lines = str.split('\n');
|
||||||
var found = false;
|
var found = false;
|
||||||
|
|
||||||
for (var i = 0; i < lines.length; i++) {
|
for (var i = 0; i < lines.length; i++) {
|
||||||
@@ -13,7 +43,7 @@ function assertHasLine(str, line) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
assert(found, "Could not find line: " + line);
|
assert(!found, 'Shouldn\'t have line: ' + line);
|
||||||
}
|
}
|
||||||
|
|
||||||
describe('Write', function () {
|
describe('Write', function () {
|
||||||
@@ -21,34 +51,53 @@ describe('Write', function () {
|
|||||||
var input = fs.readFileSync(__dirname + '/fixtures/fuzzy.po', 'utf8');
|
var input = fs.readFileSync(__dirname + '/fixtures/fuzzy.po', 'utf8');
|
||||||
var po = PO.parse(input);
|
var po = PO.parse(input);
|
||||||
var str = po.toString();
|
var str = po.toString();
|
||||||
assertHasLine(str, "#, fuzzy");
|
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);
|
||||||
|
|
||||||
|
// Flip flag
|
||||||
|
po.items[0].flags.fuzzy = false;
|
||||||
|
|
||||||
|
var str = po.toString();
|
||||||
|
assertDoesntHaveLine(str, '#, fuzzy');
|
||||||
});
|
});
|
||||||
|
|
||||||
it('write msgid', function () {
|
it('write msgid', function () {
|
||||||
var input = fs.readFileSync(__dirname + '/fixtures/fuzzy.po', 'utf8');
|
var input = fs.readFileSync(__dirname + '/fixtures/fuzzy.po', 'utf8');
|
||||||
var po = PO.parse(input);
|
var po = PO.parse(input);
|
||||||
var str = po.toString();
|
var str = po.toString();
|
||||||
assertHasLine(str, "msgid \"Sources\"");
|
assertHasLine(str, 'msgid "Sources"');
|
||||||
});
|
});
|
||||||
|
|
||||||
it('write msgstr', function () {
|
it('write msgstr', function () {
|
||||||
var input = fs.readFileSync(__dirname + '/fixtures/fuzzy.po', 'utf8');
|
var input = fs.readFileSync(__dirname + '/fixtures/fuzzy.po', 'utf8');
|
||||||
var po = PO.parse(input);
|
var po = PO.parse(input);
|
||||||
var str = po.toString();
|
var str = po.toString();
|
||||||
assertHasLine(str, "msgstr \"Source\"");
|
assertHasLine(str, 'msgstr "Source"');
|
||||||
});
|
});
|
||||||
|
|
||||||
it('write translator comment', function () {
|
it('write translator comment', function () {
|
||||||
var input = fs.readFileSync(__dirname + '/fixtures/comment.po', 'utf8');
|
var input = fs.readFileSync(__dirname + '/fixtures/comment.po', 'utf8');
|
||||||
var po = PO.parse(input);
|
var po = PO.parse(input);
|
||||||
var str = po.toString();
|
var str = po.toString();
|
||||||
assertHasLine(str, "# Translator comment");
|
assertHasLine(str, '# Translator comment');
|
||||||
});
|
});
|
||||||
|
|
||||||
it('write extracted comment', function () {
|
it('write extracted comment', function () {
|
||||||
var input = fs.readFileSync(__dirname + '/fixtures/comment.po', 'utf8');
|
var input = fs.readFileSync(__dirname + '/fixtures/comment.po', 'utf8');
|
||||||
var po = PO.parse(input);
|
var po = PO.parse(input);
|
||||||
var str = po.toString();
|
var str = po.toString();
|
||||||
|
assertHasLine(str, '#. extracted from test');
|
||||||
assertHasLine(str, '#. Extracted comment');
|
assertHasLine(str, '#. Extracted comment');
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -61,6 +110,343 @@ describe('Write', function () {
|
|||||||
assertHasLine(str, '#~ msgstr "Order toevoegen"');
|
assertHasLine(str, '#~ msgstr "Order toevoegen"');
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('write obsolete items with comment', function () {
|
||||||
|
var input = fs.readFileSync(__dirname + '/fixtures/commented.po', 'utf8');
|
||||||
|
var po = PO.parse(input);
|
||||||
|
var str = po.toString();
|
||||||
|
|
||||||
|
//this is what msgcat tool of gettext does: don't print #~ for comments
|
||||||
|
assertHasLine(str, '# commented obsolete item');
|
||||||
|
assertHasLine(str, '#, fuzzy');
|
||||||
|
|
||||||
|
//items made obsolete by commenting every keyword with #~
|
||||||
|
assertHasLine(str, '#~ msgid "Commented item"');
|
||||||
|
assertHasLine(str, '#~ msgstr "not sure"');
|
||||||
|
assertHasLine(str, '#~ msgid "Second commented item"');
|
||||||
|
assertHasLine(str, '#~ msgstr "also not sure"');
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('plurals', function () {
|
||||||
|
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] ""'
|
||||||
|
]);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
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] ""'
|
||||||
|
]);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
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] ""'
|
||||||
|
]);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
describe('C-Strings', function () {
|
describe('C-Strings', function () {
|
||||||
it('should escape "', function () {
|
it('should escape "', function () {
|
||||||
var item = new PO.Item();
|
var item = new PO.Item();
|
||||||
@@ -76,6 +462,15 @@ describe('Write', function () {
|
|||||||
assertHasLine(item.toString(), 'msgid "\\\\ should be written escaped"');
|
assertHasLine(item.toString(), 'msgid "\\\\ should be written escaped"');
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('should escape \\n', function () {
|
||||||
|
var item = new PO.Item();
|
||||||
|
|
||||||
|
item.msgid = '\n should be written escaped';
|
||||||
|
assertHasLine(item.toString(), 'msgid ""');
|
||||||
|
assertHasLine(item.toString(), '"\\n"');
|
||||||
|
assertHasLine(item.toString(), '" should be written escaped"');
|
||||||
|
});
|
||||||
|
|
||||||
it('should write identical file after parsing a file', function () {
|
it('should write identical file after parsing a file', function () {
|
||||||
var input = fs.readFileSync(__dirname + '/fixtures/c-strings.po', 'utf8');
|
var input = fs.readFileSync(__dirname + '/fixtures/c-strings.po', 'utf8');
|
||||||
var po = PO.parse(input);
|
var po = PO.parse(input);
|
||||||
@@ -109,4 +504,26 @@ describe('Write', function () {
|
|||||||
assert.ok(po.toString().indexOf('msgctxt') >= 0);
|
assert.ok(po.toString().indexOf('msgctxt') >= 0);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('should keep the header order', function () {
|
||||||
|
var input = fs.readFileSync(__dirname + '/fixtures/big.po', 'utf8');
|
||||||
|
var po = PO.parse(input);
|
||||||
|
var str = po.toString();
|
||||||
|
|
||||||
|
assertHasContiguousLines(str, [
|
||||||
|
'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:21+0100\\n"',
|
||||||
|
'"Language-Team: French\\n"',
|
||||||
|
'"MIME-Version: 1.0\\n"',
|
||||||
|
'"Content-Type: text/plain; charset=UTF-8\\n"',
|
||||||
|
'"Content-Transfer-Encoding: 8bit\\n"',
|
||||||
|
'"Plural-Forms: nplurals=2; plural=(n > 1);\\n"',
|
||||||
|
'"Last-Translator: Ruben Vermeersch <ruben@rocketeer.be>\\n"',
|
||||||
|
'"Language: fr\\n"',
|
||||||
|
'"X-Generator: Poedit 1.6.2\\n"',
|
||||||
|
]);
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|||||||
Reference in New Issue
Block a user