Compare commits
	
		
			86 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 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 | ||
|  | 42d6df5373 | ||
|  | dba7465ba7 | ||
|  | d426c114c7 | ||
|  | 5a49a3400b | ||
|  | c103f45002 | ||
|  | cb12e69ef4 | ||
|  | 851a87ebf2 | ||
|  | 94f5f4a83e | ||
|  | 6903bbf967 | ||
|  | b82dea6e42 | ||
|  | 1d74ea4a7a | ||
|  | 783a4129d5 | ||
|  | 8dba54b095 | ||
|  | ac85ba0b9c | ||
|  | ada76116b0 | ||
|  | c5515d6128 | ||
|  | c58333c4d1 | ||
|  | 136b969adb | ||
|  | 02471be49f | ||
|  | 13283fedbe | ||
|  | a5f3059661 | ||
|  | abca810905 | ||
|  | ce410f3427 | ||
|  | 4fa1ce75c5 | ||
|  | b599a76557 | ||
|  | 87e29b6c21 | ||
|  | 4ba2cf1cef | ||
|  | c046b62873 | ||
|  | 6790bfb466 | ||
|  | 94ad44f953 | ||
|  | c9811bb7aa | ||
|  | f40ab323dd | ||
|  | 554ed47ba7 | ||
|  | 28683aa4b3 | ||
|  | fdf3988cd0 | ||
|  | a79382fb9d | ||
|  | 456d889ea1 | 
							
								
								
									
										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 | ||||||
|  | } | ||||||
| @@ -2,3 +2,6 @@ | |||||||
| /dist | /dist | ||||||
| /Gruntfile.coffee | /Gruntfile.coffee | ||||||
| /test | /test | ||||||
|  | /.jshintrc | ||||||
|  | /.travis.yml | ||||||
|  | /bower.json | ||||||
|   | |||||||
| @@ -1,7 +1,6 @@ | |||||||
| language: node_js | language: node_js | ||||||
| node_js: | node_js: | ||||||
|   - "0.8" |  | ||||||
|   - "0.10" |   - "0.10" | ||||||
|   - "0.11" |   - "0.11" | ||||||
| before_script: | 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 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 | ||||||
|   | |||||||
							
								
								
									
										165
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										165
									
								
								README.md
									
									
									
									
									
								
							| @@ -1,22 +1,165 @@ | |||||||
