| /* | 
|   Copyright (C) 2013 Yusuke Suzuki <utatane.tea@gmail.com> | 
|   | 
|   Redistribution and use in source and binary forms, with or without | 
|   modification, are permitted provided that the following conditions are met: | 
|   | 
|     * Redistributions of source code must retain the above copyright | 
|       notice, this list of conditions and the following disclaimer. | 
|     * Redistributions in binary form must reproduce the above copyright | 
|       notice, this list of conditions and the following disclaimer in the | 
|       documentation and/or other materials provided with the distribution. | 
|   | 
|   THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS 'AS IS' | 
|   AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE | 
|   IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE | 
|   ARE DISCLAIMED. IN NO EVENT SHALL <COPYRIGHT HOLDER> BE LIABLE FOR ANY | 
|   DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES | 
|   (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; | 
|   LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND | 
|   ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | 
|   (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF | 
|   THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | 
| */ | 
|   | 
| (function () { | 
|     'use strict'; | 
|   | 
|     function isExpression(node) { | 
|         if (node == null) { return false; } | 
|         switch (node.type) { | 
|             case 'ArrayExpression': | 
|             case 'AssignmentExpression': | 
|             case 'BinaryExpression': | 
|             case 'CallExpression': | 
|             case 'ConditionalExpression': | 
|             case 'FunctionExpression': | 
|             case 'Identifier': | 
|             case 'Literal': | 
|             case 'LogicalExpression': | 
|             case 'MemberExpression': | 
|             case 'NewExpression': | 
|             case 'ObjectExpression': | 
|             case 'SequenceExpression': | 
|             case 'ThisExpression': | 
|             case 'UnaryExpression': | 
|             case 'UpdateExpression': | 
|                 return true; | 
|         } | 
|         return false; | 
|     } | 
|   | 
|     function isIterationStatement(node) { | 
|         if (node == null) { return false; } | 
|         switch (node.type) { | 
|             case 'DoWhileStatement': | 
|             case 'ForInStatement': | 
|             case 'ForStatement': | 
|             case 'WhileStatement': | 
|                 return true; | 
|         } | 
|         return false; | 
|     } | 
|   | 
|     function isStatement(node) { | 
|         if (node == null) { return false; } | 
|         switch (node.type) { | 
|             case 'BlockStatement': | 
|             case 'BreakStatement': | 
|             case 'ContinueStatement': | 
|             case 'DebuggerStatement': | 
|             case 'DoWhileStatement': | 
|             case 'EmptyStatement': | 
|             case 'ExpressionStatement': | 
|             case 'ForInStatement': | 
|             case 'ForStatement': | 
|             case 'IfStatement': | 
|             case 'LabeledStatement': | 
|             case 'ReturnStatement': | 
|             case 'SwitchStatement': | 
|             case 'ThrowStatement': | 
|             case 'TryStatement': | 
|             case 'VariableDeclaration': | 
|             case 'WhileStatement': | 
|             case 'WithStatement': | 
|                 return true; | 
|         } | 
|         return false; | 
|     } | 
|   | 
|     function isSourceElement(node) { | 
|       return isStatement(node) || node != null && node.type === 'FunctionDeclaration'; | 
|     } | 
|   | 
|     function trailingStatement(node) { | 
|         switch (node.type) { | 
|         case 'IfStatement': | 
|             if (node.alternate != null) { | 
|                 return node.alternate; | 
|             } | 
|             return node.consequent; | 
|   | 
|         case 'LabeledStatement': | 
|         case 'ForStatement': | 
|         case 'ForInStatement': | 
|         case 'WhileStatement': | 
|         case 'WithStatement': | 
|             return node.body; | 
|         } | 
|         return null; | 
|     } | 
|   | 
|     function isProblematicIfStatement(node) { | 
|         var current; | 
|   | 
|         if (node.type !== 'IfStatement') { | 
|             return false; | 
|         } | 
|         if (node.alternate == null) { | 
|             return false; | 
|         } | 
|         current = node.consequent; | 
|         do { | 
|             if (current.type === 'IfStatement') { | 
|                 if (current.alternate == null)  { | 
|                     return true; | 
|                 } | 
|             } | 
|             current = trailingStatement(current); | 
|         } while (current); | 
|   | 
|         return false; | 
|     } | 
|   | 
|     module.exports = { | 
|         isExpression: isExpression, | 
|         isStatement: isStatement, | 
|         isIterationStatement: isIterationStatement, | 
|         isSourceElement: isSourceElement, | 
|         isProblematicIfStatement: isProblematicIfStatement, | 
|   | 
|         trailingStatement: trailingStatement | 
|     }; | 
| }()); | 
| /* vim: set sw=4 ts=4 et tw=80 : */ |