You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					
					
						
							2626 lines
						
					
					
						
							94 KiB
						
					
					
				
			
		
		
	
	
							2626 lines
						
					
					
						
							94 KiB
						
					
					
				/* | 
						|
  Copyright (C) 2012-2014 Yusuke Suzuki <utatane.tea@gmail.com> | 
						|
  Copyright (C) 2015 Ingvar Stepanyan <me@rreverser.com> | 
						|
  Copyright (C) 2014 Ivan Nikulin <ifaaan@gmail.com> | 
						|
  Copyright (C) 2012-2013 Michael Ficarra <escodegen.copyright@michael.ficarra.me> | 
						|
  Copyright (C) 2012-2013 Mathias Bynens <mathias@qiwi.be> | 
						|
  Copyright (C) 2013 Irakli Gozalishvili <rfobic@gmail.com> | 
						|
  Copyright (C) 2012 Robert Gust-Bardon <donate@robert.gust-bardon.org> | 
						|
  Copyright (C) 2012 John Freeman <jfreeman08@gmail.com> | 
						|
  Copyright (C) 2011-2012 Ariya Hidayat <ariya.hidayat@gmail.com> | 
						|
  Copyright (C) 2012 Joost-Wim Boekesteijn <joost-wim@boekesteijn.nl> | 
						|
  Copyright (C) 2012 Kris Kowal <kris.kowal@cixar.com> | 
						|
  Copyright (C) 2012 Arpad Borsos <arpad.borsos@googlemail.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. | 
						|
*/ | 
						|
 | 
						|
/*global exports:true, require:true, global:true*/ | 
						|