| Used to load and save PO files. | # pofile - gettext .po parsing for JavaScript | ||||||
|  |  | ||||||
|  | > Parse and serialize Gettext PO files. | ||||||
|  |  | ||||||
|  | [](https://travis-ci.org/rubenv/pofile) | ||||||
|  |  | ||||||
|  | ## Usage | ||||||
|  | Add pofile to your project: | ||||||
|  |  | ||||||
|  | ### Installation (Node.JS, browser via Browserified) | ||||||
|  | ``` | ||||||
|  | npm install --save pofile | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | Reference it in your code: | ||||||
|  |  | ||||||
| ```js | ```js | ||||||
| var PO = require('pofile'); | var PO = require('pofile'); | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### Installation (via bower) | ||||||
|  | ``` | ||||||
|  | bower install --save pofile | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | Add it to your HTML file: | ||||||
|  |  | ||||||
|  | ```html | ||||||
|  | <script src="bower_components/pofile/dist/pofile.js"></script> | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | Reference it in your code: | ||||||
|  |  | ||||||
|  | ```js | ||||||
|  | var PO = require('pofile'); | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### Loading and parsing | ||||||
|  |  | ||||||
|  | You can create a new empty PO file by using the class: | ||||||
|  |  | ||||||
|  | ```js | ||||||
|  | var po = new PO(); | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | Or by loading a file (Node.JS only): | ||||||
|  |  | ||||||
|  | ```js | ||||||
| PO.load('text.po', function (err, po) { | PO.load('text.po', function (err, po) { | ||||||
|     // Handle err if needed |     // Handle err if needed | ||||||
|     console.log(po.headers); |     // Do things with po | ||||||
|     console.log(po.items); |  | ||||||
|    |  | ||||||
|     po.save('copy.po', function (err) { |  | ||||||
|         // Handle err if needed |  | ||||||
|         console.log('We copied a PO file for no reason!'); |  | ||||||
|     }); |  | ||||||
| }); | }); | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
|  | Or by parsing a string: | ||||||
|  |  | ||||||
|  | ```js | ||||||
|  | var po = PO.parse(myString); | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### The PO class | ||||||
|  |  | ||||||
|  | The `PO` class exposes three members: | ||||||
|  |  | ||||||
|  | * `comments`: An array of comments (found at the header of the file). | ||||||
|  | * `headers`: A dictionary of the headers. | ||||||
|  | * `items`: An array of `PO.Item` objects, each of which represents a string | ||||||
|  |   from the gettext catalog. | ||||||
|  |  | ||||||
|  | There are two methods available: | ||||||
|  |  | ||||||
|  | * `save`: Accepts a filename and callback, writes the po file to disk. | ||||||
|  |  | ||||||
|  | ```js | ||||||
|  | po.save('out.po', function (err) { | ||||||
|  |     // Handle err if needed | ||||||
|  | }); | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | * `toString`: Serializes the po file to a string. | ||||||
|  |  | ||||||
|  | ### The PO.Item class | ||||||
|  |  | ||||||
|  | The `PO.Item` class exposes the following members: | ||||||
|  |  | ||||||
|  | * `msgid`: The message id. | ||||||
|  | * `msgid_plural`: The plural message id (null if absent). | ||||||
|  | * `msgstr`: An array of translated strings. Items that have no plural msgid | ||||||
|  |   only have one element in this array. | ||||||
|  | * `references`: An array of reference strings. | ||||||
|  | * `comments`: An array of string translator comments. | ||||||
|  | * `extractedComments`: An array of string extracted comments. | ||||||
|  | * `flags`: A dictionary of the string flags. Each flag is mapped to a key with | ||||||
|  |   value true. For instance, a string with the fuzzy flag set will have | ||||||
|  |   `item.flags.fuzzy == true`. | ||||||
|  | * `msgctxt`: Context of the message, an arbitrary string, can be used for disambiguation. | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## Contributing | ||||||
|  |  | ||||||
|  | In lieu of a formal styleguide, take care to maintain the existing coding | ||||||
|  | style. Add unit tests for any new or changed functionality. Lint and test your | ||||||
|  | code using [Grunt](http://gruntjs.com/). | ||||||
|  |  | ||||||
| ## Credits | ## Credits | ||||||
|  |  | ||||||
|   Originally based on node-po (written by Michael Holly). Rebranded because | Originally based on node-po (written by Michael Holly). Rebranded because | ||||||
|   node-po is unmaintained and because this library is no longer limited to | node-po is unmaintained and because this library is no longer limited to | ||||||
|   Node.JS: it works in the browser too. | Node.JS: it works in the browser too. | ||||||
|  |  | ||||||
|  | ### Changes compared to node-po | ||||||
|  |  | ||||||
|  | * Proper handling of async methods that won't crash your Node.JS process when | ||||||
|  |   something goes wrong. | ||||||
|  | * Support for parsing string flags (e.g. fuzzy). | ||||||
|  | * A test suite. | ||||||
|  | * Browser support (through Browserified and bower). | ||||||
|  |  | ||||||
|  | ### Migrating from node-po | ||||||
|  |  | ||||||
|  | You'll need to update the module reference: `require('pofile')` instead of | ||||||
|  | `require('node-po')`. | ||||||
|  |  | ||||||
|  | At the initial release, node-po and pofile have identical APIs, with one small | ||||||
|  | exception: the `save` and `load` methods now take a callback that has an `err` | ||||||
|  | parameter: `(err)` for `save` and `(err, po)` for `load`. This is similar to | ||||||
|  | Node.JS conventions. | ||||||
|  |  | ||||||
|  | Change code such as: | ||||||
|  |  | ||||||
|  | ```js | ||||||
|  | PO.load('text.po', function (po) { | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | To: | ||||||
|  |  | ||||||
|  | ```js | ||||||
|  | PO.load('text.po', function (err, po) { | ||||||
|  |     // Handle err if needed | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ## License  | ||||||
|  |  | ||||||
|  |     (The MIT License) | ||||||
|  |  | ||||||
|  |     Copyright (C) 2013-2017 by Ruben Vermeersch <ruben@rocketeer.be> | ||||||
|  |     Copyright (C) 2012 by Michael Holly | ||||||
|  |  | ||||||
|  |     Permission is hereby granted, free of charge, to any person obtaining a copy | ||||||
|  |     of this software and associated documentation files (the "Software"), to deal | ||||||
|  |     in the Software without restriction, including without limitation the rights | ||||||
|  |     to use, copy, modify, merge, publish, distribute, sublicense, and/or sell | ||||||
|  |     copies of the Software, and to permit persons to whom the Software is | ||||||
|  |     furnished to do so, subject to the following conditions: | ||||||
|  |  | ||||||
|  |     The above copyright notice and this permission notice shall be included in | ||||||
|  |     all copies or substantial portions of the Software. | ||||||
|  |  | ||||||
|  |     THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | ||||||
|  |     IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | ||||||
|  |     FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE | ||||||
|  |     AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER | ||||||
|  |     LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, | ||||||
|  |     OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN | ||||||
|  |     THE SOFTWARE. | ||||||
|   | |||||||
| @@ -1,6 +1,6 @@ | |||||||
| { | { | ||||||
|   "name": "pofile", |   "name": "pofile", | ||||||
|   "version": "0.2.0", |   "version": "1.0.4", | ||||||
|   "authors": [ |   "authors": [ | ||||||
|     "Ruben Vermeersch <ruben@rocketeer.be>" |     "Ruben Vermeersch <ruben@rocketeer.be>" | ||||||
|   ], |   ], | ||||||
|   | |||||||
							
								
								
									
										351
									
								
								dist/pofile.js
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										351
									
								
								dist/pofile.js
									
									
									
									
										vendored
									
									
								
							| @@ -1,6 +1,7 @@ | |||||||
| 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})({"GPUdqu":[function(require,module,exports){ | 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);var f=new Error("Cannot find module '"+o+"'");throw f.code="MODULE_NOT_FOUND",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.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})({1:[function(require,module,exports){ | ||||||
| var fs = require('fs'), |  | ||||||
|     isArray = require('lodash.isarray'); | },{}],"pofile":[function(require,module,exports){ | ||||||
|  | var fs = require('fs'); | ||||||
|  |  | ||||||
| function trim(string) { | function trim(string) { | ||||||
|     return string.replace(/^\s+|\s+$/g, ''); |     return string.replace(/^\s+|\s+$/g, ''); | ||||||
| @@ -8,6 +9,7 @@ function trim(string) { | |||||||
|  |  | ||||||
| var PO = function () { | var PO = function () { | ||||||
|     this.comments = []; |     this.comments = []; | ||||||
|  |     this.extractedComments = []; | ||||||
|     this.headers = {}; |     this.headers = {}; | ||||||
|     this.items = []; |     this.items = []; | ||||||
| }; | }; | ||||||
| @@ -17,21 +19,26 @@ PO.prototype.save = function (filename, 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); | ||||||
|         }); |         }); | ||||||
|     } |     } | ||||||
|  |     if (this.extractedComments) { | ||||||
|  |         this.extractedComments.forEach(function (comment) { | ||||||
|  |             lines.push('#. ' + comment); | ||||||
|  |         }); | ||||||
|  |     } | ||||||
|  |  | ||||||
|     lines.push('msgid ""'); |     lines.push('msgid ""'); | ||||||
|     lines.push('msgstr ""'); |     lines.push('msgstr ""'); | ||||||
|  |  | ||||||
|     var keys = Object.keys(this.headers); |     var keys = Object.keys(this.headers); | ||||||
|  |     var self = this; | ||||||
|     keys.forEach(function (key) { |     keys.forEach(function (key) { | ||||||
|         lines.push('"' + key + ': ' + that.headers[key] + '\\n"'); |         lines.push('"' + key + ': ' + self.headers[key] + '\\n"'); | ||||||
|     }); |     }); | ||||||
|  |  | ||||||
|     lines.push(''); |     lines.push(''); | ||||||
| @@ -41,7 +48,7 @@ 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) { | ||||||
| @@ -57,10 +64,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': '', | ||||||
| @@ -75,25 +92,44 @@ PO.parse = function (data) { | |||||||
|         'Plural-Forms': '', |         'Plural-Forms': '', | ||||||
|     }; |     }; | ||||||
|  |  | ||||||
|     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; | ||||||
|         } |         } | ||||||
|     }); |     }); | ||||||
|  |  | ||||||
|     var item = new PO.Item(), |     var item = new PO.Item(); | ||||||
|         context = null, |     var context = null; | ||||||
|         plural = 0; |     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(); | ||||||
|         } |         } | ||||||
| @@ -102,57 +138,99 @@ PO.parse = function (data) { | |||||||
|     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 | ||||||
|  |             //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)); | ||||||
|  |             lineObsolete = true; | ||||||
|  |         } | ||||||
|  |  | ||||||
|         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(/^#/)) { // Comment |  | ||||||
|             finish(); |             finish(); | ||||||
|             item.comments.push(trim(line.replace(/^#/, ''))); |             item.comments.push(trim(line.replace(/^#($|\s+)/, ''))); | ||||||
|         } |         } else if (line.match(/^#\./)) { // Extracted comment | ||||||
|         else if (line.match(/^msgid_plural/)) { // Plural form |             finish(); | ||||||
|  |             item.extractedComments.push(trim(line.replace(/^#\./, ''))); | ||||||
|  |         } 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 { // Probably multiline string or blank |         } else if (line.match(/^msgctxt/)) { // Context | ||||||
|  |             finish(); | ||||||
|  |             item.msgctxt = extract(line); | ||||||
|  |             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); | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  |  | ||||||
|  |         if (lineObsolete) { | ||||||
|  |             // Count obsolete lines for this item | ||||||
|  |             obsoleteCount++; | ||||||
|  |         } | ||||||
|     } |     } | ||||||
|     finish(); |     finish(); | ||||||
|  |  | ||||||
| @@ -161,149 +239,108 @@ PO.parse = function (data) { | |||||||
|  |  | ||||||
| PO.Item = function () { | PO.Item = function () { | ||||||
|     this.msgid = ''; |     this.msgid = ''; | ||||||
|  |     this.msgctxt = null; | ||||||
|     this.references = []; |     this.references = []; | ||||||
|     this.msgid_plural = null; |     this.msgid_plural = null; | ||||||
|     this.msgstr = []; |     this.msgstr = []; | ||||||
|     this.comments = []; |     this.comments = []; // translator comments | ||||||
|  |     this.extractedComments = []; | ||||||
|     this.flags = {}; |     this.flags = {}; | ||||||
|  |     this.obsolete = false; | ||||||
| }; | }; | ||||||
|  |  | ||||||
| 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 | ||||||
|  |     var _escape = function (string) { | ||||||
|  |         // don't unescape \n, since string can never contain it | ||||||
|  |         // 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('"' + part + '"'); |                 lines.push('"' + _escape(part) + '"'); | ||||||
|             }); |             }); | ||||||
|         } |         } else { | ||||||
|         else { |             lines.push(keyword + index + ' "' + _escape(text) + '"'); | ||||||
|             lines.push(keyword + index + ' "' + text + '"'); |  | ||||||
|         } |         } | ||||||
|         return lines; |         return lines; | ||||||
|     }; |     }; | ||||||
|  |  | ||||||
|     if (this.references.length > 0) { |     // https://www.gnu.org/software/gettext/manual/html_node/PO-Files.html | ||||||
|         this.references.forEach(function (ref) { |     // says order is translator-comments, extracted-comments, references, flags | ||||||
|             lines.push('#: ' + ref); |  | ||||||
|         }); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     var flags = Object.keys(this.flags); |     this.comments.forEach(function (c) { | ||||||
|  |         lines.push('# ' + c); | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     this.extractedComments.forEach(function (c) { | ||||||
|  |         lines.push('#. ' + c); | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     this.references.forEach(function (ref) { | ||||||
|  |         lines.push('#: ' + ref); | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     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 ? '#~ ' : ''; | ||||||
|  |  | ||||||
|     ['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) { |             if (Array.isArray(text) && text.length > 1) { | ||||||
|                 text.forEach(function (t, i) { |                 text.forEach(function (t, i) { | ||||||
|                     lines = lines.concat(_process(keyword, t, i)); |                     lines = lines.concat(mkObsolete + _process(keyword, t, i)); | ||||||
|                 }); |                 }); | ||||||
|             } |             } else { | ||||||
|             else { |                 var index = (self.msgid_plural && Array.isArray(text)) ? | ||||||
|                 text = isArray(text) ? text.join() : text; |                     0 : | ||||||
|                 lines = lines.concat(_process(keyword, text)); |                     undefined; | ||||||
|  |                 text = Array.isArray(text) ? text.join() : text; | ||||||
|  |                 var processed = _process(keyword, text, index); | ||||||
|  |                 //handle \n in single-line texts (can not be handled in _escape) | ||||||
|  |                 for (var i = 1; i < processed.length - 1; i++) { | ||||||
|  |                     processed[i] = processed[i].slice(0, -1) + '\\n"'; | ||||||
|  |                 } | ||||||
|  |                 lines = lines.concat(mkObsolete + processed.join('\n' + mkObsolete)); | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|     }); |     }); | ||||||
|  |  | ||||||
|     return lines.join("\n"); |     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('GPUdqu'); |  | ||||||
| },{}],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; |  | ||||||
|  |  | ||||||
| },{}]},{},["GPUdqu"]) |  | ||||||
|   | |||||||
							
								
								
									
										2
									
								
								dist/pofile.min.js
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								dist/pofile.min.js
									
									
									
									
										vendored
									
									
								
							| @@ -1 +1 @@ | |||||||
