UNPKG

phaser-ce

Version:

Phaser CE (Community Edition) is a fast, free and fun HTML5 Game Framework for Desktop and Mobile web browsers.

344 lines (275 loc) 10.8 kB
/** * Add comments in a TypeScript definition file */ 'use strict'; var ts = require('typescript'); var fs = require('fs'); var TypeScriptDocGenerator = (function () { function TypeScriptDocGenerator (tsDefFileName, jsdocJsonFileName) { this.nbCharsAdded = 0; this.tsDefFileName = ts.normalizePath(tsDefFileName); this.tsDefFileContent = fs.readFileSync(this.tsDefFileName, 'utf-8').toString(); this.delintNodeFunction = this.delintNode.bind(this); var jsonDocsFileContent = fs.readFileSync(jsdocJsonFileName, 'utf-8').toString(); this.docs = JSON.parse(jsonDocsFileContent); var options = { target: ts.ScriptTarget.ES5, module: ts.ModuleKind.AMD }; var host = ts.createCompilerHost(options); var program = ts.createProgram([ this.tsDefFileName ], options, host); this.sourceFile = program.getSourceFile(this.tsDefFileName); } TypeScriptDocGenerator.prototype.getTsDefCommentedFileContent = function () { this.scan(); return this.tsDefFileContent; }; TypeScriptDocGenerator.prototype.repeatSpaces = function (nb) { var res = ''; for (var i = 0; i < nb; i++) { res += ' '; } return res; }; TypeScriptDocGenerator.prototype.insertComment = function (commentLines, position) { if ((commentLines != null) && (commentLines.length > 0)) { var nbChars = 0; for (var i = 0; i < commentLines.length; i++) { nbChars += commentLines[i].trim().length; } if (nbChars > 0) { var lc = this.sourceFile.getLineAndCharacterFromPosition(position); var nbSpaces = lc.character - 1; var startLinePosition = this.sourceFile.getLineStarts()[lc.line - 1]; var comment = '\r\n' + this.repeatSpaces(nbSpaces) + '/**\r\n'; for (var j = 0; j < commentLines.length; j++) { comment += this.repeatSpaces(nbSpaces) + '* ' + commentLines[j].trimRight() + '\r\n'; } comment += this.repeatSpaces(nbSpaces) + '*/\r\n'; this.tsDefFileContent = this.tsDefFileContent.substr(0, startLinePosition + this.nbCharsAdded) + comment + this.tsDefFileContent.substr(startLinePosition + this.nbCharsAdded); this.nbCharsAdded += comment.length; } } }; TypeScriptDocGenerator.prototype.cleanEndLine = function (str) { return str.replace(new RegExp('[' + '\r\n' + ']', 'g'), '\n').replace(new RegExp('[' + '\r' + ']', 'g'), '\n'); }; TypeScriptDocGenerator.prototype.findClass = function (className) { if (className.indexOf('p2.') === 0) { className = className.replace('p2.', 'Phaser.Physics.P2.'); } var elements = this.docs.classes.filter(function (element) { return (element.name === className); }); return elements[0]; }; TypeScriptDocGenerator.prototype.generateClassComments = function (className) { var c = this.findClass(className); if (c != null) { var comments = []; comments = comments.concat(this.cleanEndLine(c.description).split('\n')); return comments; } else { return null; } }; TypeScriptDocGenerator.prototype.generateMemberComments = function (className, memberName) { var c = this.findClass(className); if (c != null) { for (var i = 0; i < c.members.length; i++) { if (c.members[i].name === memberName) { var m = c.members[i]; var comments = []; comments = comments.concat(this.cleanEndLine(m.description).split('\n')); if ((m.default != null) && (m.default !== '')) { comments.push('Default: ' + m.default); } return comments; } } } else { return null; } }; TypeScriptDocGenerator.prototype.generateFunctionComments = function (className, functionName) { var c = this.findClass(className); if (c != null) { for (var i = 0; i < c.functions.length; i++) { if (c.functions[i].name === functionName) { var f = c.functions[i]; var comments = []; comments = comments.concat(this.cleanEndLine(f.description).split('\n')); if (f.parameters.length > 0) { comments.push(''); } for (var j = 0; j < f.parameters.length; j++) { var p = f.parameters[j]; if (p.type === '*') { p.name = 'args'; } var def = ''; if ((p.default != null) && (p.default !== '')) { def = ' - Default: ' + p.default; } var paramComments = this.cleanEndLine(p.description).split('\n'); for (var k = 0; k < paramComments.length; k++) { if (k === 0) { comments.push('@param ' + p.name + ' ' + paramComments[k].trim() + ((k === paramComments.length - 1) ? def : '')); } else { comments.push(this.repeatSpaces(('@param ' + p.name + ' ').length) + paramComments[k].trim() + ((k === paramComments.length - 1) ? def : '')); } } } if ((f.returns != null) && (f.returns.description.trim().length > 0)) { var returnComments = this.cleanEndLine(f.returns.description).split('\n'); for (var l = 0; l < returnComments.length; l++) { if (l === 0) { comments.push('@return ' + returnComments[l].trim()); } else { comments.push(this.repeatSpaces(('@return ').length) + returnComments[l].trim()); } } } return comments; } } } else { return null; } }; TypeScriptDocGenerator.prototype.generateConstructorComments = function (className) { var c = this.findClass(className); if (c != null) { var con = c.constructor; var comments = []; comments = comments.concat(this.cleanEndLine(con.description).split('\n')); if (con.parameters.length > 0) { comments.push(''); } for (var j = 0; j < con.parameters.length; j++) { var p = con.parameters[j]; if (p.type === '*') { p.name = 'args'; } var def = ''; if ((p.default != null) && (p.default !== '')) { def = ' - Default: ' + p.default; } var paramComments = this.cleanEndLine(p.description).split('\n'); for (var k = 0; k < paramComments.length; k++) { if (k === 0) { comments.push('@param ' + p.name + ' ' + paramComments[k].trim() + ((k === paramComments.length - 1) ? def : '')); } else { comments.push(this.repeatSpaces(('@param ' + p.name + ' ').length) + paramComments[k].trim() + ((k === paramComments.length - 1) ? def : '')); } } } return comments; } else { return null; } }; TypeScriptDocGenerator.prototype.scan = function () { this.delintNode(this.sourceFile); }; TypeScriptDocGenerator.prototype.getClassName = function (node) { var fullName = ''; if (node.kind === ts.SyntaxKind.ClassDeclaration) { fullName = node.name.getText(); } var parent = node.parent; while (parent != null) { if (parent.kind === ts.SyntaxKind.ModuleDeclaration || parent.kind === ts.SyntaxKind.ClassDeclaration) { fullName = parent.name.getText() + ((fullName !== '') ? '.' + fullName : fullName); } parent = parent.parent; } return fullName; }; TypeScriptDocGenerator.prototype.delintNode = function (node) { switch (node.kind) { case ts.SyntaxKind.Constructor: this.insertComment(this.generateConstructorComments(this.getClassName(node)), node.getStart()); break; case ts.SyntaxKind.ClassDeclaration: this.insertComment(this.generateClassComments(this.getClassName(node)), node.getStart()); break; case ts.SyntaxKind.Property: this.insertComment(this.generateMemberComments(this.getClassName(node), node.name.getText()), node.getStart()); break; case ts.SyntaxKind.Method: this.insertComment(this.generateFunctionComments(this.getClassName(node), node.name.getText()), node.getStart()); break; } ts.forEachChild(node, this.delintNodeFunction); }; return TypeScriptDocGenerator; })(); module.exports = function (grunt) { grunt.registerMultiTask('buildtsdoc', 'Generate a TypeScript def with comments', function () { var tsdg = new TypeScriptDocGenerator(this.data.tsDefFileName, this.data.jsdocJsonFileName); fs.writeFileSync(this.data.dest, tsdg.getTsDefCommentedFileContent(), 'utf8'); }); };