(function () { | 
						|
    'use strict'; | 
						|
 | 
						|
    var Syntax, | 
						|
        Precedence, | 
						|
        BinaryPrecedence, | 
						|
        SourceNode, | 
						|
        estraverse, | 
						|
        esutils, | 
						|
        base, | 
						|
        indent, | 
						|
        json, | 
						|
        renumber, | 
						|
        hexadecimal, | 
						|
        quotes, | 
						|
        escapeless, | 
						|
        newline, | 
						|
        space, | 
						|
        parentheses, | 
						|
        semicolons, | 
						|
        safeConcatenation, | 
						|
        directive, | 
						|
        extra, | 
						|
        parse, | 
						|
        sourceMap, | 
						|
        sourceCode, | 
						|
        preserveBlankLines, | 
						|
        FORMAT_MINIFY, | 
						|
        FORMAT_DEFAULTS; | 
						|
 | 
						|
    estraverse = require('estraverse'); | 
						|
    esutils = require('esutils'); | 
						|
 | 
						|
    Syntax = estraverse.Syntax; | 
						|
 | 
						|
    // Generation is done by generateExpression. | 
						|
    function isExpression(node) { | 
						|
        return CodeGenerator.Expression.hasOwnProperty(node.type); | 
						|
    } | 
						|
 | 
						|
    // Generation is done by generateStatement. | 
						|
    function isStatement(node) { | 
						|
        return CodeGenerator.Statement.hasOwnProperty(node.type); | 
						|
    } | 
						|
 | 
						|
    Precedence = { | 
						|
        Sequence: 0, | 
						|
        Yield: 1, | 
						|
        Assignment: 1, | 
						|
        Conditional: 2, | 
						|
        ArrowFunction: 2, | 
						|
        LogicalOR: 3, | 
						|
        LogicalAND: 4, | 
						|
        BitwiseOR: 5, | 
						|
        BitwiseXOR: 6, | 
						|
        BitwiseAND: 7, | 
						|
        Equality: 8, | 
						|
        Relational: 9, | 
						|
        BitwiseSHIFT: 10, | 
						|
        Additive: 11, | 
						|
        Multiplicative: 12, | 
						|
        Exponentiation: 13, | 
						|
        Await: 14, | 
						|
        Unary: 14, | 
						|
        Postfix: 15, | 
						|
        Call: 16, | 
						|
        New: 17, | 
						|
        TaggedTemplate: 18, | 
						|
        Member: 19, | 
						|
        Primary: 20 | 
						|
    }; | 
						|
 | 
						|
    BinaryPrecedence = { | 
						|
        '||': Precedence.LogicalOR, | 
						|
        '&&': Precedence.LogicalAND, | 
						|
        '|': Precedence.BitwiseOR, | 
						|
        '^': Precedence.BitwiseXOR, | 
						|
        '&': Precedence.BitwiseAND, | 
						|
        '==': Precedence.Equality, | 
						|
        '!=': Precedence.Equality, | 
						|
        '===': Precedence.Equality, | 
						|
        '!==': Precedence.Equality, | 
						|
        'is': Precedence.Equality, | 
						|
        'isnt': Precedence.Equality, | 
						|
        '<': Precedence.Relational, | 
						|
        '>': Precedence.Relational, | 
						|
        '<=': Precedence.Relational, | 
						|
        '>=': Precedence.Relational, | 
						|
        'in': Precedence.Relational, | 
						|
        'instanceof': Precedence.Relational, | 
						|
        '<<': Precedence.BitwiseSHIFT, | 
						|
        '>>': Precedence.BitwiseSHIFT, | 
						|
        '>>>': Precedence.BitwiseSHIFT, | 
						|
        '+': Precedence.Additive, | 
						|
        '-': Precedence.Additive, | 
						|
        '*': Precedence.Multiplicative, | 
						|
        '%': Precedence.Multiplicative, | 
						|
        '/': Precedence.Multiplicative, | 
						|
        '**': Precedence.Exponentiation | 
						|
    }; | 
						|
 | 
						|
    //Flags | 
						|
    var F_ALLOW_IN = 1, | 
						|
        F_ALLOW_CALL = 1 << 1, | 
						|
        F_ALLOW_UNPARATH_NEW = 1 << 2, | 
						|
        F_FUNC_BODY = 1 << 3, | 
						|
        F_DIRECTIVE_CTX = 1 << 4, | 
						|
        F_SEMICOLON_OPT = 1 << 5; | 
						|
 | 
						|
    //Expression flag sets | 
						|
    //NOTE: Flag order: | 
						|
    // F_ALLOW_IN | 
						|
    // F_ALLOW_CALL | 
						|
    // F_ALLOW_UNPARATH_NEW | 
						|
    var E_FTT = F_ALLOW_CALL | F_ALLOW_UNPARATH_NEW, | 
						|
        E_TTF = F_ALLOW_IN | F_ALLOW_CALL, | 
						|
        E_TTT = F_ALLOW_IN | F_ALLOW_CALL | F_ALLOW_UNPARATH_NEW, | 
						|
        E_TFF = F_ALLOW_IN, | 
						|
        E_FFT = F_ALLOW_UNPARATH_NEW, | 
						|
        E_TFT = F_ALLOW_IN | F_ALLOW_UNPARATH_NEW; | 
						|
 | 
						|
    //Statement flag sets | 
						|
    //NOTE: Flag order: | 
						|
    // F_ALLOW_IN | 
						|
    // F_FUNC_BODY | 
						|
    // F_DIRECTIVE_CTX | 
						|
    // F_SEMICOLON_OPT | 
						|
    var S_TFFF = F_ALLOW_IN, | 
						|
        S_TFFT = F_ALLOW_IN | F_SEMICOLON_OPT, | 
						|
        S_FFFF = 0x00, | 
						|
        S_TFTF = F_ALLOW_IN | F_DIRECTIVE_CTX, | 
						|
        S_TTFF = F_ALLOW_IN | F_FUNC_BODY; | 
						|
 | 
						|
    function getDefaultOptions() { | 
						|
        // default options | 
						|
        return { | 
						|
            indent: null, | 
						|
            base: null, | 
						|
            parse: null, | 
						|
            comment: false, | 
						|
            format: { | 
						|
                indent: { | 
						|
                    style: '    ', | 
						|
                    base: 0, | 
						|
                    adjustMultilineComment: false | 
						|
                }, | 
						|
                newline: '\n', | 
						|
                space: ' ', | 
						|
                json: false, | 
						|
                renumber: false, | 
						|
                hexadecimal: false, | 
						|
                quotes: 'single', | 
						|
                escapeless: false, | 
						|
                compact: false, | 
						|
                parentheses: true, | 
						|
                semicolons: true, | 
						|
                safeConcatenation: false, | 
						|
                preserveBlankLines: false | 
						|
            }, | 
						|
            moz: { | 
						|
                comprehensionExpressionStartsWithAssignment: false, | 
						|
                starlessGenerator: false | 
						|
            }, | 
						|
            sourceMap: null, | 
						|
            sourceMapRoot: null, | 
						|
            sourceMapWithCode: false, | 
						|
            directive: false, | 
						|
            raw: true, | 
						|
            verbatim: null, | 
						|
            sourceCode: null | 
						|
        }; | 
						|
    } | 
						|
 | 
						|
    function stringRepeat(str, num) { | 
						|
        var result = ''; | 
						|
 | 
						|
        for (num |= 0; num > 0; num >>>= 1, str += str) { | 
						|
            if (num & 1) { | 
						|
                result += str; | 
						|
            } | 
						|
        } | 
						|
 | 
						|
        return result; | 
						|
    } | 
						|
 | 
						|
    function hasLineTerminator(str) { | 
						|
        return (/[\r\n]/g).test(str); | 
						|
    } | 
						|
 | 
						|
    function endsWithLineTerminator(str) { | 
						|
        var len = str.length; | 
						|
        return len && esutils.code.isLineTerminator(str.charCodeAt(len - 1)); | 
						|
    } | 
						|
 | 
						|
    function merge(target, override) { | 
						|
        var key; | 
						|
        for (key in override) { | 
						|
            if (override.hasOwnProperty(key)) { | 
						|
                target[key] = override[key]; | 
						|
            } | 
						|
        } | 
						|
        return target; | 
						|
    } | 
						|
 | 
						|
    function updateDeeply(target, override) { | 
						|
        var key, val; | 
						|
 | 
						|
        function isHashObject(target) { | 
						|
            return typeof target === 'object' && target instanceof Object && !(target instanceof RegExp); | 
						|
        } | 
						|
 | 
						|
        for (key in override) { | 
						|
            if (override.hasOwnProperty(key)) { | 
						|
                val = override[key]; | 
						|
                if (isHashObject(val)) { | 
						|
                    if (isHashObject(target[key])) { | 
						|
                        updateDeeply(target[key], val); | 
						|
                    } else { | 
						|
                        target[key] = updateDeeply({}, val); | 
						|
                    } | 
						|
                } else { | 
						|
                    target[key] = val; | 
						|
                } | 
						|
            } | 
						|
        } | 
						|
        return target; | 
						|
    } | 
						|
 | 
						|
    function generateNumber(value) { | 
						|
        var result, point, temp, exponent, pos; | 
						|
 | 
						|
        if (value !== value) { | 
						|
            throw new Error('Numeric literal whose value is NaN'); | 
						|
        } | 
						|
        if (value < 0 || (value === 0 && 1 / value < 0)) { | 
						|
            throw new Error('Numeric literal whose value is negative'); | 
						|
        } | 
						|
 | 
						|
        if (value === 1 / 0) { | 
						|
            return json ? 'null' : renumber ? '1e400' : '1e+400'; | 
						|
        } | 
						|
 | 
						|
        result = '' + value; | 
						|
        if (!renumber || result.length < 3) { | 
						|
            return result; | 
						|
        } | 
						|
 | 
						|
        point = result.indexOf('.'); | 
						|
        if (!json && result.charCodeAt(0) === 0x30  /* 0 */ && point === 1) { | 
						|
            point = 0; | 
						|
            result = result.slice(1); | 
						|
        } | 
						|
        temp = result; | 
						|
        result = result.replace('e+', 'e'); | 
						|
        exponent = 0; | 
						|
        if ((pos = temp.indexOf('e')) > 0) { | 
						|
            exponent = +temp.slice(pos + 1); | 
						|
            temp = temp.slice(0, pos); | 
						|
        } | 
						|
        if (point >= 0) { | 
						|
            exponent -= temp.length - point - 1; | 
						|
            temp = +(temp.slice(0, point) + temp.slice(point + 1)) + ''; | 
						|
        } | 
						|
        pos = 0; | 
						|
        while (temp.charCodeAt(temp.length + pos - 1) === 0x30  /* 0 */) { | 
						|
            --pos; | 
						|
        } | 
						|
        if (pos !== 0) { | 
						|
            exponent -= pos; | 
						|
            temp = temp.slice(0, pos); | 
						|
        } | 
						|
        if (exponent !== 0) { | 
						|
            temp += 'e' + exponent; | 
						|
        } | 
						|
        if ((temp.length < result.length || | 
						|
                    (hexadecimal && value > 1e12 && Math.floor(value) === value && (temp = '0x' + value.toString(16)).length < result.length)) && | 
						|
                +temp === value) { | 
						|
            result = temp; | 
						|
        } | 
						|
 | 
						|
        return result; | 
						|
    } | 
						|
 | 
						|
    // Generate valid RegExp expression. | 
						|
    // This function is based on https://github.com/Constellation/iv Engine | 
						|
 | 
						|
    function escapeRegExpCharacter(ch, previousIsBackslash) { | 
						|
        // not handling '\' and handling \u2028 or \u2029 to unicode escape sequence | 
						|
        if ((ch & ~1) === 0x2028) { | 
						|
            return (previousIsBackslash ? 'u' : '\\u') + ((ch === 0x2028) ? '2028' : '2029'); | 
						|
        } else if (ch === 10 || ch === 13) {  // \n, \r | 
						|
            return (previousIsBackslash ? '' : '\\') + ((ch === 10) ? 'n' : 'r'); | 
						|
        } | 
						|
        return String.fromCharCode(ch); | 
						|
    } | 
						|
 | 
						|
    function generateRegExp(reg) { | 
						|
        var match, result, flags, i, iz, ch, characterInBrack, previousIsBackslash; | 
						|
 | 
						|
        result = reg.toString(); | 
						|
 | 
						|
        if (reg.source) { | 
						|
            // extract flag from toString result | 
						|
            match = result.match(/\/([^/]*)$/); | 
						|
            if (!match) { | 
						|
                return result; | 
						|
            } | 
						|
 | 
						|
            flags = match[1]; | 
						|
            result = ''; | 
						|
 | 
						|
            characterInBrack = false; | 
						|
            previousIsBackslash = false; | 
						|
            for (i = 0, iz = reg.source.length; i < iz; ++i) { | 
						|
                ch = reg.source.charCodeAt(i); | 
						|
 | 
						|
                if (!previousIsBackslash) { | 
						|
                    if (characterInBrack) { | 
						|
                        if (ch === 93) {  // ] | 
						|
                            characterInBrack = false; | 
						|
                        } | 
						|
                    } else { | 
						|
                        if (ch === 47) {  // / | 
						|
                            result += '\\'; | 
						|
                        } else if (ch === 91) {  // [ | 
						|
                            characterInBrack = true; | 
						|
                        } | 
						|
                    } | 
						|
                    result += escapeRegExpCharacter(ch, previousIsBackslash); | 
						|
                    previousIsBackslash = ch === 92;  // \ | 
						|
                } else { | 
						|
                    // if new RegExp("\\\n') is provided, create /\n/ | 
						|
                    result += escapeRegExpCharacter(ch, previousIsBackslash); | 
						|
                    // prevent like /\\[/]/ | 
						|
                    previousIsBackslash = false; | 
						|
                } | 
						|
            } | 
						|
 | 
						|
            return '/' + result + '/' + flags; | 
						|
        } | 
						|
 | 
						|
        return result; | 
						|
    } | 
						|
 | 
						|
    function escapeAllowedCharacter(code, next) { | 
						|
        var hex; | 
						|
 | 
						|
        if (code === 0x08  /* \b */) { | 
						|
            return '\\b'; | 
						|
        } | 
						|
 | 
						|
        if (code === 0x0C  /* \f */) { | 
						|
            return '\\f'; | 
						|
        } | 
						|
 | 
						|
        if (code === 0x09  /* \t */) { | 
						|
            return '\\t'; | 
						|
        } | 
						|
 | 
						|
        hex = code.toString(16).toUpperCase(); | 
						|
        if (json || code > 0xFF) { | 
						|
            return '\\u' + '0000'.slice(hex.length) + hex; | 
						|
        } else if (code === 0x0000 && !esutils.code.isDecimalDigit(next)) { | 
						|
            return '\\0'; | 
						|
        } else if (code === 0x000B  /* \v */) { // '\v' | 
						|
            return '\\x0B'; | 
						|
        } else { | 
						|
            return '\\x' + '00'.slice(hex.length) + hex; | 
						|
        } | 
						|
    } | 
						|
 | 
						|
    function escapeDisallowedCharacter(code) { | 
						|
        if (code === 0x5C  /* \ */) { | 
						|
            return '\\\\'; | 
						|
        } | 
						|
 | 
						|
        if (code === 0x0A  /* \n */) { | 
						|
            return '\\n'; | 
						|
        } | 
						|
 | 
						|
        if (code === 0x0D  /* \r */) { | 
						|
            return '\\r'; | 
						|
        } | 
						|
 | 
						|
        if (code === 0x2028) { | 
						|
            return '\\u2028'; | 
						|
        } | 
						|
 | 
						|
        if (code === 0x2029) { | 
						|
            return '\\u2029'; | 
						|
        } | 
						|
 | 
						|
        throw new Error('Incorrectly classified character'); | 
						|
    } | 
						|
 | 
						|
    function escapeDirective(str) { | 
						|
        var i, iz, code, quote; | 
						|
 | 
						|
        quote = quotes === 'double' ? '"' : '\''; | 
						|
        for (i = 0, iz = str.length; i < iz; ++i) { | 
						|
            code = str.charCodeAt(i); | 
						|
            if (code === 0x27  /* ' */) { | 
						|
                quote = '"'; | 
						|
                break; | 
						|
            } else if (code === 0x22  /* " */) { | 
						|
                quote = '\''; | 
						|
                break; | 
						|
            } else if (code === 0x5C  /* \ */) { | 
						|
                ++i; | 
						|
            } | 
						|
        } | 
						|
 | 
						|
        return quote + str + quote; | 
						|
    } | 
						|
 | 
						|
    function escapeString(str) { | 
						|
        var result = '', i, len, code, singleQuotes = 0, doubleQuotes = 0, single, quote; | 
						|
 | 
						|
        for (i = 0, len = str.length; i < len; ++i) { | 
						|
            code = str.charCodeAt(i); | 
						|
            if (code === 0x27  /* ' */) { | 
						|
                ++singleQuotes; | 
						|
            } else if (code === 0x22  /* " */) { | 
						|
                ++doubleQuotes; | 
						|
            } else if (code === 0x2F  /* / */ && json) { | 
						|
                result += '\\'; | 
						|
            } else if (esutils.code.isLineTerminator(code) || code === 0x5C  /* \ */) { | 
						|
                result += escapeDisallowedCharacter(code); | 
						|
                continue; | 
						|
            } else if (!esutils.code.isIdentifierPartES5(code) && (json && code < 0x20  /* SP */ || !json && !escapeless && (code < 0x20  /* SP */ || code > 0x7E  /* ~ */))) { | 
						|
                result += escapeAllowedCharacter(code, str.charCodeAt(i + 1)); | 
						|
                continue; | 
						|
            } | 
						|
            result += String.fromCharCode(code); | 
						|
        } | 
						|
 | 
						|
        single = !(quotes === 'double' || (quotes === 'auto' && doubleQuotes < singleQuotes)); | 
						|
        quote = single ? '\'' : '"'; | 
						|
 | 
						|
        if (!(single ? singleQuotes : doubleQuotes)) { | 
						|
            return quote + result + quote; | 
						|
        } | 
						|
 | 
						|
        str = result; | 
						|
        result = quote; | 
						|
 | 
						|
        for (i = 0, len = str.length; i < len; ++i) { | 
						|
            code = str.charCodeAt(i); | 
						|
            if ((code === 0x27  /* ' */ && single) || (code === 0x22  /* " */ && !single)) { | 
						|
                result += '\\'; | 
						|
            } | 
						|
            result += String.fromCharCode(code); | 
						|
        } | 
						|
 | 
						|
        return result + quote; | 
						|
    } | 
						|
 | 
						|
    /** | 
						|
     * flatten an array to a string, where the array can contain | 
						|
     * either strings or nested arrays | 
						|
     */ | 
						|
    function flattenToString(arr) { | 
						|
        var i, iz, elem, result = ''; | 
						|
        for (i = 0, iz = arr.length; i < iz; ++i) { | 
						|
            elem = arr[i]; | 
						|
            result += Array.isArray(elem) ? flattenToString(elem) : elem; | 
						|
        } | 
						|
        return result; | 
						|
    } | 
						|
 | 
						|
    /** | 
						|
     * convert generated to a SourceNode when source maps are enabled. | 
						|
     */ | 
						|
    function toSourceNodeWhenNeeded(generated, node) { | 
						|
        if (!sourceMap) { | 
						|
            // with no source maps, generated is either an | 
						|
            // array or a string.  if an array, flatten it. | 
						|
            // if a string, just return it | 
						|
            if (Array.isArray(generated)) { | 
						|
                return flattenToString(generated); | 
						|
            } else { | 
						|
                return generated; | 
						|
            } | 
						|
        } | 
						|
        if (node == null) { | 
						|
            if (generated instanceof SourceNode) { | 
						|
                return generated; | 
						|
            } else { | 
						|
                node = {}; | 
						|
            } | 
						|
        } | 
						|
        if (node.loc == null) { | 
						|
            return new SourceNode(null, null, sourceMap, generated, node.name || null); | 
						|
        } | 
						|
        return new SourceNode(node.loc.start.line, node.loc.start.column, (sourceMap === true ? node.loc.source || null : sourceMap), generated, node.name || null); | 
						|
    } | 
						|
 | 
						|
    function noEmptySpace() { | 
						|
        return (space) ? space : ' '; | 
						|
    } | 
						|
 | 
						|
    function join(left, right) { | 
						|
        var leftSource, | 
						|
            rightSource, | 
						|
            leftCharCode, | 
						|
            rightCharCode; | 
						|
 | 
						|
        leftSource = toSourceNodeWhenNeeded(left).toString(); | 
						|
        if (leftSource.length === 0) { | 
						|
            return [right]; | 
						|
        } | 
						|
 | 
						|
        rightSource = toSourceNodeWhenNeeded(right).toString(); | 
						|
        if (rightSource.length === 0) { | 
						|
            return [left]; | 
						|
        } | 
						|
 | 
						|
        leftCharCode = leftSource.charCodeAt(leftSource.length - 1); | 
						|
        rightCharCode = rightSource.charCodeAt(0); | 
						|
 | 
						|
        if ((leftCharCode === 0x2B  /* + */ || leftCharCode === 0x2D  /* - */) && leftCharCode === rightCharCode || | 
						|
            esutils.code.isIdentifierPartES5(leftCharCode) && esutils.code.isIdentifierPartES5(rightCharCode) || | 
						|
            leftCharCode === 0x2F  /* / */ && rightCharCode === 0x69  /* i */) { // infix word operators all start with `i` | 
						|
            return [left, noEmptySpace(), right]; | 
						|
        } else if (esutils.code.isWhiteSpace(leftCharCode) || esutils.code.isLineTerminator(leftCharCode) || | 
						|
                esutils.code.isWhiteSpace(rightCharCode) || esutils.code.isLineTerminator(rightCharCode)) { | 
						|
            return [left, right]; | 
						|
        } | 
						|
        return [left, space, right]; | 
						|
    } | 
						|
 | 
						|
    function addIndent(stmt) { | 
						|
        return [base, stmt]; | 
						|
    } | 
						|
 | 
						|
    function withIndent(fn) { | 
						|
        var previousBase; | 
						|
        previousBase = base; | 
						|
        base += indent; | 
						|
        fn(base); | 
						|
        base = previousBase; | 
						|
    } | 
						|
 | 
						|
    function calculateSpaces(str) { | 
						|
        var i; | 
						|
        for (i = str.length - 1; i >= 0; --i) { | 
						|
            if (esutils.code.isLineTerminator(str.charCodeAt(i))) { | 
						|
                break; | 
						|
            } | 
						|
        } | 
						|
        return (str.length - 1) - i; | 
						|
    } | 
						|
 | 
						|
    function adjustMultilineComment(value, specialBase) { | 
						|
        var array, i, len, line, j, spaces, previousBase, sn; | 
						|
 | 
						|
        array = value.split(/\r\n|[\r\n]/); | 
						|
        spaces = Number.MAX_VALUE; | 
						|
 | 
						|
        // first line doesn't have indentation | 
						|
        for (i = 1, len = array.length; i < len; ++i) { | 
						|
            line = array[i]; | 
						|
            j = 0; | 
						|
            while (j < line.length && esutils.code.isWhiteSpace(line.charCodeAt(j))) { | 
						|
                ++j; | 
						|
            } | 
						|
            if (spaces > j) { | 
						|
                spaces = j; | 
						|
            } | 
						|
        } | 
						|
 | 
						|
        if (typeof specialBase !== 'undefined') { | 
						|
            // pattern like | 
						|
            // { | 
						|
            //   var t = 20;  /* | 
						|
            //                 * this is comment | 
						|
            //                 */ | 
						|
            // } | 
						|
            previousBase = base; | 
						|
            if (array[1][spaces] === '*') { | 
						|
                specialBase += ' '; | 
						|
            } | 
						|
            base = specialBase; | 
						|
        } else { | 
						|
            if (spaces & 1) { | 
						|
                // /* | 
						|
                //  * | 
						|
                //  */ | 
						|
                // If spaces are odd number, above pattern is considered. | 
						|
                // We waste 1 space. | 
						|
                --spaces; | 
						|
            } | 
						|
            previousBase = base; | 
						|
        } | 
						|
 | 
						|
        for (i = 1, len = array.length; i < len; ++i) { | 
						|
            sn = toSourceNodeWhenNeeded(addIndent(array[i].slice(spaces))); | 
						|
            array[i] = sourceMap ? sn.join('') : sn; | 
						|
        } | 
						|
 | 
						|
        base = previousBase; | 
						|
 | 
						|
        return array.join('\n'); | 
						|
    } | 
						|
 | 
						|
    function generateComment(comment, specialBase) { | 
						|
        if (comment.type === 'Line') { | 
						|
            if (endsWithLineTerminator(comment.value)) { | 
						|
                return '//' + comment.value; | 
						|
            } else { | 
						|
                // Always use LineTerminator | 
						|
                var result = '//' + comment.value; | 
						|
                if (!preserveBlankLines) { | 
						|
                    result += '\n'; | 
						|
                } | 
						|
                return result; | 
						|
            } | 
						|
        } | 
						|
        if (extra.format.indent.adjustMultilineComment && /[\n\r]/.test(comment.value)) { | 
						|
            return adjustMultilineComment('/*' + comment.value + '*/', specialBase); | 
						|
        } | 
						|
        return '/*' + comment.value + '*/'; | 
						|
    } | 
						|
 | 
						|
    function addComments(stmt, result) { | 
						|
        var i, len, comment, save, tailingToStatement, specialBase, fragment, | 
						|
            extRange, range, prevRange, prefix, infix, suffix, count; | 
						|
 | 
						|
        if (stmt.leadingComments && stmt.leadingComments.length > 0) { | 
						|
            save = result; | 
						|
 | 
						|
            if (preserveBlankLines) { | 
						|
                comment = stmt.leadingComments[0]; | 
						|
                result = []; | 
						|
 | 
						|
                extRange = comment.extendedRange; | 
						|
                range = comment.range; | 
						|
 | 
						|
                prefix = sourceCode.substring(extRange[0], range[0]); | 
						|
                count = (prefix.match(/\n/g) || []).length; | 
						|
                if (count > 0) { | 
						|
                    result.push(stringRepeat('\n', count)); | 
						|
                    result.push(addIndent(generateComment(comment))); | 
						|
                } else { | 
						|
                    result.push(prefix); | 
						|
                    result.push(generateComment(comment)); | 
						|
                } | 
						|
 | 
						|
                prevRange = range; | 
						|
 | 
						|
                for (i = 1, len = stmt.leadingComments.length; i < len; i++) { | 
						|
                    comment = stmt.leadingComments[i]; | 
						|
                    range = comment.range; | 
						|
 | 
						|
                    infix = sourceCode.substring(prevRange[1], range[0]); | 
						|
                    count = (infix.match(/\n/g) || []).length; | 
						|
                    result.push(stringRepeat('\n', count)); | 
						|
                    result.push(addIndent(generateComment(comment))); | 
						|
 | 
						|
                    prevRange = range; | 
						|
                } | 
						|
 | 
						|
                suffix = sourceCode.substring(range[1], extRange[1]); | 
						|
                count = (suffix.match(/\n/g) || []).length; | 
						|
                result.push(stringRepeat('\n', count)); | 
						|
            } else { | 
						|
                comment = stmt.leadingComments[0]; | 
						|
                result = []; | 
						|
                if (safeConcatenation && stmt.type === Syntax.Program && stmt.body.length === 0) { | 
						|
                    result.push('\n'); | 
						|
                } | 
						|
                result.push(generateComment(comment)); | 
						|
                if (!endsWithLineTerminator(toSourceNodeWhenNeeded(result).toString())) { | 
						|
                    result.push('\n'); | 
						|
                } | 
						|
 | 
						|
                for (i = 1, len = stmt.leadingComments.length; i < len; ++i) { | 
						|
                    comment = stmt.leadingComments[i]; | 
						|
                    fragment = [generateComment(comment)]; | 
						|
                    if (!endsWithLineTerminator(toSourceNodeWhenNeeded(fragment).toString())) { | 
						|
                        fragment.push('\n'); | 
						|
                    } | 
						|
                    result.push(addIndent(fragment)); | 
						|
                } | 
						|
            } | 
						|
 | 
						|
            result.push(addIndent(save)); | 
						|
        } | 
						|
 | 
						|
        if (stmt.trailingComments) { | 
						|
 | 
						|
            if (preserveBlankLines) { | 
						|
                comment = stmt.trailingComments[0]; | 
						|
                extRange = comment.extendedRange; | 
						|
                range = comment.range; | 
						|
 | 
						|
                prefix = sourceCode.substring(extRange[0], range[0]); | 
						|
                count = (prefix.match(/\n/g) || []).length; | 
						|
 | 
						|
                if (count > 0) { | 
						|
                    result.push(stringRepeat('\n', count)); | 
						|
                    result.push(addIndent(generateComment(comment))); | 
						|
                } else { | 
						|
                    result.push(prefix); | 
						|
                    result.push(generateComment(comment)); | 
						|
                } | 
						|
            } else { | 
						|
                tailingToStatement = !endsWithLineTerminator(toSourceNodeWhenNeeded(result).toString()); | 
						|
                specialBase = stringRepeat(' ', calculateSpaces(toSourceNodeWhenNeeded([base, result, indent]).toString())); | 
						|
                for (i = 0, len = stmt.trailingComments.length; i < len; ++i) { | 
						|
                    comment = stmt.trailingComments[i]; | 
						|
                    if (tailingToStatement) { | 
						|
                        // We assume target like following script | 
						|
                        // | 
						|
                        // var t = 20;  /** | 
						|
                        //               * This is comment of t | 
						|
                        //               */ | 
						|
                        if (i === 0) { | 
						|
                            // first case | 
						|
                            result = [result, indent]; | 
						|
                        } else { | 
						|
                            result = [result, specialBase]; | 
						|
                        } | 
						|
                        result.push(generateComment(comment, specialBase)); | 
						|
                    } else { | 
						|
                        result = [result, addIndent(generateComment(comment))]; | 
						|
                    } | 
						|
                    if (i !== len - 1 && !endsWithLineTerminator(toSourceNodeWhenNeeded(result).toString())) { | 
						|
                        result = [result, '\n']; | 
						|
                    } | 
						|
                } | 
						|
            } | 
						|
        } | 
						|
 | 
						|
        return result; | 
						|
    } | 
						|
 | 
						|
    function generateBlankLines(start, end, result) { | 
						|
        var j, newlineCount = 0; | 
						|
 | 
						|
        for (j = start; j < end; j++) { | 
						|
            if (sourceCode[j] === '\n') { | 
						|
                newlineCount++; | 
						|
            } | 
						|
        } | 
						|
 | 
						|
        for (j = 1; j < newlineCount; j++) { | 
						|
            result.push(newline); | 
						|
        } | 
						|
    } | 
						|
 | 
						|
    function parenthesize(text, current, should) { | 
						|
        if (current < should) { | 
						|
            return ['(', text, ')']; | 
						|
        } | 
						|
        return text; | 
						|
    } | 
						|
 | 
						|
    function generateVerbatimString(string) { | 
						|
        var i, iz, result; | 
						|
        result = string.split(/\r\n|\n/); | 
						|
        for (i = 1, iz = result.length; i < iz; i++) { | 
						|
            result[i] = newline + base + result[i]; | 
						|
        } | 
						|
        return result; | 
						|
    } | 
						|
 | 
						|
    function generateVerbatim(expr, precedence) { | 
						|
        var verbatim, result, prec; | 
						|
        verbatim = expr[extra.verbatim]; | 
						|
 | 
						|
        if (typeof verbatim === 'string') { | 
						|
            result = parenthesize(generateVerbatimString(verbatim), Precedence.Sequence, precedence); | 
						|
        } else { | 
						|
            // verbatim is object | 
						|
            result = generateVerbatimString(verbatim.content); | 
						|
            prec = (verbatim.precedence != null) ? verbatim.precedence : Precedence.Sequence; | 
						|
            result = parenthesize(result, prec, precedence); | 
						|
        } | 
						|
 | 
						|
        return toSourceNodeWhenNeeded(result, expr); | 
						|
    } | 
						|
 | 
						|
    function CodeGenerator() { | 
						|
    } | 
						|
 | 
						|
    // Helpers. | 
						|
 | 
						|
    CodeGenerator.prototype.maybeBlock = function(stmt, flags) { | 
						|
        var result, noLeadingComment, that = this; | 
						|
 | 
						|
        noLeadingComment = !extra.comment || !stmt.leadingComments; | 
						|
 | 
						|
        if (stmt.type === Syntax.BlockStatement && noLeadingComment) { | 
						|
            return [space, this.generateStatement(stmt, flags)]; | 
						|
        } | 
						|
 | 
						|
        if (stmt.type === Syntax.EmptyStatement && noLeadingComment) { | 
						|
            return ';'; | 
						|
        } | 
						|
 | 
						|
        withIndent(function () { | 
						|
            result = [ | 
						|
                newline, | 
						|
                addIndent(that.generateStatement(stmt, flags)) | 
						|
            ]; | 
						|
        }); | 
						|
 | 
						|
        return result; | 
						|
    }; | 
						|
 | 
						|
    CodeGenerator.prototype.maybeBlockSuffix = function (stmt, result) { | 
						|
        var ends = endsWithLineTerminator(toSourceNodeWhenNeeded(result).toString()); | 
						|
        if (stmt.type === Syntax.BlockStatement && (!extra.comment || !stmt.leadingComments) && !ends) { | 
						|
            return [result, space]; | 
						|
        } | 
						|
        if (ends) { | 
						|
            return [result, base]; | 
						|
        } | 
						|
        return [result, newline, base]; | 
						|
    }; | 
						|
 | 
						|
    function generateIdentifier(node) { | 
						|
        return toSourceNodeWhenNeeded(node.name, node); | 
						|
    } | 
						|
 | 
						|
    function generateAsyncPrefix(node, spaceRequired) { | 
						|
        return node.async ? 'async' + (spaceRequired ? noEmptySpace() : space) : ''; | 
						|
    } | 
						|
 | 
						|
    function generateStarSuffix(node) { | 
						|
        var isGenerator = node.generator && !extra.moz.starlessGenerator; | 
						|
        return isGenerator ? '*' + space : ''; | 
						|
    } | 
						|
 | 
						|
    function generateMethodPrefix(prop) { | 
						|
        var func = prop.value, prefix = ''; | 
						|
        if (func.async) { | 
						|
            prefix += generateAsyncPrefix(func, !prop.computed); | 
						|
        } | 
						|
        if (func.generator) { | 
						|
            // avoid space before method name | 
						|
            prefix += generateStarSuffix(func) ? '*' : ''; | 
						|
        } | 
						|
        return prefix; | 
						|
    } | 
						|
 | 
						|
    CodeGenerator.prototype.generatePattern = function (node, precedence, flags) { | 
						|
        if (node.type === Syntax.Identifier) { | 
						|
            return generateIdentifier(node); | 
						|
        } | 
						|
        return this.generateExpression(node, precedence, flags); | 
						|
    }; | 
						|
 | 
						|
    CodeGenerator.prototype.generateFunctionParams = function (node) { | 
						|
        var i, iz, result, hasDefault; | 
						|
 | 
						|
        hasDefault = false; | 
						|
 | 
						|
        if (node.type === Syntax.ArrowFunctionExpression && | 
						|
                !node.rest && (!node.defaults || node.defaults.length === 0) && | 
						|
                node.params.length === 1 && node.params[0].type === Syntax.Identifier) { | 
						|
            // arg => { } case | 
						|
            result = [generateAsyncPrefix(node, true), generateIdentifier(node.params[0])]; | 
						|
        } else { | 
						|
            result = node.type === Syntax.ArrowFunctionExpression ? [generateAsyncPrefix(node, false)] : []; | 
						|
            result.push('('); | 
						|
            if (node.defaults) { | 
						|
                hasDefault = true; | 
						|
            } | 
						|
            for (i = 0, iz = node.params.length; i < iz; ++i) { | 
						|
                if (hasDefault && node.defaults[i]) { | 
						|
                    // Handle default values. | 
						|
                    result.push(this.generateAssignment(node.params[i], node.defaults[i], '=', Precedence.Assignment, E_TTT)); | 
						|
                } else { | 
						|
                    result.push(this.generatePattern(node.params[i], Precedence.Assignment, E_TTT)); | 
						|
                } | 
						|
                if (i + 1 < iz) { | 
						|
                    result.push(',' + space); | 
						|
                } | 
						|
            } | 
						|
 | 
						|
            if (node.rest) { | 
						|
                if (node.params.length) { | 
						|
                    result.push(',' + space); | 
						|
                } | 
						|
                result.push('...'); | 
						|
                result.push(generateIdentifier(node.rest)); | 
						|
            } | 
						|
 | 
						|
            result.push(')'); | 
						|
        } | 
						|
 | 
						|
        return result; | 
						|
    }; | 
						|
 | 
						|
    CodeGenerator.prototype.generateFunctionBody = function (node) { | 
						|
        var result, expr; | 
						|
 | 
						|
        result = this.generateFunctionParams(node); | 
						|
 | 
						|
        if (node.type === Syntax.ArrowFunctionExpression) { | 
						|
            result.push(space); | 
						|
            result.push('=>'); | 
						|
        } | 
						|
 | 
						|
        if (node.expression) { | 
						|
            result.push(space); | 
						|
            expr = this.generateExpression(node.body, Precedence.Assignment, E_TTT); | 
						|
            if (expr.toString().charAt(0) === '{') { | 
						|
                expr = ['(', expr, ')']; | 
						|
            } | 
						|
            result.push(expr); | 
						|
        } else { | 
						|
            result.push(this.maybeBlock(node.body, S_TTFF)); | 
						|
        } | 
						|
 | 
						|
        return result; | 
						|
    }; | 
						|
 | 
						|
    CodeGenerator.prototype.generateIterationForStatement = function (operator, stmt, flags) { | 
						|
        var result = ['for' + (stmt.await ? noEmptySpace() + 'await' : '') + space + '('], that = this; | 
						|
        withIndent(function () { | 
						|
            if (stmt.left.type === Syntax.VariableDeclaration) { | 
						|
                withIndent(function () { | 
						|
                    result.push(stmt.left.kind + noEmptySpace()); | 
						|
                    result.push(that.generateStatement(stmt.left.declarations[0], S_FFFF)); | 
						|
                }); | 
						|
            } else { | 
						|
                result.push(that.generateExpression(stmt.left, Precedence.Call, E_TTT)); | 
						|
            } | 
						|
 | 
						|
            result = join(result, operator); | 
						|
            result = [join( | 
						|
                result, | 
						|
                that.generateExpression(stmt.right, Precedence.Assignment, E_TTT) | 
						|
            ), ')']; | 
						|
        }); | 
						|
        result.push(this.maybeBlock(stmt.body, flags)); | 
						|
        return result; | 
						|
    }; | 
						|
 | 
						|
    CodeGenerator.prototype.generatePropertyKey = function (expr, computed) { | 
						|
        var result = []; | 
						|
 | 
						|
        if (computed) { | 
						|
            result.push('['); | 
						|
        } | 
						|
 | 
						|
        result.push(this.generateExpression(expr, Precedence.Assignment, E_TTT)); | 
						|
 | 
						|
        if (computed) { | 
						|
            result.push(']'); | 
						|
        } | 
						|
 | 
						|
        return result; | 
						|
    }; | 
						|
 | 
						|
    CodeGenerator.prototype.generateAssignment = function (left, right, operator, precedence, flags) { | 
						|
        if (Precedence.Assignment < precedence) { | 
						|
            flags |= F_ALLOW_IN; | 
						|
        } | 
						|
 | 
						|
        return parenthesize( | 
						|
            [ | 
						|
                this.generateExpression(left, Precedence.Call, flags), | 
						|
                space + operator + space, | 
						|
                this.generateExpression(right, Precedence.Assignment, flags) | 
						|
            ], | 
						|
            Precedence.Assignment, | 
						|
            precedence | 
						|
        ); | 
						|
    }; | 
						|
 | 
						|
    CodeGenerator.prototype.semicolon = function (flags) { | 
						|
        if (!semicolons && flags & F_SEMICOLON_OPT) { | 
						|
            return ''; | 
						|
        } | 
						|
        return ';'; | 
						|
    }; | 
						|
 | 
						|
    // Statements. | 
						|
 | 
						|
    CodeGenerator.Statement = { | 
						|
 | 
						|
        BlockStatement: function (stmt, flags) { | 
						|
            var range, content, result = ['{', newline], that = this; | 
						|
 | 
						|
            withIndent(function () { | 
						|
                // handle functions without any code | 
						|
                if (stmt.body.length === 0 && preserveBlankLines) { | 
						|
                    range = stmt.range; | 
						|
                    if (range[1] - range[0] > 2) { | 
						|
                        content = sourceCode.substring(range[0] + 1, range[1] - 1); | 
						|
                        if (content[0] === '\n') { | 
						|
                            result = ['{']; | 
						|
                        } | 
						|
                        result.push(content); | 
						|
                    } | 
						|
                } | 
						|
 | 
						|
                var i, iz, fragment, bodyFlags; | 
						|
                bodyFlags = S_TFFF; | 
						|
                if (flags & F_FUNC_BODY) { | 
						|
                    bodyFlags |= F_DIRECTIVE_CTX; | 
						|
                } | 
						|
 | 
						|
                for (i = 0, iz = stmt.body.length; i < iz; ++i) { | 
						|
                    if (preserveBlankLines) { | 
						|
                        // handle spaces before the first line | 
						|
                        if (i === 0) { | 
						|
                            if (stmt.body[0].leadingComments) { | 
						|
                                range = stmt.body[0].leadingComments[0].extendedRange; | 
						|
                                content = sourceCode.substring(range[0], range[1]); | 
						|
                                if (content[0] === '\n') { | 
						|
                                    result = ['{']; | 
						|
                                } | 
						|
                            } | 
						|
                            if (!stmt.body[0].leadingComments) { | 
						|
                                generateBlankLines(stmt.range[0], stmt.body[0].range[0], result); | 
						|
                            } | 
						|
                        } | 
						|
 | 
						|
                        // handle spaces between lines | 
						|
                        if (i > 0) { | 
						|
                            if (!stmt.body[i - 1].trailingComments  && !stmt.body[i].leadingComments) { | 
						|
                                generateBlankLines(stmt.body[i - 1].range[1], stmt.body[i].range[0], result); | 
						|
                            } | 
						|
                        } | 
						|
                    } | 
						|
 | 
						|
                    if (i === iz - 1) { | 
						|
                        bodyFlags |= F_SEMICOLON_OPT; | 
						|
                    } | 
						|
 | 
						|
                    if (stmt.body[i].leadingComments && preserveBlankLines) { | 
						|
                        fragment = that.generateStatement(stmt.body[i], bodyFlags); | 
						|
                    } else { | 
						|
                        fragment = addIndent(that.generateStatement(stmt.body[i], bodyFlags)); | 
						|
                    } | 
						|
 | 
						|
                    result.push(fragment); | 
						|
                    if (!endsWithLineTerminator(toSourceNodeWhenNeeded(fragment).toString())) { | 
						|
                        if (preserveBlankLines && i < iz - 1) { | 
						|
                            // don't add a new line if there are leading coments | 
						|
                            // in the next statement | 
						|
                            if (!stmt.body[i + 1].leadingComments) { | 
						|
                                result.push(newline); | 
						|
                            } | 
						|
                        } else { | 
						|
                            result.push(newline); | 
						|
                        } | 
						|
                    } | 
						|
 | 
						|
                    if (preserveBlankLines) { | 
						|
                        // handle spaces after the last line | 
						|
                        if (i === iz - 1) { | 
						|
                            if (!stmt.body[i].trailingComments) { | 
						|
                                generateBlankLines(stmt.body[i].range[1], stmt.range[1], result); | 
						|
                            } | 
						|
                        } | 
						|
                    } | 
						|
                } | 
						|
            }); | 
						|
 | 
						|
            result.push(addIndent('}')); | 
						|
            return result; | 
						|
        }, | 
						|
 | 
						|
        BreakStatement: function (stmt, flags) { | 
						|
            if (stmt.label) { | 
						|
                return 'break ' + stmt.label.name + this.semicolon(flags); | 
						|
            } | 
						|
            return 'break' + this.semicolon(flags); | 
						|
        }, | 
						|
 | 
						|
        ContinueStatement: function (stmt, flags) { | 
						|
            if (stmt.label) { | 
						|
                return 'continue ' + stmt.label.name + this.semicolon(flags); | 
						|
            } | 
						|
            return 'continue' + this.semicolon(flags); | 
						|
        }, | 
						|
 | 
						|
        ClassBody: function (stmt, flags) { | 
						|
            var result = [ '{', newline], that = this; | 
						|
 | 
						|
            withIndent(function (indent) { | 
						|
                var i, iz; | 
						|
 | 
						|
                for (i = 0, iz = stmt.body.length; i < iz; ++i) { | 
						|
                    result.push(indent); | 
						|
                    result.push(that.generateExpression(stmt.body[i], Precedence.Sequence, E_TTT)); | 
						|
                    if (i + 1 < iz) { | 
						|
                        result.push(newline); | 
						|
                    } | 
						|
                } | 
						|
            }); | 
						|
 | 
						|
            if (!endsWithLineTerminator(toSourceNodeWhenNeeded(result).toString())) { | 
						|
                result.push(newline); | 
						|
            } | 
						|
            result.push(base); | 
						|
            result.push('}'); | 
						|
            return result; | 
						|
        }, | 
						|
 | 
						|
        ClassDeclaration: function (stmt, flags) { | 
						|
            var result, fragment; | 
						|
            result  = ['class']; | 
						|
            if (stmt.id) { | 
						|
                result = join(result, this.generateExpression(stmt.id, Precedence.Sequence, E_TTT)); | 
						|
            } | 
						|
            if (stmt.superClass) { | 
						|
                fragment = join('extends', this.generateExpression(stmt.superClass, Precedence.Unary, E_TTT)); | 
						|
                result = join(result, fragment); | 
						|
            } | 
						|
            result.push(space); | 
						|
            result.push(this.generateStatement(stmt.body, S_TFFT)); | 
						|
            return result; | 
						|
        }, | 
						|
 | 
						|
        DirectiveStatement: function (stmt, flags) { | 
						|
            if (extra.raw && stmt.raw) { | 
						|
                return stmt.raw + this.semicolon(flags); | 
						|
            } | 
						|
            return escapeDirective(stmt.directive) + this.semicolon(flags); | 
						|
        }, | 
						|
 | 
						|
        DoWhileStatement: function (stmt, flags) { | 
						|
            // Because `do 42 while (cond)` is Syntax Error. We need semicolon. | 
						|
            var result = join('do', this.maybeBlock(stmt.body, S_TFFF)); | 
						|
            result = this.maybeBlockSuffix(stmt.body, result); | 
						|
            return join(result, [ | 
						|
                'while' + space + '(', | 
						|
                this.generateExpression(stmt.test, Precedence.Sequence, E_TTT), | 
						|
                ')' + this.semicolon(flags) | 
						|
            ]); | 
						|
        }, | 
						|
 | 
						|
        CatchClause: function (stmt, flags) { | 
						|
            var result, that = this; | 
						|
            withIndent(function () { | 
						|
                var guard; | 
						|
 | 
						|
                if (stmt.param) { | 
						|
                    result = [ | 
						|
                        'catch' + space + '(', | 
						|
                        that.generateExpression(stmt.param, Precedence.Sequence, E_TTT), | 
						|
                        ')' | 
						|
                    ]; | 
						|
 | 
						|
                    if (stmt.guard) { | 
						|
                        guard = that.generateExpression(stmt.guard, Precedence.Sequence, E_TTT); | 
						|
                        result.splice(2, 0, ' if ', guard); | 
						|
                    } | 
						|
                } else { | 
						|
                    result = ['catch']; | 
						|
                } | 
						|
            }); | 
						|
            result.push(this.maybeBlock(stmt.body, S_TFFF)); | 
						|
            return result; | 
						|
        }, | 
						|
 | 
						|
        DebuggerStatement: function (stmt, flags) { | 
						|
            return 'debugger' + this.semicolon(flags); | 
						|
        }, | 
						|
 | 
						|
        EmptyStatement: function (stmt, flags) { | 
						|
            return ';'; | 
						|
        }, | 
						|
 | 
						|
        ExportDefaultDeclaration: function (stmt, flags) { | 
						|
            var result = [ 'export' ], bodyFlags; | 
						|
 | 
						|
            bodyFlags = (flags & F_SEMICOLON_OPT) ? S_TFFT : S_TFFF; | 
						|
 | 
						|
            // export default HoistableDeclaration[Default] | 
						|
            // export default AssignmentExpression[In] ; | 
						|
            result = join(result, 'default'); | 
						|
            if (isStatement(stmt.declaration)) { | 
						|
                result = join(result, this.generateStatement(stmt.declaration, bodyFlags)); | 
						|
            } else { | 
						|
                result = join(result, this.generateExpression(stmt.declaration, Precedence.Assignment, E_TTT) + this.semicolon(flags)); | 
						|
            } | 
						|
            return result; | 
						|
        }, | 
						|
 | 
						|
        ExportNamedDeclaration: function (stmt, flags) { | 
						|
            var result = [ 'export' ], bodyFlags, that = this; | 
						|
 | 
						|
            bodyFlags = (flags & F_SEMICOLON_OPT) ? S_TFFT : S_TFFF; | 
						|
 | 
						|
            // export VariableStatement | 
						|
            // export Declaration[Default] | 
						|
            if (stmt.declaration) { | 
						|
                return join(result, this.generateStatement(stmt.declaration, bodyFlags)); | 
						|
            } | 
						|
 | 
						|
            // export ExportClause[NoReference] FromClause ; | 
						|
            // export ExportClause ; | 
						|
            if (stmt.specifiers) { | 
						|
                if (stmt.specifiers.length === 0) { | 
						|
                    result = join(result, '{' + space + '}'); | 
						|
                } else if (stmt.specifiers[0].type === Syntax.ExportBatchSpecifier) { | 
						|
                    result = join(result, this.generateExpression(stmt.specifiers[0], Precedence.Sequence, E_TTT)); | 
						|
                } else { | 
						|
                    result = join(result, '{'); | 
						|
                    withIndent(function (indent) { | 
						|
                        var i, iz; | 
						|
                        result.push(newline); | 
						|
                        for (i = 0, iz = stmt.specifiers.length; i < iz; ++i) { | 
						|
                            result.push(indent); | 
						|
                            result.push(that.generateExpression(stmt.specifiers[i], Precedence.Sequence, E_TTT)); | 
						|
                            if (i + 1 < iz) { | 
						|
                                result.push(',' + newline); | 
						|
                            } | 
						|
                        } | 
						|
                    }); | 
						|
                    if (!endsWithLineTerminator(toSourceNodeWhenNeeded(result).toString())) { | 
						|
                        result.push(newline); | 
						|
                    } | 
						|
                    result.push(base + '}'); | 
						|
                } | 
						|
 | 
						|
                if (stmt.source) { | 
						|
                    result = join(result, [ | 
						|
                        'from' + space, | 
						|
                        // ModuleSpecifier | 
						|
                        this.generateExpression(stmt.source, Precedence.Sequence, E_TTT), | 
						|
                        this.semicolon(flags) | 
						|
                    ]); | 
						|
                } else { | 
						|
                    result.push(this.semicolon(flags)); | 
						|
                } | 
						|
            } | 
						|
            return result; | 
						|
        }, | 
						|
 | 
						|
        ExportAllDeclaration: function (stmt, flags) { | 
						|
            // export * FromClause ; | 
						|
            return [ | 
						|
                'export' + space, | 
						|
                '*' + space, | 
						|
                'from' + space, | 
						|
                // ModuleSpecifier | 
						|
                this.generateExpression(stmt.source, Precedence.Sequence, E_TTT), | 
						|
                this.semicolon(flags) | 
						|
            ]; | 
						|
        }, | 
						|
 | 
						|
        ExpressionStatement: function (stmt, flags) { | 
						|
            var result, fragment; | 
						|
 | 
						|
            function isClassPrefixed(fragment) { | 
						|
                var code; | 
						|
                if (fragment.slice(0, 5) !== 'class') { | 
						|
                    return false; | 
						|
                } | 
						|
                code = fragment.charCodeAt(5); | 
						|
                return code === 0x7B  /* '{' */ || esutils.code.isWhiteSpace(code) || esutils.code.isLineTerminator(code); | 
						|
            } | 
						|
 | 
						|
            function isFunctionPrefixed(fragment) { | 
						|
                var code; | 
						|
                if (fragment.slice(0, 8) !== 'function') { | 
						|
                    return false; | 
						|
                } | 
						|
                code = fragment.charCodeAt(8); | 
						|
                return code === 0x28 /* '(' */ || esutils.code.isWhiteSpace(code) || code === 0x2A  /* '*' */ || esutils.code.isLineTerminator(code); | 
						|
            } | 
						|
 | 
						|
            function isAsyncPrefixed(fragment) { | 
						|
                var code, i, iz; | 
						|
                if (fragment.slice(0, 5) !== 'async') { | 
						|
                    return false; | 
						|
                } | 
						|
                if (!esutils.code.isWhiteSpace(fragment.charCodeAt(5))) { | 
						|
                    return false; | 
						|
                } | 
						|
                for (i = 6, iz = fragment.length; i < iz; ++i) { | 
						|
                    if (!esutils.code.isWhiteSpace(fragment.charCodeAt(i))) { | 
						|
                        break; | 
						|
                    } | 
						|
                } | 
						|
                if (i === iz) { | 
						|
                    return false; | 
						|
                } | 
						|
                if (fragment.slice(i, i + 8) !== 'function') { | 
						|
                    return false; | 
						|
                } | 
						|
                code = fragment.charCodeAt(i + 8); | 
						|
                return code === 0x28 /* '(' */ || esutils.code.isWhiteSpace(code) || code === 0x2A  /* '*' */ || esutils.code.isLineTerminator(code); | 
						|
            } | 
						|
 | 
						|
            result = [this.generateExpression(stmt.expression, Precedence.Sequence, E_TTT)]; | 
						|
            // 12.4 '{', 'function', 'class' is not allowed in this position. | 
						|
            // wrap expression with parentheses | 
						|
            fragment = toSourceNodeWhenNeeded(result).toString(); | 
						|
            if (fragment.charCodeAt(0) === 0x7B  /* '{' */ ||  // ObjectExpression | 
						|
                    isClassPrefixed(fragment) || | 
						|
                    isFunctionPrefixed(fragment) || | 
						|
                    isAsyncPrefixed(fragment) || | 
						|
                    (directive && (flags & F_DIRECTIVE_CTX) && stmt.expression.type === Syntax.Literal && typeof stmt.expression.value === 'string')) { | 
						|
                result = ['(', result, ')' + this.semicolon(flags)]; | 
						|
            } else { | 
						|
                result.push(this.semicolon(flags)); | 
						|
            } | 
						|
            return result; | 
						|
        }, | 
						|
 | 
						|
        ImportDeclaration: function (stmt, flags) { | 
						|
            // ES6: 15.2.1 valid import declarations: | 
						|
            //     - import ImportClause FromClause ; | 
						|
            //     - import ModuleSpecifier ; | 
						|
            var result, cursor, that = this; | 
						|
 | 
						|
            // If no ImportClause is present, | 
						|
            // this should be `import ModuleSpecifier` so skip `from` | 
						|
            // ModuleSpecifier is StringLiteral. | 
						|
            if (stmt.specifiers.length === 0) { | 
						|
                // import ModuleSpecifier ; | 
						|
                return [ | 
						|
                    'import', | 
						|
                    space, | 
						|
                    // ModuleSpecifier | 
						|
                    this.generateExpression(stmt.source, Precedence.Sequence, E_TTT), | 
						|
                    this.semicolon(flags) | 
						|
                ]; | 
						|
            } | 
						|
 | 
						|
            // import ImportClause FromClause ; | 
						|
            result = [ | 
						|
                'import' | 
						|
            ]; | 
						|
            cursor = 0; | 
						|
 | 
						|
            // ImportedBinding | 
						|
            if (stmt.specifiers[cursor].type === Syntax.ImportDefaultSpecifier) { | 
						|
                result = join(result, [ | 
						|
                        this.generateExpression(stmt.specifiers[cursor], Precedence.Sequence, E_TTT) | 
						|
                ]); | 
						|
                ++cursor; | 
						|
            } | 
						|
 | 
						|
            if (stmt.specifiers[cursor]) { | 
						|
                if (cursor !== 0) { | 
						|
                    result.push(','); | 
						|
                } | 
						|
 | 
						|
                if (stmt.specifiers[cursor].type === Syntax.ImportNamespaceSpecifier) { | 
						|
                    // NameSpaceImport | 
						|
                    result = join(result, [ | 
						|
                            space, | 
						|
                            this.generateExpression(stmt.specifiers[cursor], Precedence.Sequence, E_TTT) | 
						|
                    ]); | 
						|
                } else { | 
						|
                    // NamedImports | 
						|
                    result.push(space + '{'); | 
						|
 | 
						|
                    if ((stmt.specifiers.length - cursor) === 1) { | 
						|
                        // import { ... } from "..."; | 
						|
                        result.push(space); | 
						|
                        result.push(this.generateExpression(stmt.specifiers[cursor], Precedence.Sequence, E_TTT)); | 
						|
                        result.push(space + '}' + space); | 
						|
                    } else { | 
						|
                        // import { | 
						|
                        //    ..., | 
						|
                        //    ..., | 
						|
                        // } from "..."; | 
						|
                        withIndent(function (indent) { | 
						|
                            var i, iz; | 
						|
                            result.push(newline); | 
						|
                            for (i = cursor, iz = stmt.specifiers.length; i < iz; ++i) { | 
						|
                                result.push(indent); | 
						|
                                result.push(that.generateExpression(stmt.specifiers[i], Precedence.Sequence, E_TTT)); | 
						|
                                if (i + 1 < iz) { | 
						|
                                    result.push(',' + newline); | 
						|
                                } | 
						|
                            } | 
						|
                        }); | 
						|
                        if (!endsWithLineTerminator(toSourceNodeWhenNeeded(result).toString())) { | 
						|
                            result.push(newline); | 
						|
                        } | 
						|
                        result.push(base + '}' + space); | 
						|
                    } | 
						|
                } | 
						|
            } | 
						|
 | 
						|
            result = join(result, [ | 
						|
                'from' + space, | 
						|
                // ModuleSpecifier | 
						|
                this.generateExpression(stmt.source, Precedence.Sequence, E_TTT), | 
						|
                this.semicolon(flags) | 
						|
            ]); | 
						|
            return result; | 
						|
        }, | 
						|
 | 
						|
        VariableDeclarator: function (stmt, flags) { | 
						|
            var itemFlags = (flags & F_ALLOW_IN) ? E_TTT : E_FTT; | 
						|
            if (stmt.init) { | 
						|
                return [ | 
						|
                    this.generateExpression(stmt.id, Precedence.Assignment, itemFlags), | 
						|
                    space, | 
						|
                    '=', | 
						|
                    space, | 
						|
                    this.generateExpression(stmt.init, Precedence.Assignment, itemFlags) | 
						|
                ]; | 
						|
            } | 
						|
            return this.generatePattern(stmt.id, Precedence.Assignment, itemFlags); | 
						|
        }, | 
						|
 | 
						|
        VariableDeclaration: function (stmt, flags) { | 
						|
            // VariableDeclarator is typed as Statement, | 
						|
            // but joined with comma (not LineTerminator). | 
						|
            // So if comment is attached to target node, we should specialize. | 
						|
            var result, i, iz, node, bodyFlags, that = this; | 
						|
 | 
						|
            result = [ stmt.kind ]; | 
						|
 | 
						|
            bodyFlags = (flags & F_ALLOW_IN) ? S_TFFF : S_FFFF; | 
						|
 | 
						|
            function block() { | 
						|
                node = stmt.declarations[0]; | 
						|
                if (extra.comment && node.leadingComments) { | 
						|
                    result.push('\n'); | 
						|
                    result.push(addIndent(that.generateStatement(node, bodyFlags))); | 
						|
                } else { | 
						|
                    result.push(noEmptySpace()); | 
						|
                    result.push(that.generateStatement(node, bodyFlags)); | 
						|
                } | 
						|
 | 
						|
                for (i = 1, iz = stmt.declarations.length; i < iz; ++i) { | 
						|
                    node = stmt.declarations[i]; | 
						|
                    if (extra.comment && node.leadingComments) { | 
						|
                        result.push(',' + newline); | 
						|
                        result.push(addIndent(that.generateStatement(node, bodyFlags))); | 
						|
                    } else { | 
						|
                        result.push(',' + space); | 
						|
                        result.push(that.generateStatement(node, bodyFlags)); | 
						|
                    } | 
						|
                } | 
						|
            } | 
						|
 | 
						|
            if (stmt.declarations.length > 1) { | 
						|
                withIndent(block); | 
						|
            } else { | 
						|
                block(); | 
						|
            } | 
						|
 | 
						|
            result.push(this.semicolon(flags)); | 
						|
 | 
						|
            return result; | 
						|
        }, | 
						|
 | 
						|
        ThrowStatement: function (stmt, flags) { | 
						|
            return [join( | 
						|
                'throw', | 
						|
                this.generateExpression(stmt.argument, Precedence.Sequence, E_TTT) | 
						|
            ), this.semicolon(flags)]; | 
						|
        }, | 
						|
 | 
						|
        TryStatement: function (stmt, flags) { | 
						|
            var result, i, iz, guardedHandlers; | 
						|
 | 
						|
            result = ['try', this.maybeBlock(stmt.block, S_TFFF)]; | 
						|
            result = this.maybeBlockSuffix(stmt.block, result); | 
						|
 | 
						|
            if (stmt.handlers) { | 
						|
                // old interface | 
						|
                for (i = 0, iz = stmt.handlers.length; i < iz; ++i) { | 
						|
                    result = join(result, this.generateStatement(stmt.handlers[i], S_TFFF)); | 
						|
                    if (stmt.finalizer || i + 1 !== iz) { | 
						|
                        result = this.maybeBlockSuffix(stmt.handlers[i].body, result); | 
						|
                    } | 
						|
                } | 
						|
            } else { | 
						|
                guardedHandlers = stmt.guardedHandlers || []; | 
						|
 | 
						|
                for (i = 0, iz = guardedHandlers.length; i < iz; ++i) { | 
						|
                    result = join(result, this.generateStatement(guardedHandlers[i], S_TFFF)); | 
						|
                    if (stmt.finalizer || i + 1 !== iz) { | 
						|
                        result = this.maybeBlockSuffix(guardedHandlers[i].body, result); | 
						|
                    } | 
						|
                } | 
						|
 | 
						|
                // new interface | 
						|
                if (stmt.handler) { | 
						|
                    if (Array.isArray(stmt.handler)) { | 
						|
                        for (i = 0, iz = stmt.handler.length; i < iz; ++i) { | 
						|
                            result = join(result, this.generateStatement(stmt.handler[i], S_TFFF)); | 
						|
                            if (stmt.finalizer || i + 1 !== iz) { | 
						|
                                result = this.maybeBlockSuffix(stmt.handler[i].body, result); | 
						|
                            } | 
						|
                        } | 
						|
                    } else { | 
						|
                        result = join(result, this.generateStatement(stmt.handler, S_TFFF)); | 
						|
                        if (stmt.finalizer) { | 
						|
                            result = this.maybeBlockSuffix(stmt.handler.body, result); | 
						|
                        } | 
						|
                    } | 
						|
                } | 
						|
            } | 
						|
            if (stmt.finalizer) { | 
						|
                result = join(result, ['finally', this.maybeBlock(stmt.finalizer, S_TFFF)]); | 
						|
            } | 
						|
            return result; | 
						|
        }, | 
						|
 | 
						|
        SwitchStatement: function (stmt, flags) { | 
						|
            var result, fragment, i, iz, bodyFlags, that = this; | 
						|
            withIndent(function () { | 
						|
                result = [ | 
						|
                    'switch' + space + '(', | 
						|
                    that.generateExpression(stmt.discriminant, Precedence.Sequence, E_TTT), | 
						|
                    ')' + space + '{' + newline | 
						|
                ]; | 
						|
            }); | 
						|
            if (stmt.cases) { | 
						|
                bodyFlags = S_TFFF; | 
						|
                for (i = 0, iz = stmt.cases.length; i < iz; ++i) { | 
						|
                    if (i === iz - 1) { | 
						|
                        bodyFlags |= F_SEMICOLON_OPT; | 
						|
                    } | 
						|
                    fragment = addIndent(this.generateStatement(stmt.cases[i], bodyFlags)); | 
						|
                    result.push(fragment); | 
						|
                    if (!endsWithLineTerminator(toSourceNodeWhenNeeded(fragment).toString())) { | 
						|
                        result.push(newline); | 
						|
                    } | 
						|
                } | 
						|
            } | 
						|
            result.push(addIndent('}')); | 
						|
            return result; | 
						|
        }, | 
						|
 | 
						|
        SwitchCase: function (stmt, flags) { | 
						|
            var result, fragment, i, iz, bodyFlags, that = this; | 
						|
            withIndent(function () { | 
						|
                if (stmt.test) { | 
						|
                    result = [ | 
						|
                        join('case', that.generateExpression(stmt.test, Precedence.Sequence, E_TTT)), | 
						|
                        ':' | 
						|
                    ]; | 
						|
                } else { | 
						|
                    result = ['default:']; | 
						|
                } | 
						|
 | 
						|
                i = 0; | 
						|
                iz = stmt.consequent.length; | 
						|
                if (iz && stmt.consequent[0].type === Syntax.BlockStatement) { | 
						|
                    fragment = that.maybeBlock(stmt.consequent[0], S_TFFF); | 
						|
                    result.push(fragment); | 
						|
                    i = 1; | 
						|
                } | 
						|
 | 
						|
                if (i !== iz && !endsWithLineTerminator(toSourceNodeWhenNeeded(result).toString())) { | 
						|
                    result.push(newline); | 
						|
                } | 
						|
 | 
						|
                bodyFlags = S_TFFF; | 
						|
                for (; i < iz; ++i) { | 
						|
                    if (i === iz - 1 && flags & F_SEMICOLON_OPT) { | 
						|
                        bodyFlags |= F_SEMICOLON_OPT; | 
						|
                    } | 
						|
                    fragment = addIndent(that.generateStatement(stmt.consequent[i], bodyFlags)); | 
						|
                    result.push(fragment); | 
						|
                    if (i + 1 !== iz && !endsWithLineTerminator(toSourceNodeWhenNeeded(fragment).toString())) { | 
						|
                        result.push(newline); | 
						|
                    } | 
						|
                } | 
						|
            }); | 
						|
            return result; | 
						|
        }, | 
						|
 | 
						|
        IfStatement: function (stmt, flags) { | 
						|
            var result, bodyFlags, semicolonOptional, that = this; | 
						|
            withIndent(function () { | 
						|
                result = [ | 
						|
                    'if' + space + '(', | 
						|
                    that.generateExpression(stmt.test, Precedence.Sequence, E_TTT), | 
						|
                    ')' | 
						|
                ]; | 
						|
            }); | 
						|
            semicolonOptional = flags & F_SEMICOLON_OPT; | 
						|
            bodyFlags = S_TFFF; | 
						|
            if (semicolonOptional) { | 
						|
                bodyFlags |= F_SEMICOLON_OPT; | 
						|
            } | 
						|
            if (stmt.alternate) { | 
						|
                result.push(this.maybeBlock(stmt.consequent, S_TFFF)); | 
						|
                result = this.maybeBlockSuffix(stmt.consequent, result); | 
						|
                if (stmt.alternate.type === Syntax.IfStatement) { | 
						|
                    result = join(result, ['else ', this.generateStatement(stmt.alternate, bodyFlags)]); | 
						|
                } else { | 
						|
                    result = join(result, join('else', this.maybeBlock(stmt.alternate, bodyFlags))); | 
						|
                } | 
						|
            } else { | 
						|
                result.push(this.maybeBlock(stmt.consequent, bodyFlags)); | 
						|
            } | 
						|
            return result; | 
						|
        }, | 
						|
 | 
						|
        ForStatement: function (stmt, flags) { | 
						|
            var result, that = this; | 
						|
            withIndent(function () { | 
						|
                result = ['for' + space + '(']; | 
						|
                if (stmt.init) { | 
						|
                    if (stmt.init.type === Syntax.VariableDeclaration) { | 
						|
                        result.push(that.generateStatement(stmt.init, S_FFFF)); | 
						|
                    } else { | 
						|
                        // F_ALLOW_IN becomes false. | 
						|
                        result.push(that.generateExpression(stmt.init, Precedence.Sequence, E_FTT)); | 
						|
                        result.push(';'); | 
						|
                    } | 
						|
                } else { | 
						|
                    result.push(';'); | 
						|
                } | 
						|
 | 
						|
                if (stmt.test) { | 
						|
                    result.push(space); | 
						|
                    result.push(that.generateExpression(stmt.test, Precedence.Sequence, E_TTT)); | 
						|
                    result.push(';'); | 
						|
                } else { | 
						|
                    result.push(';'); | 
						|
                } | 
						|
 | 
						|
                if (stmt.update) { | 
						|
                    result.push(space); | 
						|
                    result.push(that.generateExpression(stmt.update, Precedence.Sequence, E_TTT)); | 
						|
                    result.push(')'); | 
						|
                } else { | 
						|
                    result.push(')'); | 
						|
                } | 
						|
            }); | 
						|
 | 
						|
            result.push(this.maybeBlock(stmt.body, flags & F_SEMICOLON_OPT ? S_TFFT : S_TFFF)); | 
						|
            return result; | 
						|
        }, | 
						|
 | 
						|
        ForInStatement: function (stmt, flags) { | 
						|
            return this.generateIterationForStatement('in', stmt, flags & F_SEMICOLON_OPT ? S_TFFT : S_TFFF); | 
						|
        }, | 
						|
 | 
						|
        ForOfStatement: function (stmt, flags) { | 
						|
            return this.generateIterationForStatement('of', stmt, flags & F_SEMICOLON_OPT ? S_TFFT : S_TFFF); | 
						|
        }, | 
						|
 | 
						|
        LabeledStatement: function (stmt, flags) { | 
						|
            return [stmt.label.name + ':', this.maybeBlock(stmt.body, flags & F_SEMICOLON_OPT ? S_TFFT : S_TFFF)]; | 
						|
        }, | 
						|
 | 
						|
        Program: function (stmt, flags) { | 
						|
            var result, fragment, i, iz, bodyFlags; | 
						|
            iz = stmt.body.length; | 
						|
            result = [safeConcatenation && iz > 0 ? '\n' : '']; | 
						|
            bodyFlags = S_TFTF; | 
						|
            for (i = 0; i < iz; ++i) { | 
						|
                if (!safeConcatenation && i === iz - 1) { | 
						|
                    bodyFlags |= F_SEMICOLON_OPT; | 
						|
                } | 
						|
 | 
						|
                if (preserveBlankLines) { | 
						|
                    // handle spaces before the first line | 
						|
                    if (i === 0) { | 
						|
                        if (!stmt.body[0].leadingComments) { | 
						|
                            generateBlankLines(stmt.range[0], stmt.body[i].range[0], result); | 
						|
                        } | 
						|
                    } | 
						|
 | 
						|
                    // handle spaces between lines | 
						|
                    if (i > 0) { | 
						|
                        if (!stmt.body[i - 1].trailingComments && !stmt.body[i].leadingComments) { | 
						|
                            generateBlankLines(stmt.body[i - 1].range[1], stmt.body[i].range[0], result); | 
						|
                        } | 
						|
                    } | 
						|
                } | 
						|
 | 
						|
                fragment = addIndent(this.generateStatement(stmt.body[i], bodyFlags)); | 
						|
                result.push(fragment); | 
						|
                if (i + 1 < iz && !endsWithLineTerminator(toSourceNodeWhenNeeded(fragment).toString())) { | 
						|
                    if (preserveBlankLines) { | 
						|
                        if (!stmt.body[i + 1].leadingComments) { | 
						|
                            result.push(newline); | 
						|
                        } | 
						|
                    } else { | 
						|
                        result.push(newline); | 
						|
                    } | 
						|
                } | 
						|
 | 
						|
                if (preserveBlankLines) { | 
						|
                    // handle spaces after the last line | 
						|
                    if (i === iz - 1) { | 
						|
                        if (!stmt.body[i].trailingComments) { | 
						|
                            generateBlankLines(stmt.body[i].range[1], stmt.range[1], result); | 
						|
                        } | 
						|
                    } | 
						|
                } | 
						|
            } | 
						|
            return result; | 
						|
        }, | 
						|
 | 
						|
        FunctionDeclaration: function (stmt, flags) { | 
						|
            return [ | 
						|
                generateAsyncPrefix(stmt, true), | 
						|
                'function', | 
						|
                generateStarSuffix(stmt) || noEmptySpace(), | 
						|
                stmt.id ? generateIdentifier(stmt.id) : '', | 
						|
                this.generateFunctionBody(stmt) | 
						|
            ]; | 
						|
        }, | 
						|
 | 
						|
        ReturnStatement: function (stmt, flags) { | 
						|
            if (stmt.argument) { | 
						|
                return [join( | 
						|
                    'return', | 
						|
                    this.generateExpression(stmt.argument, Precedence.Sequence, E_TTT) | 
						|
                ), this.semicolon(flags)]; | 
						|
            } | 
						|
            return ['return' + this.semicolon(flags)]; | 
						|
        }, | 
						|
 | 
						|
        WhileStatement: function (stmt, flags) { | 
						|
            var result, that = this; | 
						|
            withIndent(function () { | 
						|
                result = [ | 
						|
                    'while' + space + '(', | 
						|
                    that.generateExpression(stmt.test, Precedence.Sequence, E_TTT), | 
						|
                    ')' | 
						|
                ]; | 
						|
            }); | 
						|
            result.push(this.maybeBlock(stmt.body, flags & F_SEMICOLON_OPT ? S_TFFT : S_TFFF)); | 
						|
            return result; | 
						|
        }, | 
						|
 | 
						|
        WithStatement: function (stmt, flags) { | 
						|
            var result, that = this; | 
						|
            withIndent(function () { | 
						|
                result = [ | 
						|
                    'with' + space + '(', | 
						|
                    that.generateExpression(stmt.object, Precedence.Sequence, E_TTT), | 
						|
                    ')' | 
						|
                ]; | 
						|
            }); | 
						|
            result.push(this.maybeBlock(stmt.body, flags & F_SEMICOLON_OPT ? S_TFFT : S_TFFF)); | 
						|
            return result; | 
						|
        } | 
						|
 | 
						|
    }; | 
						|
 | 
						|
    merge(CodeGenerator.prototype, CodeGenerator.Statement); | 
						|
 | 
						|
    // Expressions. | 
						|
 | 
						|
    CodeGenerator.Expression = { | 
						|
 | 
						|
        SequenceExpression: function (expr, precedence, flags) { | 
						|
            var result, i, iz; | 
						|
            if (Precedence.Sequence < precedence) { | 
						|
                flags |= F_ALLOW_IN; | 
						|
            } | 
						|
            result = []; | 
						|
            for (i = 0, iz = expr.expressions.length; i < iz; ++i) { | 
						|
                result.push(this.generateExpression(expr.expressions[i], Precedence.Assignment, flags)); | 
						|
                if (i + 1 < iz) { | 
						|
                    result.push(',' + space); | 
						|
                } | 
						|
            } | 
						|
            return parenthesize(result, Precedence.Sequence, precedence); | 
						|
        }, | 
						|
 | 
						|
        AssignmentExpression: function (expr, precedence, flags) { | 
						|
            return this.generateAssignment(expr.left, expr.right, expr.operator, precedence, flags); | 
						|
        }, | 
						|
 | 
						|
        ArrowFunctionExpression: function (expr, precedence, flags) { | 
						|
            return parenthesize(this.generateFunctionBody(expr), Precedence.ArrowFunction, precedence); | 
						|
        }, | 
						|
 | 
						|
        ConditionalExpression: function (expr, precedence, flags) { | 
						|
            if (Precedence.Conditional < precedence) { | 
						|
                flags |= F_ALLOW_IN; | 
						|
            } | 
						|
            return parenthesize( | 
						|
                [ | 
						|
                    this.generateExpression(expr.test, Precedence.LogicalOR, flags), | 
						|
                    space + '?' + space, | 
						|
                    this.generateExpression(expr.consequent, Precedence.Assignment, flags), | 
						|
                    space + ':' + space, | 
						|
                    this.generateExpression(expr.alternate, Precedence.Assignment, flags) | 
						|
                ], | 
						|
                Precedence.Conditional, | 
						|
                precedence | 
						|
            ); | 
						|
        }, | 
						|
 | 
						|
        LogicalExpression: function (expr, precedence, flags) { | 
						|
            return this.BinaryExpression(expr, precedence, flags); | 
						|
        }, | 
						|
 | 
						|
        BinaryExpression: function (expr, precedence, flags) { | 
						|
            var result, leftPrecedence, rightPrecedence, currentPrecedence, fragment, leftSource; | 
						|
            currentPrecedence = BinaryPrecedence[expr.operator]; | 
						|
            leftPrecedence = expr.operator === '**' ? Precedence.Postfix : currentPrecedence; | 
						|
            rightPrecedence = expr.operator === '**' ? currentPrecedence : currentPrecedence + 1; | 
						|
 | 
						|
            if (currentPrecedence < precedence) { | 
						|
                flags |= F_ALLOW_IN; | 
						|
            } | 
						|
 | 
						|
            fragment = this.generateExpression(expr.left, leftPrecedence, flags); | 
						|
 | 
						|
            leftSource = fragment.toString(); | 
						|
 | 
						|
            if (leftSource.charCodeAt(leftSource.length - 1) === 0x2F /* / */ && esutils.code.isIdentifierPartES5(expr.operator.charCodeAt(0))) { | 
						|
                result = [fragment, noEmptySpace(), expr.operator]; | 
						|
            } else { | 
						|
                result = join(fragment, expr.operator); | 
						|
            } | 
						|
 | 
						|
            fragment = this.generateExpression(expr.right, rightPrecedence, flags); | 
						|
 | 
						|
            if (expr.operator === '/' && fragment.toString().charAt(0) === '/' || | 
						|
            expr.operator.slice(-1) === '<' && fragment.toString().slice(0, 3) === '!--') { | 
						|
                // If '/' concats with '/' or `<` concats with `!--`, it is interpreted as comment start | 
						|
                result.push(noEmptySpace()); | 
						|
                result.push(fragment); | 
						|
            } else { | 
						|
                result = join(result, fragment); | 
						|
            } | 
						|
 | 
						|
            if (expr.operator === 'in' && !(flags & F_ALLOW_IN)) { | 
						|
                return ['(', result, ')']; | 
						|
            } | 
						|
            return parenthesize(result, currentPrecedence, precedence); | 
						|
        }, | 
						|
 | 
						|
        CallExpression: function (expr, precedence, flags) { | 
						|
            var result, i, iz; | 
						|
            // F_ALLOW_UNPARATH_NEW becomes false. | 
						|
            result = [this.generateExpression(expr.callee, Precedence.Call, E_TTF)]; | 
						|
            result.push('('); | 
						|
            for (i = 0, iz = expr['arguments'].length; i < iz; ++i) { | 
						|
                result.push(this.generateExpression(expr['arguments'][i], Precedence.Assignment, E_TTT)); | 
						|
                if (i + 1 < iz) { | 
						|
                    result.push(',' + space); | 
						|
                } | 
						|
            } | 
						|
            result.push(')'); | 
						|
 | 
						|
            if (!(flags & F_ALLOW_CALL)) { | 
						|
                return ['(', result, ')']; | 
						|
            } | 
						|
            return parenthesize(result, Precedence.Call, precedence); | 
						|
        }, | 
						|
 | 
						|
        NewExpression: function (expr, precedence, flags) { | 
						|
            var result, length, i, iz, itemFlags; | 
						|
            length = expr['arguments'].length; | 
						|
 | 
						|
            // F_ALLOW_CALL becomes false. | 
						|
            // F_ALLOW_UNPARATH_NEW may become false. | 
						|
            itemFlags = (flags & F_ALLOW_UNPARATH_NEW && !parentheses && length === 0) ? E_TFT : E_TFF; | 
						|
 | 
						|
            result = join( | 
						|
                'new', | 
						|
                this.generateExpression(expr.callee, Precedence.New, itemFlags) | 
						|
            ); | 
						|
 | 
						|
            if (!(flags & F_ALLOW_UNPARATH_NEW) || parentheses || length > 0) { | 
						|
                result.push('('); | 
						|
                for (i = 0, iz = length; i < iz; ++i) { | 
						|
                    result.push(this.generateExpression(expr['arguments'][i], Precedence.Assignment, E_TTT)); | 
						|
                    if (i + 1 < iz) { | 
						|
                        result.push(',' + space); | 
						|
                    } | 
						|
                } | 
						|
                result.push(')'); | 
						|
            } | 
						|
 | 
						|
            return parenthesize(result, Precedence.New, precedence); | 
						|
        }, | 
						|
 | 
						|
        MemberExpression: function (expr, precedence, flags) { | 
						|
            var result, fragment; | 
						|
 | 
						|
            // F_ALLOW_UNPARATH_NEW becomes false. | 
						|
            result = [this.generateExpression(expr.object, Precedence.Call, (flags & F_ALLOW_CALL) ? E_TTF : E_TFF)]; | 
						|
 | 
						|
            if (expr.computed) { | 
						|
                result.push('['); | 
						|
                result.push(this.generateExpression(expr.property, Precedence.Sequence, flags & F_ALLOW_CALL ? E_TTT : E_TFT)); | 
						|
                result.push(']'); | 
						|
            } else { | 
						|
                if (expr.object.type === Syntax.Literal && typeof expr.object.value === 'number') { | 
						|
                    fragment = toSourceNodeWhenNeeded(result).toString(); | 
						|
                    // When the following conditions are all true, | 
						|
                    //   1. No floating point | 
						|
                    //   2. Don't have exponents | 
						|
                    //   3. The last character is a decimal digit | 
						|
                    //   4. Not hexadecimal OR octal number literal | 
						|
                    // we should add a floating point. | 
						|
                    if ( | 
						|
                            fragment.indexOf('.') < 0 && | 
						|
                            !/[eExX]/.test(fragment) && | 
						|
                            esutils.code.isDecimalDigit(fragment.charCodeAt(fragment.length - 1)) && | 
						|
                            !(fragment.length >= 2 && fragment.charCodeAt(0) === 48)  // '0' | 
						|
                            ) { | 
						|
                        result.push(' '); | 
						|
                    } | 
						|
                } | 
						|
                result.push('.'); | 
						|
                result.push(generateIdentifier(expr.property)); | 
						|
            } | 
						|
 | 
						|
            return parenthesize(result, Precedence.Member, precedence); | 
						|
        }, | 
						|
 | 
						|
        MetaProperty: function (expr, precedence, flags) { | 
						|
            var result; | 
						|
            result = []; | 
						|
            result.push(typeof expr.meta === "string" ? expr.meta : generateIdentifier(expr.meta)); | 
						|
            result.push('.'); | 
						|
            result.push(typeof expr.property === "string" ? expr.property : generateIdentifier(expr.property)); | 
						|
            return parenthesize(result, Precedence.Member, precedence); | 
						|
        }, | 
						|
 | 
						|
        UnaryExpression: function (expr, precedence, flags) { | 
						|
            var result, fragment, rightCharCode, leftSource, leftCharCode; | 
						|
            fragment = this.generateExpression(expr.argument, Precedence.Unary, E_TTT); | 
						|
 | 
						|
            if (space === '') { | 
						|
                result = join(expr.operator, fragment); | 
						|
            } else { | 
						|
                result = [expr.operator]; | 
						|
                if (expr.operator.length > 2) { | 
						|
                    // delete, void, typeof | 
						|
                    // get `typeof []`, not `typeof[]` | 
						|
                    result = join(result, fragment); | 
						|
                } else { | 
						|
                    // Prevent inserting spaces between operator and argument if it is unnecessary | 
						|
                    // like, `!cond` | 
						|
                    leftSource = toSourceNodeWhenNeeded(result).toString(); | 
						|
                    leftCharCode = leftSource.charCodeAt(leftSource.length - 1); | 
						|
                    rightCharCode = fragment.toString().charCodeAt(0); | 
						|
 | 
						|
                    if (((leftCharCode === 0x2B  /* + */ || leftCharCode === 0x2D  /* - */) && leftCharCode === rightCharCode) || | 
						|
                            (esutils.code.isIdentifierPartES5(leftCharCode) && esutils.code.isIdentifierPartES5(rightCharCode))) { | 
						|
                        result.push(noEmptySpace()); | 
						|
                        result.push(fragment); | 
						|
                    } else { | 
						|
                        result.push(fragment); | 
						|
                    } | 
						|
                } | 
						|
            } | 
						|
            return parenthesize(result, Precedence.Unary, precedence); | 
						|
        }, | 
						|
 | 
						|
        YieldExpression: function (expr, precedence, flags) { | 
						|
            var result; | 
						|
            if (expr.delegate) { | 
						|
                result = 'yield*'; | 
						|
            } else { | 
						|
                result = 'yield'; | 
						|
            } | 
						|
            if (expr.argument) { | 
						|
                result = join( | 
						|
                    result, | 
						|
                    this.generateExpression(expr.argument, Precedence.Yield, E_TTT) | 
						|
                ); | 
						|
            } | 
						|
            return parenthesize(result, Precedence.Yield, precedence); | 
						|
        }, | 
						|
 | 
						|
        AwaitExpression: function (expr, precedence, flags) { | 
						|
            var result = join( | 
						|
                expr.all ? 'await*' : 'await', | 
						|
                this.generateExpression(expr.argument, Precedence.Await, E_TTT) | 
						|
            ); | 
						|
            return parenthesize(result, Precedence.Await, precedence); | 
						|
        }, | 
						|
 | 
						|
        UpdateExpression: function (expr, precedence, flags) { | 
						|
            if (expr.prefix) { | 
						|
                return parenthesize( | 
						|
                    [ | 
						|
                        expr.operator, | 
						|
                        this.generateExpression(expr.argument, Precedence.Unary, E_TTT) | 
						|
                    ], | 
						|
                    Precedence.Unary, | 
						|
                    precedence | 
						|
                ); | 
						|
            } | 
						|
            return parenthesize( | 
						|
                [ | 
						|
                    this.generateExpression(expr.argument, Precedence.Postfix, E_TTT), | 
						|
                    expr.operator | 
						|
                ], | 
						|
                Precedence.Postfix, | 
						|
                precedence | 
						|
            ); | 
						|
        }, | 
						|
 | 
						|
        FunctionExpression: function (expr, precedence, flags) { | 
						|
            var result = [ | 
						|
                generateAsyncPrefix(expr, true), | 
						|
                'function' | 
						|
            ]; | 
						|
            if (expr.id) { | 
						|
                result.push(generateStarSuffix(expr) || noEmptySpace()); | 
						|
                result.push(generateIdentifier(expr.id)); | 
						|
            } else { | 
						|
                result.push(generateStarSuffix(expr) || space); | 
						|
            } | 
						|
            result.push(this.generateFunctionBody(expr)); | 
						|
            return result; | 
						|
        }, | 
						|
 | 
						|
        ArrayPattern: function (expr, precedence, flags) { | 
						|
            return this.ArrayExpression(expr, precedence, flags, true); | 
						|
        }, | 
						|
 | 
						|
        ArrayExpression: function (expr, precedence, flags, isPattern) { | 
						|
            var result, multiline, that = this; | 
						|
            if (!expr.elements.length) { | 
						|
                return '[]'; | 
						|
            } | 
						|
            multiline = isPattern ? false : expr.elements.length > 1; | 
						|
            result = ['[', multiline ? newline : '']; | 
						|
            withIndent(function (indent) { | 
						|
                var i, iz; | 
						|
                for (i = 0, iz = expr.elements.length; i < iz; ++i) { | 
						|
                    if (!expr.elements[i]) { | 
						|
                        if (multiline) { | 
						|
                            result.push(indent); | 
						|
                        } | 
						|
                        if (i + 1 === iz) { | 
						|
                            result.push(','); | 
						|
                        } | 
						|
                    } else { | 
						|
                        result.push(multiline ? indent : ''); | 
						|
                        result.push(that.generateExpression(expr.elements[i], Precedence.Assignment, E_TTT)); | 
						|
                    } | 
						|
                    if (i + 1 < iz) { | 
						|
                        result.push(',' + (multiline ? newline : space)); | 
						|
                    } | 
						|
                } | 
						|
            }); | 
						|
            if (multiline && !endsWithLineTerminator(toSourceNodeWhenNeeded(result).toString())) { | 
						|
                result.push(newline); | 
						|
            } | 
						|
            result.push(multiline ? base : ''); | 
						|
            result.push(']'); | 
						|
            return result; | 
						|
        }, | 
						|
 | 
						|
        RestElement: function(expr, precedence, flags) { | 
						|
            return '...' + this.generatePattern(expr.argument); | 
						|
        }, | 
						|
 | 
						|
        ClassExpression: function (expr, precedence, flags) { | 
						|
            var result, fragment; | 
						|
            result = ['class']; | 
						|
            if (expr.id) { | 
						|
                result = join(result, this.generateExpression(expr.id, Precedence.Sequence, E_TTT)); | 
						|
            } | 
						|
            if (expr.superClass) { | 
						|
                fragment = join('extends', this.generateExpression(expr.superClass, Precedence.Unary, E_TTT)); | 
						|
                result = join(result, fragment); | 
						|
            } | 
						|
            result.push(space); | 
						|
            result.push(this.generateStatement(expr.body, S_TFFT)); | 
						|
            return result; | 
						|
        }, | 
						|
 | 
						|
        MethodDefinition: function (expr, precedence, flags) { | 
						|
            var result, fragment; | 
						|
            if (expr['static']) { | 
						|
                result = ['static' + space]; | 
						|
            } else { | 
						|
                result = []; | 
						|
            } | 
						|
            if (expr.kind === 'get' || expr.kind === 'set') { | 
						|
                fragment = [ | 
						|
                    join(expr.kind, this.generatePropertyKey(expr.key, expr.computed)), | 
						|
                    this.generateFunctionBody(expr.value) | 
						|
                ]; | 
						|
            } else { | 
						|
                fragment = [ | 
						|
                    generateMethodPrefix(expr), | 
						|
                    this.generatePropertyKey(expr.key, expr.computed), | 
						|
                    this.generateFunctionBody(expr.value) | 
						|
                ]; | 
						|
            } | 
						|
            return join(result, fragment); | 
						|
        }, | 
						|
 | 
						|
        Property: function (expr, precedence, flags) { | 
						|
            if (expr.kind === 'get' || expr.kind === 'set') { | 
						|
                return [ | 
						|
                    expr.kind, noEmptySpace(), | 
						|
                    this.generatePropertyKey(expr.key, expr.computed), | 
						|
                    this.generateFunctionBody(expr.value) | 
						|
                ]; | 
						|
            } | 
						|
 | 
						|
            if (expr.shorthand) { | 
						|
                if (expr.value.type === "AssignmentPattern") { | 
						|
                    return this.AssignmentPattern(expr.value, Precedence.Sequence, E_TTT); | 
						|
                } | 
						|
                return this.generatePropertyKey(expr.key, expr.computed); | 
						|
            } | 
						|
 | 
						|
            if (expr.method) { | 
						|
                return [ | 
						|
                    generateMethodPrefix(expr), | 
						|
                    this.generatePropertyKey(expr.key, expr.computed), | 
						|
                    this.generateFunctionBody(expr.value) | 
						|
                ]; | 
						|
            } | 
						|
 | 
						|
            return [ | 
						|
                this.generatePropertyKey(expr.key, expr.computed), | 
						|
                ':' + space, | 
						|
                this.generateExpression(expr.value, Precedence.Assignment, E_TTT) | 
						|
            ]; | 
						|
        }, | 
						|
 | 
						|
        ObjectExpression: function (expr, precedence, flags) { | 
						|
            var multiline, result, fragment, that = this; | 
						|
 | 
						|
            if (!expr.properties.length) { | 
						|
                return '{}'; | 
						|
            } | 
						|
            multiline = expr.properties.length > 1; | 
						|
 | 
						|
            withIndent(function () { | 
						|
                fragment = that.generateExpression(expr.properties[0], Precedence.Sequence, E_TTT); | 
						|
            }); | 
						|
 | 
						|
            if (!multiline) { | 
						|
                // issues 4 | 
						|
                // Do not transform from | 
						|
                //   dejavu.Class.declare({ | 
						|
                //       method2: function () {} | 
						|
                //   }); | 
						|
                // to | 
						|
                //   dejavu.Class.declare({method2: function () { | 
						|
                //       }}); | 
						|
                if (!hasLineTerminator(toSourceNodeWhenNeeded(fragment).toString())) { | 
						|
                    return [ '{', space, fragment, space, '}' ]; | 
						|
                } | 
						|
            } | 
						|
 | 
						|
            withIndent(function (indent) { | 
						|
                var i, iz; | 
						|
                result = [ '{', newline, indent, fragment ]; | 
						|
 | 
						|
                if (multiline) { | 
						|
                    result.push(',' + newline); | 
						|
                    for (i = 1, iz = expr.properties.length; i < iz; ++i) { | 
						|
                        result.push(indent); | 
						|
                        result.push(that.generateExpression(expr.properties[i], Precedence.Sequence, E_TTT)); | 
						|
                        if (i + 1 < iz) { | 
						|
                            result.push(',' + newline); | 
						|
                        } | 
						|
                    } | 
						|
                } | 
						|
            }); | 
						|
 | 
						|
            if (!endsWithLineTerminator(toSourceNodeWhenNeeded(result).toString())) { | 
						|
                result.push(newline); | 
						|
            } | 
						|
            result.push(base); | 
						|
            result.push('}'); | 
						|
            return result; | 
						|
        }, | 
						|
 | 
						|
        AssignmentPattern: function(expr, precedence, flags) { | 
						|
            return this.generateAssignment(expr.left, expr.right, '=', precedence, flags); | 
						|
        }, | 
						|
 | 
						|
        ObjectPattern: function (expr, precedence, flags) { | 
						|
            var result, i, iz, multiline, property, that = this; | 
						|
            if (!expr.properties.length) { | 
						|
                return '{}'; | 
						|
            } | 
						|
 | 
						|
            multiline = false; | 
						|
            if (expr.properties.length === 1) { | 
						|
                property = expr.properties[0]; | 
						|
                if ( | 
						|
                    property.type === Syntax.Property | 
						|
                    && property.value.type !== Syntax.Identifier | 
						|
                ) { | 
						|
                    multiline = true; | 
						|
                } | 
						|
            } else { | 
						|
                for (i = 0, iz = expr.properties.length; i < iz; ++i) { | 
						|
                    property = expr.properties[i]; | 
						|
                    if ( | 
						|
                        property.type === Syntax.Property | 
						|
                        && !property.shorthand | 
						|
                    ) { | 
						|
                        multiline = true; | 
						|
                        break; | 
						|
                    } | 
						|
                } | 
						|
            } | 
						|
            result = ['{', multiline ? newline : '' ]; | 
						|
 | 
						|
            withIndent(function (indent) { | 
						|
                var i, iz; | 
						|
                for (i = 0, iz = expr.properties.length; i < iz; ++i) { | 
						|
                    result.push(multiline ? indent : ''); | 
						|
                    result.push(that.generateExpression(expr.properties[i], Precedence.Sequence, E_TTT)); | 
						|
                    if (i + 1 < iz) { | 
						|
                        result.push(',' + (multiline ? newline : space)); | 
						|
                    } | 
						|
                } | 
						|
            }); | 
						|
 | 
						|
            if (multiline && !endsWithLineTerminator(toSourceNodeWhenNeeded(result).toString())) { | 
						|
                result.push(newline); | 
						|
            } | 
						|
            result.push(multiline ? base : ''); | 
						|
            result.push('}'); | 
						|
            return result; | 
						|
        }, | 
						|
 | 
						|
        ThisExpression: function (expr, precedence, flags) { | 
						|
            return 'this'; | 
						|
        }, | 
						|
 | 
						|
        Super: function (expr, precedence, flags) { | 
						|
            return 'super'; | 
						|
        }, | 
						|
 | 
						|
        Identifier: function (expr, precedence, flags) { | 
						|
            return generateIdentifier(expr); | 
						|
        }, | 
						|
 | 
						|
        ImportDefaultSpecifier: function (expr, precedence, flags) { | 
						|
            return generateIdentifier(expr.id || expr.local); | 
						|
        }, | 
						|
 | 
						|
        ImportNamespaceSpecifier: function (expr, precedence, flags) { | 
						|
            var result = ['*']; | 
						|
            var id = expr.id || expr.local; | 
						|
            if (id) { | 
						|
                result.push(space + 'as' + noEmptySpace() + generateIdentifier(id)); | 
						|
            } | 
						|
            return result; | 
						|
        }, | 
						|
 | 
						|
        ImportSpecifier: function (expr, precedence, flags) { | 
						|
            var imported = expr.imported; | 
						|
            var result = [ imported.name ]; | 
						|
            var local = expr.local; | 
						|
            if (local && local.name !== imported.name) { | 
						|
                result.push(noEmptySpace() + 'as' + noEmptySpace() + generateIdentifier(local)); | 
						|
            } | 
						|
            return result; | 
						|
        }, | 
						|
 | 
						|
        ExportSpecifier: function (expr, precedence, flags) { | 
						|
            var local = expr.local; | 
						|
            var result = [ local.name ]; | 
						|
            var exported = expr.exported; | 
						|
            if (exported && exported.name !== local.name) { | 
						|
                result.push(noEmptySpace() + 'as' + noEmptySpace() + generateIdentifier(exported)); | 
						|
            } | 
						|
            return result; | 
						|
        }, | 
						|
 | 
						|
        Literal: function (expr, precedence, flags) { | 
						|
            var raw; | 
						|
            if (expr.hasOwnProperty('raw') && parse && extra.raw) { | 
						|
                try { | 
						|
                    raw = parse(expr.raw).body[0].expression; | 
						|
                    if (raw.type === Syntax.Literal) { | 
						|
                        if (raw.value === expr.value) { | 
						|
                            return expr.raw; | 
						|
                        } | 
						|
                    } | 
						|
                } catch (e) { | 
						|
                    // not use raw property | 
						|
                } | 
						|
            } | 
						|
 | 
						|
            if (expr.regex) { | 
						|
              return '/' + expr.regex.pattern + '/' + expr.regex.flags; | 
						|
            } | 
						|
 | 
						|
            if (expr.value === null) { | 
						|
                return 'null'; | 
						|
            } | 
						|
 | 
						|
            if (typeof expr.value === 'string') { | 
						|
                return escapeString(expr.value); | 
						|
            } | 
						|
 | 
						|
            if (typeof expr.value === 'number') { | 
						|
                return generateNumber(expr.value); | 
						|
            } | 
						|
 | 
						|
            if (typeof expr.value === 'boolean') { | 
						|
                return expr.value ? 'true' : 'false'; | 
						|
            } | 
						|
 | 
						|
            return generateRegExp(expr.value); | 
						|
        }, | 
						|
 | 
						|
        GeneratorExpression: function (expr, precedence, flags) { | 
						|
            return this.ComprehensionExpression(expr, precedence, flags); | 
						|
        }, | 
						|
 | 
						|
        ComprehensionExpression: function (expr, precedence, flags) { | 
						|
            // GeneratorExpression should be parenthesized with (...), ComprehensionExpression with [...] | 
						|
            // Due to https://bugzilla.mozilla.org/show_bug.cgi?id=883468 position of expr.body can differ in Spidermonkey and ES6 | 
						|
 | 
						|
            var result, i, iz, fragment, that = this; | 
						|
            result = (expr.type === Syntax.GeneratorExpression) ? ['('] : ['[']; | 
						|
 | 
						|
            if (extra.moz.comprehensionExpressionStartsWithAssignment) { | 
						|
                fragment = this.generateExpression(expr.body, Precedence.Assignment, E_TTT); | 
						|
                result.push(fragment); | 
						|
            } | 
						|
 | 
						|
            if (expr.blocks) { | 
						|
                withIndent(function () { | 
						|
                    for (i = 0, iz = expr.blocks.length; i < iz; ++i) { | 
						|
                        fragment = that.generateExpression(expr.blocks[i], Precedence.Sequence, E_TTT); | 
						|
                        if (i > 0 || extra.moz.comprehensionExpressionStartsWithAssignment) { | 
						|
                            result = join(result, fragment); | 
						|
                        } else { | 
						|
                            result.push(fragment); | 
						|
                        } | 
						|
                    } | 
						|
                }); | 
						|
            } | 
						|
 | 
						|
            if (expr.filter) { | 
						|
                result = join(result, 'if' + space); | 
						|
                fragment = this.generateExpression(expr.filter, Precedence.Sequence, E_TTT); | 
						|
                result = join(result, [ '(', fragment, ')' ]); | 
						|
            } | 
						|
 | 
						|
            if (!extra.moz.comprehensionExpressionStartsWithAssignment) { | 
						|
                fragment = this.generateExpression(expr.body, Precedence.Assignment, E_TTT); | 
						|
 | 
						|
                result = join(result, fragment); | 
						|
            } | 
						|
 | 
						|
            result.push((expr.type === Syntax.GeneratorExpression) ? ')' : ']'); | 
						|
            return result; | 
						|
        }, | 
						|
 | 
						|
        ComprehensionBlock: function (expr, precedence, flags) { | 
						|
            var fragment; | 
						|
            if (expr.left.type === Syntax.VariableDeclaration) { | 
						|
                fragment = [ | 
						|
                    expr.left.kind, noEmptySpace(), | 
						|
                    this.generateStatement(expr.left.declarations[0], S_FFFF) | 
						|
                ]; | 
						|
            } else { | 
						|
                fragment = this.generateExpression(expr.left, Precedence.Call, E_TTT); | 
						|
            } | 
						|
 | 
						|
            fragment = join(fragment, expr.of ? 'of' : 'in'); | 
						|
            fragment = join(fragment, this.generateExpression(expr.right, Precedence.Sequence, E_TTT)); | 
						|
 | 
						|
            return [ 'for' + space + '(', fragment, ')' ]; | 
						|
        }, | 
						|
 | 
						|
        SpreadElement: function (expr, precedence, flags) { | 
						|
            return [ | 
						|
                '...', | 
						|
                this.generateExpression(expr.argument, Precedence.Assignment, E_TTT) | 
						|
            ]; | 
						|
        }, | 
						|
 | 
						|
        TaggedTemplateExpression: function (expr, precedence, flags) { | 
						|
            var itemFlags = E_TTF; | 
						|
            if (!(flags & F_ALLOW_CALL)) { | 
						|
                itemFlags = E_TFF; | 
						|
            } | 
						|
            var result = [ | 
						|
                this.generateExpression(expr.tag, Precedence.Call, itemFlags), | 
						|
                this.generateExpression(expr.quasi, Precedence.Primary, E_FFT) | 
						|
            ]; | 
						|
            return parenthesize(result, Precedence.TaggedTemplate, precedence); | 
						|
        }, | 
						|
 | 
						|
        TemplateElement: function (expr, precedence, flags) { | 
						|
            // Don't use "cooked". Since tagged template can use raw template | 
						|
            // representation. So if we do so, it breaks the script semantics. | 
						|
            return expr.value.raw; | 
						|
        }, | 
						|
 | 
						|
        TemplateLiteral: function (expr, precedence, flags) { | 
						|
            var result, i, iz; | 
						|
            result = [ '`' ]; | 
						|
            for (i = 0, iz = expr.quasis.length; i < iz; ++i) { | 
						|
                result.push(this.generateExpression(expr.quasis[i], Precedence.Primary, E_TTT)); | 
						|
                if (i + 1 < iz) { | 
						|
                    result.push('${' + space); | 
						|
                    result.push(this.generateExpression(expr.expressions[i], Precedence.Sequence, E_TTT)); | 
						|
                    result.push(space + '}'); | 
						|
                } | 
						|
            } | 
						|
            result.push('`'); | 
						|
            return result; | 
						|
        }, | 
						|
 | 
						|
        ModuleSpecifier: function (expr, precedence, flags) { | 
						|
            return this.Literal(expr, precedence, flags); | 
						|
        }, | 
						|
 | 
						|
        ImportExpression: function(expr, precedence, flag) { | 
						|
            return parenthesize([ | 
						|
                'import(', | 
						|
                this.generateExpression(expr.source, Precedence.Assignment, E_TTT), | 
						|
                ')' | 
						|
            ], Precedence.Call, precedence); | 
						|
        }, | 
						|
 | 
						|
    }; | 
						|
 | 
						|
    merge(CodeGenerator.prototype, CodeGenerator.Expression); | 
						|
 | 
						|
    CodeGenerator.prototype.generateExpression = function (expr, precedence, flags) { | 
						|
        var result, type; | 
						|
 | 
						|
        type = expr.type || Syntax.Property; | 
						|
 | 
						|
        if (extra.verbatim && expr.hasOwnProperty(extra.verbatim)) { | 
						|
            return generateVerbatim(expr, precedence); | 
						|
        } | 
						|
 | 
						|
        result = this[type](expr, precedence, flags); | 
						|
 | 
						|
 | 
						|
        if (extra.comment) { | 
						|
            result = addComments(expr, result); | 
						|
        } | 
						|
        return toSourceNodeWhenNeeded(result, expr); | 
						|
    }; | 
						|
 | 
						|
    CodeGenerator.prototype.generateStatement = function (stmt, flags) { | 
						|
        var result, | 
						|
            fragment; | 
						|
 | 
						|
        result = this[stmt.type](stmt, flags); | 
						|
 | 
						|
        // Attach comments | 
						|
 | 
						|
        if (extra.comment) { | 
						|
            result = addComments(stmt, result); | 
						|
        } | 
						|
 | 
						|
        fragment = toSourceNodeWhenNeeded(result).toString(); | 
						|
        if (stmt.type === Syntax.Program && !safeConcatenation && newline === '' &&  fragment.charAt(fragment.length - 1) === '\n') { | 
						|
            result = sourceMap ? toSourceNodeWhenNeeded(result).replaceRight(/\s+$/, '') : fragment.replace(/\s+$/, ''); | 
						|
        } | 
						|
 | 
						|
        return toSourceNodeWhenNeeded(result, stmt); | 
						|
    }; | 
						|
 | 
						|
    function generateInternal(node) { | 
						|
        var codegen; | 
						|
 | 
						|
        codegen = new CodeGenerator(); | 
						|
        if (isStatement(node)) { | 
						|
            return codegen.generateStatement(node, S_TFFF); | 
						|
        } | 
						|
 | 
						|
        if (isExpression(node)) { | 
						|
            return codegen.generateExpression(node, Precedence.Sequence, E_TTT); | 
						|
        } | 
						|
 | 
						|
        throw new Error('Unknown node type: ' + node.type); | 
						|
    } | 
						|
 | 
						|
    function generate(node, options) { | 
						|
        var defaultOptions = getDefaultOptions(), result, pair; | 
						|
 | 
						|
        if (options != null) { | 
						|
            // Obsolete options | 
						|
            // | 
						|
            //   `options.indent` | 
						|
            //   `options.base` | 
						|
            // | 
						|
            // Instead of them, we can use `option.format.indent`. | 
						|
            if (typeof options.indent === 'string') { | 
						|
                defaultOptions.format.indent.style = options.indent; | 
						|
            } | 
						|
            if (typeof options.base === 'number') { | 
						|
                defaultOptions.format.indent.base = options.base; | 
						|
            } | 
						|
            options = updateDeeply(defaultOptions, options); | 
						|
            indent = options.format.indent.style; | 
						|
            if (typeof options.base === 'string') { | 
						|
                base = options.base; | 
						|
            } else { | 
						|
                base = stringRepeat(indent, options.format.indent.base); | 
						|
            } | 
						|
        } else { | 
						|
            options = defaultOptions; | 
						|
            indent = options.format.indent.style; | 
						|
            base = stringRepeat(indent, options.format.indent.base); | 
						|
        } | 
						|
        json = options.format.json; | 
						|
        renumber = options.format.renumber; | 
						|
        hexadecimal = json ? false : options.format.hexadecimal; | 
						|
        quotes = json ? 'double' : options.format.quotes; | 
						|
        escapeless = options.format.escapeless; | 
						|
        newline = options.format.newline; | 
						|
        space = options.format.space; | 
						|
        if (options.format.compact) { | 
						|
            newline = space = indent = base = ''; | 
						|
        } | 
						|
        parentheses = options.format.parentheses; | 
						|
        semicolons = options.format.semicolons; | 
						|
        safeConcatenation = options.format.safeConcatenation; | 
						|
        directive = options.directive; | 
						|
        parse = json ? null : options.parse; | 
						|
        sourceMap = options.sourceMap; | 
						|
        sourceCode = options.sourceCode; | 
						|
        preserveBlankLines = options.format.preserveBlankLines && sourceCode !== null; | 
						|
        extra = options; | 
						|
 | 
						|
        if (sourceMap) { | 
						|
            if (!exports.browser) { | 
						|
                // We assume environment is node.js | 
						|
                // And prevent from including source-map by browserify | 
						|
                SourceNode = require('source-map').SourceNode; | 
						|
            } else { | 
						|
                SourceNode = global.sourceMap.SourceNode; | 
						|
            } | 
						|
        } | 
						|
 | 
						|
        result = generateInternal(node); | 
						|
 | 
						|
        if (!sourceMap) { | 
						|
            pair = {code: result.toString(), map: null}; | 
						|
            return options.sourceMapWithCode ? pair : pair.code; | 
						|
        } | 
						|
 | 
						|
 | 
						|
        pair = result.toStringWithSourceMap({ | 
						|
            file: options.file, | 
						|
            sourceRoot: options.sourceMapRoot | 
						|
        }); | 
						|
 | 
						|
        if (options.sourceContent) { | 
						|
            pair.map.setSourceContent(options.sourceMap, | 
						|
                                      options.sourceContent); | 
						|
        } | 
						|
 | 
						|
        if (options.sourceMapWithCode) { | 
						|
            return pair; | 
						|
        } | 
						|
 | 
						|
        return pair.map.toString(); | 
						|
    } | 
						|
 | 
						|
    FORMAT_MINIFY = { | 
						|
        indent: { | 
						|
            style: '', | 
						|
            base: 0 | 
						|
        }, | 
						|
        renumber: true, | 
						|
        hexadecimal: true, | 
						|
        quotes: 'auto', | 
						|
        escapeless: true, | 
						|
        compact: true, | 
						|
        parentheses: false, | 
						|
        semicolons: false | 
						|
    }; | 
						|
 | 
						|
    FORMAT_DEFAULTS = getDefaultOptions().format; | 
						|
 | 
						|
    exports.version = require('./package.json').version; | 
						|
    exports.generate = generate; | 
						|
    exports.attachComments = estraverse.attachComments; | 
						|
    exports.Precedence = updateDeeply({}, Precedence); | 
						|
    exports.browser = false; | 
						|
    exports.FORMAT_MINIFY = FORMAT_MINIFY; | 
						|
    exports.FORMAT_DEFAULTS = FORMAT_DEFAULTS; | 
						|
}()); | 
						|
/* vim: set sw=4 ts=4 et tw=80 : */
 | 
						|
 |