| require=function a(b,c,d){function e(g,h){if(!c[g]){if(!b[g]){var i="function"==typeof require&&require;if(!h&&i)return i(g,!0);if(f)return f(g,!0);throw new Error("Cannot find module '"+g+"'")}var j=c[g]={exports:{}};b[g][0].call(j.exports,function(a){var c=b[g][1][a];return e(c?c:a)},j,j.exports,a,b,c,d)}return c[g].exports}for(var f="function"==typeof require&&require,g=0;g<d.length;g++)e(d[g]);return e}({GPUdqu:[function(a,b){function c(a){return a.replace(/^\s+|\s+$/g,"")}var d=a("fs"),e=a("lodash.isarray"),f=function(){this.comments=[],this.headers={},this.items=[]};f.prototype.save=function(a,b){d.writeFile(a,this.toString(),b)},f.prototype.toString=function(){var a=[],b=this;this.comments&&this.comments.forEach(function(b){a.push("# "+b)}),a.push('msgid ""'),a.push('msgstr ""');var c=Object.keys(this.headers);return c.forEach(function(c){a.push('"'+c+": "+b.headers[c]+'\\n"')}),a.push(""),this.items.forEach(function(b){a.push(b.toString()),a.push("")}),a.join("\n")},f.load=function(a,b){d.readFile(a,"utf-8",function(a,c){if(a)return b(a);var d=f.parse(c);b(null,d)})},f.parse=function(a){function b(){j.msgid.length>0&&(e.items.push(j),j=new f.Item)}function d(a){return a=c(a),a=a.replace(/^[^"]*"|"$/g,""),a=a.replace(/\\"/g,'"'),a=a.replace(/\\\\/g,"\\")}a=a.replace(/\r\n/g,"\n");var e=new f,g=a.split(/\n\n/),h=g.shift(),i=g.join("\n").split(/\n/);e.headers={"Project-Id-Version":"","Report-Msgid-Bugs-To":"","POT-Creation-Date":"","PO-Revision-Date":"","Last-Translator":"",Language:"","Language-Team":"","Content-Type":"","Content-Transfer-Encoding":"","Plural-Forms":""},h.split(/\n/).forEach(function(a){if(a.match(/^#/)&&e.comments.push(a.replace(/^#\s*/,"")),a.match(/^"/)){a=a.trim().replace(/^"/,"").replace(/\\n"$/,"");var b=a.split(/:/),c=b.shift().trim(),d=b.join(":").trim();e.headers[c]=d}});for(var j=new f.Item,k=null,l=0;i.length>0;){var m=c(i.shift());if(m.match(/^#:/))b(),j.references.push(c(m.replace(/^#:/,"")));else if(m.match(/^#,/)){b();for(var n=c(m.replace(/^#,/,"")).split(","),o=0;o<n.length;o++)j.flags[n[o]]=!0}else if(m.match(/^#/))b(),j.comments.push(c(m.replace(/^#/,"")));else if(m.match(/^msgid_plural/))j.msgid_plural=d(m),k="msgid_plural";else if(m.match(/^msgid/))b(),j.msgid=d(m),k="msgid";else if(m.match(/^msgstr/)){var p=m.match(/^msgstr\[(\d+)\]/);l=p&&p[1]?parseInt(p[1]):0,j.msgstr[l]=d(m),k="msgstr"}else m.length>0&&("msgstr"===k?j.msgstr[l]+=d(m):"msgid"===k?j.msgid+=d(m):"msgid_plural"===k&&(j.msgid_plural+=d(m)))}return b(),e},f.Item=function(){this.msgid="",this.references=[],this.msgid_plural=null,this.msgstr=[],this.comments=[],this.flags={}},f.Item.prototype.toString=function(){var a=[],b=this,c=function(a,b,c){var d=[],e=b.split(/\n/),f="undefined"!=typeof c?"["+c+"]":"";return e.length>1?(d.push(a+f+' ""'),e.forEach(function(a){d.push('"'+a+'"')})):d.push(a+f+' "'+b+'"'),d};this.references.length>0&&this.references.forEach(function(b){a.push("#: "+b)});var d=Object.keys(this.flags);return d.length>0&&a.push("#, "+d.join(",")),["msgid","msgid_plural","msgstr"].forEach(function(d){var f=b[d];null!=f&&(e(f)&&f.length>1?f.forEach(function(b,e){a=a.concat(c(d,b,e))}):(f=e(f)?f.join():f,a=a.concat(c(d,f))))}),a.join("\n")},b.exports=f},{fs:3,"lodash.isarray":4}],pofile:[function(a,b){b.exports=a("GPUdqu")},{}],3:[function(){},{}],4:[function(a,b){var c=a("lodash._isnative"),d="[object Array]",e=Object.prototype,f=e.toString,g=c(g=Array.isArray)&&g,h=g||function(a){return a&&"object"==typeof a&&"number"==typeof a.length&&f.call(a)==d||!1};b.exports=h},{"lodash._isnative":5}],5:[function(a,b){function c(a){return"function"==typeof a&&f.test(a)}var d=Object.prototype,e=d.toString,f=RegExp("^"+String(e).replace(/[.*+?^${}()|[\]\\]/g,"\\$&").replace(/toString| for [^\]]+/g,".*?")+"$");b.exports=c},{}]},{},["GPUdqu"]); | require=function a(b,c,d){function e(g,h){if(!c[g]){if(!b[g]){var i="function"==typeof require&&require;if(!h&&i)return i(g,!0);if(f)return f(g,!0);var j=new Error("Cannot find module '"+g+"'");throw j.code="MODULE_NOT_FOUND",j}var k=c[g]={exports:{}};b[g][0].call(k.exports,function(a){var c=b[g][1][a];return e(c?c:a)},k,k.exports,a,b,c,d)}return c[g].exports}for(var f="function"==typeof require&&require,g=0;g<d.length;g++)e(d[g]);return e}({1:[function(a,b,c){},{}],pofile:[function(a,b,c){function d(a){return a.replace(/^\s+|\s+$/g,"")}var e=a("fs"),f=function(){this.comments=[],this.extractedComments=[],this.headers={},this.items=[]};f.prototype.save=function(a,b){e.writeFile(a,this.toString(),b)},f.prototype.toString=function(){var a=[];this.comments&&this.comments.forEach(function(b){a.push("# "+b)}),this.extractedComments&&this.extractedComments.forEach(function(b){a.push("#. "+b)}),a.push('msgid ""'),a.push('msgstr ""');var b=Object.keys(this.headers),c=this;return b.forEach(function(b){a.push('"'+b+": "+c.headers[b]+'\\n"')}),a.push(""),this.items.forEach(function(b){a.push(b.toString()),a.push("")}),a.join("\n")},f.load=function(a,b){e.readFile(a,"utf-8",function(a,c){if(a)return b(a);var d=f.parse(c);b(null,d)})},f.parse=function(a){function b(){j.msgid.length>0&&(m>=n&&(j.obsolete=!0),m=0,n=0,e.items.push(j),j=new f.Item)}function c(a){return a=d(a),a=a.replace(/^[^"]*"|"$/g,""),a=a.replace(/\\([abtnvfr'"\\?]|([0-7]{3})|x([0-9a-fA-F]{2}))/g,function(a,b,c,d){if(c)return String.fromCharCode(parseInt(c,8));if(d)return String.fromCharCode(parseInt(d,16));switch(b){case"a":return"";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 b}})}a=a.replace(/\r\n/g,"\n");for(var e=new f,g=a.split(/\n\n/),h=[];g[0]&&(0===h.length||h[h.length-1].indexOf('msgid ""')<0);)g[0].match(/msgid "[^"]/)?h.push('msgid ""'):h.push(g.shift());h=h.join("\n");var i=g.join("\n").split(/\n/);e.headers={"Project-Id-Version":"","Report-Msgid-Bugs-To":"","POT-Creation-Date":"","PO-Revision-Date":"","Last-Translator":"",Language:"","Language-Team":"","Content-Type":"","Content-Transfer-Encoding":"","Plural-Forms":""},h.split(/\n/).reduce(function(a,b){return a.merge&&(b=a.pop().slice(0,-1)+b.slice(1),delete a.merge),/^".*"$/.test(b)&&!/^".*\\n"$/.test(b)&&(a.merge=!0),a.push(b),a},[]).forEach(function(a){if(a.match(/^#\./))e.extractedComments.push(a.replace(/^#\.\s*/,""));else if(a.match(/^#/))e.comments.push(a.replace(/^#\s*/,""));else if(a.match(/^"/)){a=a.trim().replace(/^"/,"").replace(/\\n"$/,"");var b=a.split(/:/),c=b.shift().trim(),d=b.join(":").trim();e.headers[c]=d}});for(var j=new f.Item,k=null,l=0,m=0,n=0;i.length>0;){var o=d(i.shift()),p=!1;if(o.match(/^#\~/)&&(o=d(o.substring(2)),p=!0),o.match(/^#:/))b(),j.references.push(d(o.replace(/^#:/,"")));else if(o.match(/^#,/)){b();for(var q=d(o.replace(/^#,/,"")).split(","),r=0;r<q.length;r++)j.flags[q[r]]=!0}else if(o.match(/^#($|\s+)/))b(),j.comments.push(d(o.replace(/^#($|\s+)/,"")));else if(o.match(/^#\./))b(),j.extractedComments.push(d(o.replace(/^#\./,"")));else if(o.match(/^msgid_plural/))j.msgid_plural=c(o),k="msgid_plural",n++;else if(o.match(/^msgid/))b(),j.msgid=c(o),k="msgid",n++;else if(o.match(/^msgstr/)){var s=o.match(/^msgstr\[(\d+)\]/);l=s&&s[1]?parseInt(s[1]):0,j.msgstr[l]=c(o),k="msgstr",n++}else o.match(/^msgctxt/)?(b(),j.msgctxt=c(o),n++):o.length>0&&(n++,"msgstr"===k?j.msgstr[l]+=c(o):"msgid"===k?j.msgid+=c(o):"msgid_plural"===k&&(j.msgid_plural+=c(o)));p&&m++}return b(),e},f.Item=function(){this.msgid="",this.msgctxt=null,this.references=[],this.msgid_plural=null,this.msgstr=[],this.comments=[],this.extractedComments=[],this.flags={},this.obsolete=!1},f.Item.prototype.toString=function(){var a=[],b=this,c=function(a){return a=a.replace(/[\x07\b\t\v\f\r"\\]/g,function(a){switch(a){case"":return"\\a";case"\b":return"\\b";case"\t":return"\\t";case"\v":return"\\v";case"\f":return"\\f";case"\r":return"\\r";default:return"\\"+a}})},d=function(a,b,d){var e=[],f=b.split(/\n/),g="undefined"!=typeof d?"["+d+"]":"";return f.length>1?(e.push(a+g+' ""'),f.forEach(function(a){e.push('"'+c(a)+'"')})):e.push(a+g+' "'+c(b)+'"'),e};this.comments.forEach(function(b){a.push("# "+b)}),this.extractedComments.forEach(function(b){a.push("#. "+b)}),this.references.forEach(function(b){a.push("#: "+b)});var e=Object.keys(this.flags).filter(function(a){return!!this.flags[a]},this);e.length>0&&a.push("#, "+e.join(","));var f=this.obsolete?"#~ ":"";return["msgctxt","msgid","msgid_plural","msgstr"].forEach(function(c){var e=b[c];if(null!=e)if(Array.isArray(e)&&e.length>1)e.forEach(function(b,e){a=a.concat(f+d(c,b,e))});else{var g=b.msgid_plural&&Array.isArray(e)?0:void 0;e=Array.isArray(e)?e.join():e;for(var h=d(c,e,g),i=1;i<h.length-1;i++)h[i]=h[i].slice(0,-1)+'\\n"';a=a.concat(f+h.join("\n"+f))}}),a.join("\n")},b.exports=f},{fs:1}]},{},["pofile"]); | ||||||
							
								
								
									
										258
									
								
								lib/po.js
									
									
									
									
									
								
							
							
						
						
									
										258
									
								
								lib/po.js
									
									
									
									
									
								
							| @@ -1,5 +1,4 @@ | |||||||
| var fs = require('fs'), | 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, ''); | ||||||
| @@ -7,6 +6,7 @@ function trim(string) { | |||||||
|  |  | ||||||
| var PO = function () { | var PO = function () { | ||||||
|     this.comments = []; |     this.comments = []; | ||||||
|  |     this.extractedComments = []; | ||||||
|     this.headers = {}; |     this.headers = {}; | ||||||
|     this.items = []; |     this.items = []; | ||||||
| }; | }; | ||||||
| @@ -16,21 +16,26 @@ PO.prototype.save = function (filename, 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); | ||||||
|         }); |         }); | ||||||
|     } |     } | ||||||
|  |     if (this.extractedComments) { | ||||||
|  |         this.extractedComments.forEach(function (comment) { | ||||||
|  |             lines.push('#. ' + comment); | ||||||
|  |         }); | ||||||
|  |     } | ||||||
|  |  | ||||||
|     lines.push('msgid ""'); |     lines.push('msgid ""'); | ||||||
|     lines.push('msgstr ""'); |     lines.push('msgstr ""'); | ||||||
|  |  | ||||||
|     var keys = Object.keys(this.headers); |     var keys = Object.keys(this.headers); | ||||||
|  |     var self = this; | ||||||
|     keys.forEach(function (key) { |     keys.forEach(function (key) { | ||||||
|         lines.push('"' + key + ': ' + that.headers[key] + '\\n"'); |         lines.push('"' + key + ': ' + self.headers[key] + '\\n"'); | ||||||
|     }); |     }); | ||||||
|  |  | ||||||
|     lines.push(''); |     lines.push(''); | ||||||
| @@ -40,7 +45,7 @@ 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) { | ||||||
| @@ -56,10 +61,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,25 +89,44 @@ PO.parse = function (data) { | |||||||
|         'Plural-Forms': '', |         'Plural-Forms': '', | ||||||
|     }; |     }; | ||||||
|  |  | ||||||
|     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; | ||||||
|         } |         } | ||||||
|     }); |     }); | ||||||
|  |  | ||||||
|     var item = new PO.Item(), |     var item = new PO.Item(); | ||||||
|         context = null, |     var context = null; | ||||||
|         plural = 0; |     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(); | ||||||
|         } |         } | ||||||
| @@ -101,57 +135,99 @@ PO.parse = function (data) { | |||||||
|     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 | ||||||
|  |             //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)); | ||||||
|  |             lineObsolete = true; | ||||||
|  |         } | ||||||
|  |  | ||||||
|         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(/^#/)) { // Comment |  | ||||||
|             finish(); |             finish(); | ||||||
|             item.comments.push(trim(line.replace(/^#/, ''))); |             item.comments.push(trim(line.replace(/^#($|\s+)/, ''))); | ||||||
|         } |         } else if (line.match(/^#\./)) { // Extracted comment | ||||||
|         else if (line.match(/^msgid_plural/)) { // Plural form |             finish(); | ||||||
|  |             item.extractedComments.push(trim(line.replace(/^#\./, ''))); | ||||||
|  |         } 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 { // Probably multiline string or blank |         } else if (line.match(/^msgctxt/)) { // Context | ||||||
|  |             finish(); | ||||||
|  |             item.msgctxt = extract(line); | ||||||
|  |             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); | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  |  | ||||||
|  |         if (lineObsolete) { | ||||||
|  |             // Count obsolete lines for this item | ||||||
|  |             obsoleteCount++; | ||||||
|  |         } | ||||||
|     } |     } | ||||||
|     finish(); |     finish(); | ||||||
|  |  | ||||||
| @@ -160,60 +236,106 @@ PO.parse = function (data) { | |||||||
|  |  | ||||||
| PO.Item = function () { | PO.Item = function () { | ||||||
|     this.msgid = ''; |     this.msgid = ''; | ||||||
|  |     this.msgctxt = null; | ||||||
|     this.references = []; |     this.references = []; | ||||||
|     this.msgid_plural = null; |     this.msgid_plural = null; | ||||||
|     this.msgstr = []; |     this.msgstr = []; | ||||||
|     this.comments = []; |     this.comments = []; // translator comments | ||||||
|  |     this.extractedComments = []; | ||||||
|     this.flags = {}; |     this.flags = {}; | ||||||
|  |     this.obsolete = false; | ||||||
| }; | }; | ||||||
|  |  | ||||||
| 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 | ||||||
|  |     var _escape = function (string) { | ||||||
|  |         // don't unescape \n, since string can never contain it | ||||||
|  |         // 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('"' + part + '"'); |                 lines.push('"' + _escape(part) + '"'); | ||||||
|             }); |             }); | ||||||
|         } |         } else { | ||||||
|         else { |             lines.push(keyword + index + ' "' + _escape(text) + '"'); | ||||||
|             lines.push(keyword + index + ' "' + text + '"'); |  | ||||||
|         } |         } | ||||||
|         return lines; |         return lines; | ||||||
|     }; |     }; | ||||||
|  |  | ||||||
|     if (this.references.length > 0) { |     // https://www.gnu.org/software/gettext/manual/html_node/PO-Files.html | ||||||
|         this.references.forEach(function (ref) { |     // says order is translator-comments, extracted-comments, references, flags | ||||||
|             lines.push('#: ' + ref); |  | ||||||
|         }); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     var flags = Object.keys(this.flags); |     this.comments.forEach(function (c) { | ||||||
|  |         lines.push('# ' + c); | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     this.extractedComments.forEach(function (c) { | ||||||
|  |         lines.push('#. ' + c); | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     this.references.forEach(function (ref) { | ||||||
|  |         lines.push('#: ' + ref); | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     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 ? '#~ ' : ''; | ||||||
|  |  | ||||||
|     ['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) { |             if (Array.isArray(text) && text.length > 1) { | ||||||
|                 text.forEach(function (t, i) { |                 text.forEach(function (t, i) { | ||||||
|                     lines = lines.concat(_process(keyword, t, i)); |                     lines = lines.concat(mkObsolete + _process(keyword, t, i)); | ||||||
|                 }); |                 }); | ||||||
|             } |             } else { | ||||||
|             else { |                 var index = (self.msgid_plural && Array.isArray(text)) ? | ||||||
|                 text = isArray(text) ? text.join() : text; |                     0 : | ||||||
|                 lines = lines.concat(_process(keyword, text)); |                     undefined; | ||||||
|  |                 text = Array.isArray(text) ? text.join() : text; | ||||||
|  |                 var processed = _process(keyword, text, index); | ||||||
|  |                 //handle \n in single-line texts (can not be handled in _escape) | ||||||
|  |                 for (var i = 1; i < processed.length - 1; i++) { | ||||||
|  |                     processed[i] = processed[i].slice(0, -1) + '\\n"'; | ||||||
|  |                 } | ||||||
|  |                 lines = lines.concat(mkObsolete + processed.join('\n' + mkObsolete)); | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|     }); |     }); | ||||||
|  |  | ||||||
|     return lines.join("\n"); |     return lines.join('\n'); | ||||||
| }; | }; | ||||||
|  |  | ||||||
| module.exports = PO; | module.exports = PO; | ||||||
|   | |||||||
							
								
								
									
										34
									
								
								package.json
									
									
									
									
									
								
							
							
						
						
									
										34
									
								
								package.json
									
									
									
									
									
								
							| @@ -1,14 +1,18 @@ | |||||||
| { | { | ||||||
|   "name": "pofile", |   "name": "pofile", | ||||||
|   "description": "Parse and serialize Gettext PO files.", |   "description": "Parse and serialize Gettext PO files.", | ||||||
|   "version": "0.2.0", |   "version": "1.0.4", | ||||||
|   "author": { |   "author": { | ||||||
|     "name": "Ruben Vermeersch", |     "name": "Ruben Vermeersch", | ||||||
|     "email": "ruben@savanne.be", |     "email": "ruben@savanne.be", | ||||||
|     "url": "http://savanne.be/" |     "url": "http://savanne.be/" | ||||||
|   }, |   }, | ||||||
|   "contributors": [ |   "contributors": [ | ||||||
|     "Mike Holly" |     "Eyal Lewinsohn", | ||||||
|  |     "Gabe Gorelick", | ||||||
|  |     "Julian Bäume", | ||||||
|  |     "Mike Holly", | ||||||
|  |     "Sander Houttekier" | ||||||
|   ], |   ], | ||||||
|   "homepage": "http://github.com/rubenv/pofile", |   "homepage": "http://github.com/rubenv/pofile", | ||||||
|   "repository": { |   "repository": { | ||||||
| @@ -24,23 +28,23 @@ | |||||||
|     "po" |     "po" | ||||||
|   ], |   ], | ||||||
|   "scripts": { |   "scripts": { | ||||||
|     "test": "grunt test" |     "test": "grunt test", | ||||||
|  |     "prepublish": "grunt build" | ||||||
|   }, |   }, | ||||||
|   "directories": { |   "directories": { | ||||||
|     "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" |  | ||||||
|   } |  | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										11
									
								
								test/fixtures/big.po
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										11
									
								
								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 "" | ||||||
| @@ -285,3 +286,13 @@ msgstr "Attribut 'title' du lien" | |||||||
| # Comment | # 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" | ||||||
|  |  | ||||||
|  | # Empty should be adjective | ||||||
|  | msgctxt "folder display" | ||||||
|  | msgid "Empty folder" | ||||||
|  | msgstr "This folder is empty." | ||||||
|  |  | ||||||
|  | # Empty should be verb | ||||||
|  | msgctxt "folder action" | ||||||
|  | msgid "Empty folder" | ||||||
|  | msgstr "Make this folder empty." | ||||||
|   | |||||||
							
								
								
									
										59
									
								
								test/fixtures/c-strings.po
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										59
									
								
								test/fixtures/c-strings.po
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,59 @@ | |||||||
|  | # 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" | ||||||
|  | "Report-Msgid-Bugs-To: \n" | ||||||
|  | "POT-Creation-Date: 2011-12-31 23:39+0000\n" | ||||||
|  | "PO-Revision-Date: 2013-12-17 14:59+0100\n" | ||||||
|  | "Last-Translator: Ruben Vermeersch <ruben@rocketeer.be>\n" | ||||||
|  | "Language: fr\n" | ||||||
|  | "Language-Team: French\n" | ||||||
|  | "Content-Type: text/plain; charset=UTF-8\n" | ||||||
|  | "Content-Transfer-Encoding: 8bit\n" | ||||||
|  | "Plural-Forms: nplurals=2; plural=(n > 1);\n" | ||||||
|  | "MIME-Version: 1.0\n" | ||||||
|  | "X-Generator: Poedit 1.6.2\n" | ||||||
|  |  | ||||||
|  | msgid "The name field must not contain characters like \" or \\" | ||||||
|  | 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" | ||||||
							
								
								
									
										12
									
								
								test/fixtures/comment.po
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										12
									
								
								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" | ||||||
| @@ -15,6 +16,15 @@ msgstr "" | |||||||
| "Language: fr\n" | "Language: fr\n" | ||||||
| "X-Generator: Poedit 1.6.2\n" | "X-Generator: Poedit 1.6.2\n" | ||||||
|  |  | ||||||
| # Comment | # Translator 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" | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										33
									
								
								test/fixtures/commented.po
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										33
									
								
								test/fixtures/commented.po
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,33 @@ | |||||||
|  | msgid "" | ||||||
|  | msgstr "" | ||||||
|  | "Project-Id-Version: Test\n" | ||||||
|  | "POT-Creation-Date: \n" | ||||||
|  | "PO-Revision-Date: 2014-02-17 14:11+0100\n" | ||||||
|  | "Last-Translator: Ruben Vermeersch <ruben@rocketeer.be>\n" | ||||||
|  | "Language-Team: \n" | ||||||
|  | "Language: nl\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" | ||||||
|  | "X-Generator: Poedit 1.6.4\n" | ||||||
|  | "X-Poedit-SourceCharset: UTF-8\n" | ||||||
|  | "X-POOTLE-MTIME: 1390921449.000000\n" | ||||||
|  |  | ||||||
|  |  | ||||||
|  | #: .tmp/ui/settings/views/console-modal.html | ||||||
|  | msgid "{{dataLoader.data.length}} results" | ||||||
|  | msgstr "{{dataLoader.data.length}} resultaten" | ||||||
|  |  | ||||||
|  | #~ msgid "Add order" | ||||||
|  | #~ 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 "" | ||||||
|  |  | ||||||
							
								
								
									
										32
									
								
								test/fixtures/plural.po
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										32
									
								
								test/fixtures/plural.po
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,32 @@ | |||||||
|  | # 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 source" | ||||||
|  | msgid_plural "{{$count}} sources" | ||||||
|  | msgstr[0] "1 source" | ||||||
|  | msgstr[1] "{{$count}} sources" | ||||||
|  |  | ||||||
|  | # correct plurals, with no translation | ||||||
|  | msgid "1 destination" | ||||||
|  | msgid_plural "{{$count}} destinations" | ||||||
|  | msgstr[0] "" | ||||||
|  |  | ||||||
|  | # incorrect plurals, with no translation | ||||||
|  | msgid "1 mistake" | ||||||
|  | msgid_plural "{{$count}} 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); | ||||||
|  |         }); | ||||||
|  |     }); | ||||||
|  | }); | ||||||
|   | |||||||
							
								
								
									
										156
									
								
								test/parse.js
									
									
									
									
									
								
							
							
						
						
									
										156
									
								
								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, 67); |         assert.equal(po.items.length, 69); | ||||||
