From b97bf86bb6a029625393b45fcb4021975e88f3e6 Mon Sep 17 00:00:00 2001 From: Rich-Harris Date: Sun, 4 Oct 2015 09:30:57 -0400 Subject: [PATCH] unbound default exports --- src/Bundle.js | 6 ++++++ src/Module.js | 36 +++++++++++++++++++++++++----------- src/finalisers/es6.js | 10 ---------- 3 files changed, 31 insertions(+), 21 deletions(-) diff --git a/src/Bundle.js b/src/Bundle.js index 4255728..a447ea0 100644 --- a/src/Bundle.js +++ b/src/Bundle.js @@ -99,6 +99,12 @@ export default class Bundle { this.modules.forEach( module => { keys( module.declarations ).forEach( originalName => { const declaration = module.declarations[ originalName ]; + + if ( originalName === 'default' ) { + const defaultExport = module.exports.default; + if ( defaultExport.identifier && !declaration.original.isReassigned ) return; + } + declaration.name = getSafeName( declaration.name ); }); }); diff --git a/src/Module.js b/src/Module.js index 571d2cf..539f7b5 100644 --- a/src/Module.js +++ b/src/Module.js @@ -24,8 +24,14 @@ class SyntheticDefaultDeclaration { this.name = reference.name; } + bind ( declaration ) { + this.original = declaration; + } + render () { - return this.name; + return !this.original || this.original.isReassigned ? + this.name : + this.original.render(); } } @@ -120,10 +126,8 @@ export default class Module { isModified: false // in case of `export default foo; foo = somethingElse` }; - if ( !identifier ) { - // create a synthetic declaration - this.declarations.default = new SyntheticDefaultDeclaration( node, statement, this.defaultName() ); - } + // create a synthetic declaration + this.declarations.default = new SyntheticDefaultDeclaration( node, statement, this.defaultName() ); } // export { foo, bar, baz } @@ -231,6 +235,13 @@ export default class Module { } bindReferences () { + if ( this.declarations.default ) { + if ( this.exports.default.identifier ) { + const declaration = this.trace( this.exports.default.identifier ); + if ( declaration ) this.declarations.default.bind( declaration ); + } + } + this.statements.forEach( statement => { statement.references.forEach( reference => { const declaration = reference.scope.findDeclaration( reference.name ) || @@ -484,8 +495,16 @@ export default class Module { } else if ( statement.node.type === 'ExportDefaultDeclaration' ) { + // TODO unify these const defaultDeclaration = this.declarations.default; - const defaultName = defaultDeclaration.name; + + // prevent `var foo = foo` + if ( defaultDeclaration.original && !defaultDeclaration.original.isReassigned ) { + magicString.remove( statement.start, statement.next ); + return; + } + + const defaultName = defaultDeclaration.render(); // prevent `var undefined = sideEffectyDefault(foo)` if ( !defaultDeclaration.isExported && !defaultDeclaration.references.length ) { @@ -530,12 +549,7 @@ export default class Module { const exportDeclaration = this.exports[ name ]; if ( exportDeclaration ) { - // TODO defaults should not be live... if ( name === 'default' ) { - if ( exportDeclaration.identifier ) { - return this.trace( exportDeclaration.identifier ); - } - return this.declarations.default; } diff --git a/src/finalisers/es6.js b/src/finalisers/es6.js index f0031e8..6a0053c 100644 --- a/src/finalisers/es6.js +++ b/src/finalisers/es6.js @@ -1,15 +1,5 @@ import { blank, keys } from '../utils/object'; -function uniqueNames ( declarations ) { - let uniques = blank(); - - declarations - .filter( declaration => !/^(default|\*)$/.test( declaration.name ) ) - .forEach( declaration => uniques[ declaration.name ] = true ); - - return keys( uniques ); -} - function notDefault ( name ) { return name !== 'default'; }