Compare commits
	
		
			8 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 94ad44f953 | ||
|  | c9811bb7aa | ||
|  | f40ab323dd | ||
|  | 554ed47ba7 | ||
|  | 28683aa4b3 | ||
|  | fdf3988cd0 | ||
|  | a79382fb9d | ||
|  | 456d889ea1 | 
| @@ -2,3 +2,6 @@ | |||||||
| /dist | /dist | ||||||
| /Gruntfile.coffee | /Gruntfile.coffee | ||||||
| /test | /test | ||||||
|  | /.jshintrc | ||||||
|  | /.travis.yml | ||||||
|  | /bower.json | ||||||
|   | |||||||
							
								
								
									
										163
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										163
									
								
								README.md
									
									
									
									
									
								
							| @@ -1,22 +1,163 @@ | |||||||
| 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` 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 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`. | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## 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 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": "0.2.2", | ||||||
|   "authors": [ |   "authors": [ | ||||||
|     "Ruben Vermeersch <ruben@rocketeer.be>" |     "Ruben Vermeersch <ruben@rocketeer.be>" | ||||||
|   ], |   ], | ||||||
|   | |||||||
							
								
								
									
										6
									
								
								dist/pofile.js
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										6
									
								
								dist/pofile.js
									
									
									
									
										vendored
									
									
								
							| @@ -1,4 +1,4 @@ | |||||||
| 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);throw new Error("Cannot find module '"+o+"'")}var f=n[o]={exports:{}};t[o][0].call(f.exports,function(e){var n=t[o][1][e];return s(n?n:e)},f,f.exports,e,t,n,r)}return n[o].exports}var i=typeof require=="function"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})({"W8CkM0":[function(require,module,exports){ | ||||||
| var fs = require('fs'), | var fs = require('fs'), | ||||||
|     isArray = require('lodash.isarray'); |     isArray = require('lodash.isarray'); | ||||||
|  |  | ||||||
| @@ -220,7 +220,7 @@ PO.Item.prototype.toString = function () { | |||||||
| module.exports = PO; | module.exports = PO; | ||||||
|  |  | ||||||
| },{"fs":3,"lodash.isarray":4}],"pofile":[function(require,module,exports){ | },{"fs":3,"lodash.isarray":4}],"pofile":[function(require,module,exports){ | ||||||
| module.exports=require('GPUdqu'); | module.exports=require('W8CkM0'); | ||||||
| },{}],3:[function(require,module,exports){ | },{}],3:[function(require,module,exports){ | ||||||
|  |  | ||||||
| },{}],4:[function(require,module,exports){ | },{}],4:[function(require,module,exports){ | ||||||
| @@ -306,4 +306,4 @@ function isNative(value) { | |||||||
|  |  | ||||||
| module.exports = isNative; | module.exports = isNative; | ||||||
|  |  | ||||||
| },{}]},{},["GPUdqu"]) | },{}]},{},["W8CkM0"]) | ||||||
							
								
								
									
										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);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}({W8CkM0:[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("W8CkM0")},{}],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},{}]},{},["W8CkM0"]); | ||||||
| @@ -1,7 +1,7 @@ | |||||||
| { | { | ||||||
|   "name": "pofile", |   "name": "pofile", | ||||||
|   "description": "Parse and serialize Gettext PO files.", |   "description": "Parse and serialize Gettext PO files.", | ||||||
|   "version": "0.2.0", |   "version": "0.2.2", | ||||||
|   "author": { |   "author": { | ||||||
|     "name": "Ruben Vermeersch", |     "name": "Ruben Vermeersch", | ||||||
|     "email": "ruben@savanne.be", |     "email": "ruben@savanne.be", | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user