|  |  | ||||||
|         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,36 +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 string comments', function () { |     it('Handles multi-line headers', function () { | ||||||
|         var po = PO.parse(fs.readFileSync(__dirname + '/fixtures/comment.po', 'utf8')); |         var po = PO.parse(fs.readFileSync(__dirname + '/fixtures/multi-line.po', 'utf8')); | ||||||
|         assert.notEqual(po, null); |         assert.notEqual(po, null); | ||||||
|         assert.equal(po.items.length, 1); |         assert.equal(po.items.length, 1); | ||||||
|  |  | ||||||
|         var item = po.items[0]; |         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;'); | ||||||
|         assert.equal(item.msgid, "Title, as plain text"); |  | ||||||
|         assert.equal(item.msgstr, "Attribut title, en tant que texte brut"); |  | ||||||
|         assert.deepEqual(item.comments, ["Comment"]); |  | ||||||
|     }); |     }); | ||||||
|  |  | ||||||
|     it('Handles string references', function () { |     it('Handle empty comments', function (done) { | ||||||
|         var po = PO.parse(fs.readFileSync(__dirname + '/fixtures/reference.po', 'utf8')); |         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 () { | ||||||
|  |         var po = PO.parse(fs.readFileSync(__dirname + '/fixtures/comment.po', 'utf8')); | ||||||
|         assert.notEqual(po, null); |         assert.notEqual(po, null); | ||||||
|         assert.equal(po.items.length, 2); |         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, ["Comment"]); |         assert.deepEqual(item.comments, ['Translator comment']); | ||||||
|         assert.deepEqual(item.references, [".tmp/crm/controllers/map.js"]); |     }); | ||||||
|  |  | ||||||
|         item = po.items[1]; |     it('Handles extracted comments', function () { | ||||||
|         assert.equal(item.msgid, "X"); |         var po = PO.parse(fs.readFileSync(__dirname + '/fixtures/comment.po', 'utf8')); | ||||||
|         assert.equal(item.msgstr, "Y"); |         assert.notEqual(po, null); | ||||||
|         assert.deepEqual(item.references, ["a", "b"]); |         assert.equal(po.items.length, 2); | ||||||
|  |  | ||||||
|  |         assert.equal(po.extractedComments.length, 1); | ||||||
|  |         assert.equal(po.extractedComments[0], 'extracted from test'); | ||||||
|  |  | ||||||
|  |         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']); | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     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 () { | ||||||
| @@ -57,9 +104,72 @@ 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); | ||||||
|     }); |     }); | ||||||
|  |  | ||||||
|  |     it('Parses item context', function () { | ||||||
|  |         var po = PO.parse(fs.readFileSync(__dirname + '/fixtures/big.po', 'utf8')); | ||||||
|  |  | ||||||
|  |         var ambiguousItems = po.items.filter(function (item) { | ||||||
|  |             return item.msgid === 'Empty folder'; | ||||||
|  |         }); | ||||||
|  |  | ||||||
|  |         assert.equal(ambiguousItems[0].msgctxt, 'folder display'); | ||||||
|  |         assert.equal(ambiguousItems[1].msgctxt, 'folder action'); | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     it('Handles obsolete items', function () { | ||||||
|  |         var po = PO.parse(fs.readFileSync(__dirname + '/fixtures/commented.po', 'utf8')); | ||||||
|  |  | ||||||
|  |         assert.equal(po.items.length, 4); | ||||||
|  |         var item = po.items[0]; | ||||||
|  |         assert.equal(item.obsolete, false); | ||||||
|  |         assert.equal(item.msgid, '{{dataLoader.data.length}} results'); | ||||||
|  |         assert.equal(item.msgstr, '{{dataLoader.data.length}} resultaten'); | ||||||
|  |  | ||||||
|  |         item = po.items[1]; | ||||||
|  |         assert.equal(item.obsolete, true); | ||||||
|  |         assert.equal(item.msgid, 'Add order'); | ||||||
|  |         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 () { | ||||||
|  |         var po = PO.parse(fs.readFileSync(__dirname + '/fixtures/c-strings.po', 'utf8')); | ||||||
|  |         it('should parse the c-strings.po file', function () { | ||||||
|  |             assert.notEqual(po, null); | ||||||
|  |         }); | ||||||
|  |  | ||||||
|  |         it('should extract strings containing " and \\ characters', function () { | ||||||
|  |             var items = po.items.filter(function (item) { | ||||||
|  |                 return (/^The name field must not contain/).test(item.msgid); | ||||||
|  |             }); | ||||||
|  |             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'); | ||||||
|  |         }); | ||||||
|  |     }); | ||||||
| }); | }); | ||||||
|   | |||||||
							
								
								
									
										180
									
								
								test/write.js
									
									
									
									
									
								
							
							
						
						
									
										180
									
								
								test/write.js
									
									
									
									
									
								
							| @@ -3,7 +3,7 @@ var fs = require('fs'); | |||||||
| var PO = require('..'); | var PO = require('..'); | ||||||
|  |  | ||||||
| function assertHasLine(str, line) { | function assertHasLine(str, line) { | ||||||
|     var lines = str.split("\n"); |     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 +13,36 @@ function assertHasLine(str, line) { | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     assert(found, "Could not find line: " + line); |     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; | ||||||
|  |  | ||||||
|  |     for (var i = 0; i < lines.length; i++) { | ||||||
|  |         if (lines[i].trim() === line) { | ||||||
|  |             found = true; | ||||||
|  |             break; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     assert(!found, 'Shouldn\'t have line: ' + line); | ||||||
| } | } | ||||||
|  |  | ||||||
| describe('Write', function () { | describe('Write', function () { | ||||||
| @@ -21,20 +50,161 @@ 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 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 () { | ||||||
|  |         var input = fs.readFileSync(__dirname + '/fixtures/comment.po', 'utf8'); | ||||||
|  |         var po = PO.parse(input); | ||||||
|  |         var str = po.toString(); | ||||||
|  |         assertHasLine(str, '# Translator comment'); | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     it('write extracted comment', function () { | ||||||
|  |         var input = fs.readFileSync(__dirname + '/fixtures/comment.po', 'utf8'); | ||||||
|  |         var po = PO.parse(input); | ||||||
|  |         var str = po.toString(); | ||||||
|  |         assertHasLine(str, '#. extracted from test'); | ||||||
|  |         assertHasLine(str, '#. Extracted comment'); | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     it('write obsolete items', function () { | ||||||
|  |         var input = fs.readFileSync(__dirname + '/fixtures/commented.po', 'utf8'); | ||||||
|  |         var po = PO.parse(input); | ||||||
|  |         var str = po.toString(); | ||||||
|  |  | ||||||
|  |         assertHasLine(str, '#~ msgid "Add order"'); | ||||||
|  |         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 () { | ||||||
|  |         it('should write multiple msgstrs', function () { | ||||||
|  |             var input = fs.readFileSync(__dirname + '/fixtures/plural.po', 'utf8'); | ||||||
|  |             var po = PO.parse(input); | ||||||
|  |             var str = po.toString(); | ||||||
|  |             assertHasContiguousLines(str, [ | ||||||
|  |                 'msgstr[0] "1 source"', | ||||||
|  |                 'msgstr[1] "{{$count}} sources"' | ||||||
|  |             ]); | ||||||
|  |         }); | ||||||
|  |  | ||||||
|  |         it('should write msgstr[0] when there is no translation', function () { | ||||||
|  |             var input = fs.readFileSync(__dirname + '/fixtures/plural.po', 'utf8'); | ||||||
|  |             var po = PO.parse(input); | ||||||
|  |             var str = po.toString(); | ||||||
|  |             assertHasContiguousLines(str, [ | ||||||
|  |                 'msgid_plural "{{$count}} destinations"', | ||||||
|  |                 'msgstr[0] ""' | ||||||
|  |             ]); | ||||||
|  |         }); | ||||||
|  |  | ||||||
|  |         it('should write msgstr[0] when there is no translation and empty plural translation is stored in msgstr ""', function () { | ||||||
|  |             var input = fs.readFileSync(__dirname + '/fixtures/plural.po', 'utf8'); | ||||||
|  |             var po = PO.parse(input); | ||||||
|  |             var str = po.toString(); | ||||||
|  |             assertHasContiguousLines(str, [ | ||||||
|  |                 'msgid_plural "{{$count}} mistakes"', | ||||||
|  |                 'msgstr[0] ""' | ||||||
|  |             ]); | ||||||
|  |         }); | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     describe('C-Strings', function () { | ||||||
|  |         it('should escape "', function () { | ||||||
|  |             var item = new PO.Item(); | ||||||
|  |  | ||||||
|  |             item.msgid = '" should be written escaped'; | ||||||
|  |             assertHasLine(item.toString(), 'msgid "\\" should be written escaped"'); | ||||||
|  |         }); | ||||||
|  |  | ||||||
|  |         it('shoudl escape \\', function () { | ||||||
|  |             var item = new PO.Item(); | ||||||
|  |  | ||||||
|  |             item.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 () { | ||||||
|  |             var input = fs.readFileSync(__dirname + '/fixtures/c-strings.po', 'utf8'); | ||||||
|  |             var po = PO.parse(input); | ||||||
|  |             var str = po.toString(); | ||||||
|  |  | ||||||
|  |             assert.equal(str, input); | ||||||
|  |         }); | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     describe('msgctxt', function () { | ||||||
|  |         it('should write context field to file', function () { | ||||||
|  |             var input = fs.readFileSync(__dirname + '/fixtures/big.po', 'utf8'); | ||||||
|  |             var po = PO.parse(input); | ||||||
|  |             var str = po.toString(); | ||||||
|  |             assertHasLine(str, 'msgctxt "folder action"'); | ||||||
|  |         }); | ||||||
|  |  | ||||||
|  |         it('should ignore omitted context field', function () { | ||||||
|  |             var po = new PO(); | ||||||
|  |             var item = new PO.Item(); | ||||||
|  |             po.items.push(item); | ||||||
|  |             assert.ok(po.toString().indexOf('msgctxt') < 0); | ||||||
|  |         }); | ||||||
|  |  | ||||||
|  |         it('should write empty context field', function () { | ||||||
|  |             var po = new PO(); | ||||||
|  |             var item = new PO.Item(); | ||||||
|  |  | ||||||
|  |             item.msgctxt = ''; | ||||||
|  |             po.items.push(item); | ||||||
|  |             assert.ok(po.toString().indexOf('msgctxt') >= 0); | ||||||
|  |         }); | ||||||
|     }); |     }); | ||||||
| }); | }); | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user