1273 lines
34 KiB
Go
1273 lines
34 KiB
Go
// Package js minifies ECMAScript 2021 following the language specification at https://tc39.es/ecma262/.
|
|
package js
|
|
|
|
import (
|
|
"bytes"
|
|
"io"
|
|
|
|
"github.com/tdewolff/minify/v2"
|
|
"github.com/tdewolff/parse/v2"
|
|
"github.com/tdewolff/parse/v2/js"
|
|
)
|
|
|
|
type blockType int
|
|
|
|
const (
|
|
defaultBlock blockType = iota
|
|
functionBlock
|
|
iterationBlock
|
|
)
|
|
|
|
// Minifier is a JS minifier.
|
|
type Minifier struct {
|
|
Precision int // number of significant digits
|
|
KeepVarNames bool
|
|
useAlphabetVarNames bool
|
|
NoNullishOperator bool
|
|
}
|
|
|
|
// Minify minifies JS data, it reads from r and writes to w.
|
|
func Minify(m *minify.M, w io.Writer, r io.Reader, params map[string]string) error {
|
|
return (&Minifier{}).Minify(m, w, r, params)
|
|
}
|
|
|
|
// Minify minifies JS data, it reads from r and writes to w.
|
|
func (o *Minifier) Minify(_ *minify.M, w io.Writer, r io.Reader, _ map[string]string) error {
|
|
z := parse.NewInput(r)
|
|
ast, err := js.Parse(z, js.Options{WhileToFor: true})
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// license comments
|
|
for _, comment := range ast.Comments {
|
|
if 3 < len(comment) && comment[2] == '!' {
|
|
w.Write(comment)
|
|
if comment[1] == '/' {
|
|
w.Write(newlineBytes)
|
|
}
|
|
} else if 2 < len(comment) && comment[0] == '#' && comment[1] == '!' {
|
|
w.Write(comment)
|
|
}
|
|
}
|
|
|
|
m := &jsMinifier{
|
|
o: o,
|
|
w: w,
|
|
renamer: newRenamer(!o.KeepVarNames, !o.useAlphabetVarNames),
|
|
}
|
|
m.hoistVars(&ast.BlockStmt)
|
|
ast.List = optimizeStmtList(ast.List, functionBlock)
|
|
for _, item := range ast.List {
|
|
m.writeSemicolon()
|
|
m.minifyStmt(item)
|
|
}
|
|
|
|
if _, err := w.Write(nil); err != nil {
|
|
return err
|
|
}
|
|
return nil
|
|
}
|
|
|
|
type expectExpr int
|
|
|
|
const (
|
|
expectAny expectExpr = iota
|
|
expectExprStmt // in statement
|
|
expectExprBody // in arrow function body
|
|
)
|
|
|
|
type jsMinifier struct {
|
|
o *Minifier
|
|
w io.Writer
|
|
|
|
prev []byte
|
|
needsSemicolon bool // write a semicolon if required
|
|
needsSpace bool // write a space if next token is an identifier
|
|
expectExpr expectExpr // avoid ambiguous syntax such as an expression starting with function
|
|
groupedStmt bool // avoid ambiguous syntax by grouping the expression statement
|
|
inFor bool
|
|
spaceBefore byte
|
|
|
|
renamer *renamer
|
|
}
|
|
|
|
func (m *jsMinifier) write(b []byte) {
|
|
// 0 < len(b)
|
|
if m.needsSpace && js.IsIdentifierContinue(b) || m.spaceBefore == b[0] {
|
|
m.w.Write(spaceBytes)
|
|
}
|
|
m.w.Write(b)
|
|
m.prev = b
|
|
m.needsSpace = false
|
|
m.expectExpr = expectAny
|
|
m.spaceBefore = 0
|
|
}
|
|
|
|
func (m *jsMinifier) writeSpaceAfterIdent() {
|
|
// space after identifier and after regular expression (to prevent confusion with its tag)
|
|
if js.IsIdentifierEnd(m.prev) || 1 < len(m.prev) && m.prev[0] == '/' {
|
|
m.w.Write(spaceBytes)
|
|
}
|
|
}
|
|
|
|
func (m *jsMinifier) writeSpaceBeforeIdent() {
|
|
m.needsSpace = true
|
|
}
|
|
|
|
func (m *jsMinifier) writeSpaceBefore(c byte) {
|
|
m.spaceBefore = c
|
|
}
|
|
|
|
func (m *jsMinifier) requireSemicolon() {
|
|
m.needsSemicolon = true
|
|
}
|
|
|
|
func (m *jsMinifier) writeSemicolon() {
|
|
if m.needsSemicolon {
|
|
m.w.Write(semicolonBytes)
|
|
m.needsSemicolon = false
|
|
m.needsSpace = false
|
|
}
|
|
}
|
|
|
|
func (m *jsMinifier) minifyStmt(i js.IStmt) {
|
|
switch stmt := i.(type) {
|
|
case *js.ExprStmt:
|
|
m.expectExpr = expectExprStmt
|
|
m.minifyExpr(stmt.Value, js.OpExpr)
|
|
if m.groupedStmt {
|
|
m.write(closeParenBytes)
|
|
m.groupedStmt = false
|
|
}
|
|
m.requireSemicolon()
|
|
case *js.VarDecl:
|
|
m.minifyVarDecl(stmt, false)
|
|
m.requireSemicolon()
|
|
case *js.IfStmt:
|
|
hasIf := !isEmptyStmt(stmt.Body)
|
|
hasElse := !isEmptyStmt(stmt.Else)
|
|
if !hasIf && !hasElse {
|
|
break
|
|
}
|
|
|
|
m.write(ifOpenBytes)
|
|
m.minifyExpr(stmt.Cond, js.OpExpr)
|
|
m.write(closeParenBytes)
|
|
|
|
if !hasIf && hasElse {
|
|
m.requireSemicolon()
|
|
} else if hasIf {
|
|
if hasElse && endsInIf(stmt.Body) {
|
|
// prevent: if(a){if(b)c}else d; => if(a)if(b)c;else d;
|
|
m.write(openBraceBytes)
|
|
m.minifyStmt(stmt.Body)
|
|
m.write(closeBraceBytes)
|
|
m.needsSemicolon = false
|
|
} else {
|
|
m.minifyStmt(stmt.Body)
|
|
}
|
|
}
|
|
if hasElse {
|
|
m.writeSemicolon()
|
|
m.write(elseBytes)
|
|
m.writeSpaceBeforeIdent()
|
|
m.minifyStmt(stmt.Else)
|
|
}
|
|
case *js.BlockStmt:
|
|
m.renamer.renameScope(stmt.Scope)
|
|
m.minifyBlockStmt(stmt)
|
|
case *js.ReturnStmt:
|
|
m.write(returnBytes)
|
|
m.writeSpaceBeforeIdent()
|
|
m.minifyExpr(stmt.Value, js.OpExpr)
|
|
m.requireSemicolon()
|
|
case *js.LabelledStmt:
|
|
m.write(stmt.Label)
|
|
m.write(colonBytes)
|
|
m.minifyStmtOrBlock(stmt.Value, defaultBlock)
|
|
case *js.BranchStmt:
|
|
m.write(stmt.Type.Bytes())
|
|
if stmt.Label != nil {
|
|
m.write(spaceBytes)
|
|
m.write(stmt.Label)
|
|
}
|
|
m.requireSemicolon()
|
|
case *js.WithStmt:
|
|
m.write(withOpenBytes)
|
|
m.minifyExpr(stmt.Cond, js.OpExpr)
|
|
m.write(closeParenBytes)
|
|
m.minifyStmtOrBlock(stmt.Body, defaultBlock)
|
|
case *js.DoWhileStmt:
|
|
m.write(doBytes)
|
|
m.writeSpaceBeforeIdent()
|
|
m.minifyStmtOrBlock(stmt.Body, iterationBlock)
|
|
m.writeSemicolon()
|
|
m.write(whileOpenBytes)
|
|
m.minifyExpr(stmt.Cond, js.OpExpr)
|
|
m.write(closeParenBytes)
|
|
case *js.WhileStmt:
|
|
m.write(whileOpenBytes)
|
|
m.minifyExpr(stmt.Cond, js.OpExpr)
|
|
m.write(closeParenBytes)
|
|
m.minifyStmtOrBlock(stmt.Body, iterationBlock)
|
|
case *js.ForStmt:
|
|
stmt.Body.List = optimizeStmtList(stmt.Body.List, iterationBlock)
|
|
m.renamer.renameScope(stmt.Body.Scope)
|
|
m.write(forOpenBytes)
|
|
m.inFor = true
|
|
if decl, ok := stmt.Init.(*js.VarDecl); ok {
|
|
m.minifyVarDecl(decl, true)
|
|
} else {
|
|
m.minifyExpr(stmt.Init, js.OpLHS)
|
|
}
|
|
m.inFor = false
|
|
m.write(semicolonBytes)
|
|
m.minifyExpr(stmt.Cond, js.OpExpr)
|
|
m.write(semicolonBytes)
|
|
m.minifyExpr(stmt.Post, js.OpExpr)
|
|
m.write(closeParenBytes)
|
|
m.minifyBlockAsStmt(stmt.Body)
|
|
case *js.ForInStmt:
|
|
stmt.Body.List = optimizeStmtList(stmt.Body.List, iterationBlock)
|
|
m.renamer.renameScope(stmt.Body.Scope)
|
|
m.write(forOpenBytes)
|
|
m.inFor = true
|
|
if decl, ok := stmt.Init.(*js.VarDecl); ok {
|
|
m.minifyVarDecl(decl, false)
|
|
} else {
|
|
m.minifyExpr(stmt.Init, js.OpLHS)
|
|
}
|
|
m.inFor = false
|
|
m.writeSpaceAfterIdent()
|
|
m.write(inBytes)
|
|
m.writeSpaceBeforeIdent()
|
|
m.minifyExpr(stmt.Value, js.OpExpr)
|
|
m.write(closeParenBytes)
|
|
m.minifyBlockAsStmt(stmt.Body)
|
|
case *js.ForOfStmt:
|
|
stmt.Body.List = optimizeStmtList(stmt.Body.List, iterationBlock)
|
|
m.renamer.renameScope(stmt.Body.Scope)
|
|
if stmt.Await {
|
|
m.write(forAwaitOpenBytes)
|
|
} else {
|
|
m.write(forOpenBytes)
|
|
}
|
|
m.inFor = true
|
|
if decl, ok := stmt.Init.(*js.VarDecl); ok {
|
|
m.minifyVarDecl(decl, false)
|
|
} else {
|
|
m.minifyExpr(stmt.Init, js.OpLHS)
|
|
}
|
|
m.inFor = false
|
|
m.writeSpaceAfterIdent()
|
|
m.write(ofBytes)
|
|
m.writeSpaceBeforeIdent()
|
|
m.minifyExpr(stmt.Value, js.OpAssign)
|
|
m.write(closeParenBytes)
|
|
m.minifyBlockAsStmt(stmt.Body)
|
|
case *js.SwitchStmt:
|
|
m.write(switchOpenBytes)
|
|
m.minifyExpr(stmt.Init, js.OpExpr)
|
|
m.write(closeParenOpenBracketBytes)
|
|
m.needsSemicolon = false
|
|
for i, _ := range stmt.List {
|
|
stmt.List[i].List = optimizeStmtList(stmt.List[i].List, defaultBlock)
|
|
}
|
|
m.renamer.renameScope(stmt.Scope)
|
|
for _, clause := range stmt.List {
|
|
m.writeSemicolon()
|
|
m.write(clause.TokenType.Bytes())
|
|
if clause.Cond != nil {
|
|
m.writeSpaceBeforeIdent()
|
|
m.minifyExpr(clause.Cond, js.OpExpr)
|
|
}
|
|
m.write(colonBytes)
|
|
for _, item := range clause.List {
|
|
m.writeSemicolon()
|
|
m.minifyStmt(item)
|
|
}
|
|
}
|
|
m.write(closeBraceBytes)
|
|
m.needsSemicolon = false
|
|
case *js.ThrowStmt:
|
|
m.write(throwBytes)
|
|
m.writeSpaceBeforeIdent()
|
|
m.minifyExpr(stmt.Value, js.OpExpr)
|
|
m.requireSemicolon()
|
|
case *js.TryStmt:
|
|
m.write(tryBytes)
|
|
stmt.Body.List = optimizeStmtList(stmt.Body.List, defaultBlock)
|
|
m.renamer.renameScope(stmt.Body.Scope)
|
|
m.minifyBlockStmt(stmt.Body)
|
|
if stmt.Catch != nil {
|
|
m.write(catchBytes)
|
|
stmt.Catch.List = optimizeStmtList(stmt.Catch.List, defaultBlock)
|
|
if v, ok := stmt.Binding.(*js.Var); ok && v.Uses == 1 {
|
|
stmt.Catch.Scope.Declared = stmt.Catch.Scope.Declared[1:]
|
|
stmt.Binding = nil
|
|
}
|
|
m.renamer.renameScope(stmt.Catch.Scope)
|
|
if stmt.Binding != nil {
|
|
m.write(openParenBytes)
|
|
m.minifyBinding(stmt.Binding)
|
|
m.write(closeParenBytes)
|
|
}
|
|
m.minifyBlockStmt(stmt.Catch)
|
|
}
|
|
if stmt.Finally != nil {
|
|
m.write(finallyBytes)
|
|
stmt.Finally.List = optimizeStmtList(stmt.Finally.List, defaultBlock)
|
|
m.renamer.renameScope(stmt.Finally.Scope)
|
|
m.minifyBlockStmt(stmt.Finally)
|
|
}
|
|
case *js.FuncDecl:
|
|
m.minifyFuncDecl(stmt, false)
|
|
case *js.ClassDecl:
|
|
m.minifyClassDecl(stmt)
|
|
case *js.DebuggerStmt:
|
|
m.write(debuggerBytes)
|
|
m.requireSemicolon()
|
|
case *js.EmptyStmt:
|
|
case *js.ImportStmt:
|
|
m.write(importBytes)
|
|
if stmt.Default != nil {
|
|
m.write(spaceBytes)
|
|
m.write(stmt.Default)
|
|
if len(stmt.List) != 0 {
|
|
m.write(commaBytes)
|
|
} else if stmt.Default != nil || len(stmt.List) != 0 {
|
|
m.write(spaceBytes)
|
|
}
|
|
}
|
|
if len(stmt.List) == 1 && len(stmt.List[0].Name) == 1 && stmt.List[0].Name[0] == '*' {
|
|
m.writeSpaceBeforeIdent()
|
|
m.minifyAlias(stmt.List[0])
|
|
if stmt.Default != nil || len(stmt.List) != 0 {
|
|
m.write(spaceBytes)
|
|
}
|
|
} else if 0 < len(stmt.List) {
|
|
m.write(openBraceBytes)
|
|
for i, item := range stmt.List {
|
|
if i != 0 {
|
|
m.write(commaBytes)
|
|
}
|
|
m.minifyAlias(item)
|
|
}
|
|
m.write(closeBraceBytes)
|
|
}
|
|
if stmt.Default != nil || len(stmt.List) != 0 {
|
|
m.write(fromBytes)
|
|
}
|
|
m.write(minifyString(stmt.Module, false))
|
|
m.requireSemicolon()
|
|
case *js.ExportStmt:
|
|
m.write(exportBytes)
|
|
if stmt.Decl != nil {
|
|
if stmt.Default {
|
|
m.write(spaceDefaultBytes)
|
|
m.writeSpaceBeforeIdent()
|
|
m.minifyExpr(stmt.Decl, js.OpAssign)
|
|
_, isHoistable := stmt.Decl.(*js.FuncDecl)
|
|
_, isClass := stmt.Decl.(*js.ClassDecl)
|
|
if !isHoistable && !isClass {
|
|
m.requireSemicolon()
|
|
}
|
|
} else {
|
|
m.writeSpaceBeforeIdent()
|
|
m.minifyStmt(stmt.Decl.(js.IStmt)) // can only be variable, function, or class decl
|
|
}
|
|
} else {
|
|
if len(stmt.List) == 1 && (len(stmt.List[0].Name) == 1 && stmt.List[0].Name[0] == '*' || stmt.List[0].Name == nil && len(stmt.List[0].Binding) == 1 && stmt.List[0].Binding[0] == '*') {
|
|
m.writeSpaceBeforeIdent()
|
|
m.minifyAlias(stmt.List[0])
|
|
if stmt.Module != nil && stmt.List[0].Name != nil {
|
|
m.write(spaceBytes)
|
|
}
|
|
} else if 0 < len(stmt.List) {
|
|
m.write(openBraceBytes)
|
|
for i, item := range stmt.List {
|
|
if i != 0 {
|
|
m.write(commaBytes)
|
|
}
|
|
m.minifyAlias(item)
|
|
}
|
|
m.write(closeBraceBytes)
|
|
}
|
|
if stmt.Module != nil {
|
|
m.write(fromBytes)
|
|
m.write(minifyString(stmt.Module, false))
|
|
}
|
|
m.requireSemicolon()
|
|
}
|
|
case *js.DirectivePrologueStmt:
|
|
stmt.Value[0] = '"'
|
|
stmt.Value[len(stmt.Value)-1] = '"'
|
|
m.write(stmt.Value)
|
|
m.requireSemicolon()
|
|
}
|
|
}
|
|
|
|
func (m *jsMinifier) minifyBlockStmt(stmt *js.BlockStmt) {
|
|
m.write(openBraceBytes)
|
|
m.needsSemicolon = false
|
|
for _, item := range stmt.List {
|
|
m.writeSemicolon()
|
|
m.minifyStmt(item)
|
|
}
|
|
m.write(closeBraceBytes)
|
|
m.needsSemicolon = false
|
|
}
|
|
|
|
func (m *jsMinifier) minifyBlockAsStmt(blockStmt *js.BlockStmt) {
|
|
// minify block when statement is expected, i.e. semicolon if empty or remove braces for single statement
|
|
// assume we already renamed the scope
|
|
hasLexicalVars := false
|
|
for _, v := range blockStmt.Scope.Declared[blockStmt.Scope.NumForDecls:] {
|
|
if v.Decl == js.LexicalDecl {
|
|
hasLexicalVars = true
|
|
break
|
|
}
|
|
}
|
|
if 1 < len(blockStmt.List) || hasLexicalVars {
|
|
m.minifyBlockStmt(blockStmt)
|
|
} else if len(blockStmt.List) == 1 {
|
|
m.minifyStmt(blockStmt.List[0])
|
|
} else {
|
|
m.write(semicolonBytes)
|
|
m.needsSemicolon = false
|
|
}
|
|
}
|
|
|
|
func (m *jsMinifier) minifyStmtOrBlock(i js.IStmt, blockType blockType) {
|
|
// minify stmt or a block
|
|
if blockStmt, ok := i.(*js.BlockStmt); ok {
|
|
blockStmt.List = optimizeStmtList(blockStmt.List, blockType)
|
|
m.renamer.renameScope(blockStmt.Scope)
|
|
m.minifyBlockAsStmt(blockStmt)
|
|
} else {
|
|
// optimizeStmtList can in some cases expand one stmt to two shorter stmts
|
|
list := optimizeStmtList([]js.IStmt{i}, blockType)
|
|
if len(list) == 1 {
|
|
m.minifyStmt(list[0])
|
|
} else if len(list) == 0 {
|
|
m.write(semicolonBytes)
|
|
m.needsSemicolon = false
|
|
} else {
|
|
m.minifyBlockStmt(&js.BlockStmt{List: list, Scope: js.Scope{}})
|
|
}
|
|
}
|
|
}
|
|
|
|
func (m *jsMinifier) minifyAlias(alias js.Alias) {
|
|
if alias.Name != nil {
|
|
if alias.Name[0] == '"' || alias.Name[0] == '\'' {
|
|
m.write(minifyString(alias.Name, false))
|
|
} else {
|
|
m.write(alias.Name)
|
|
}
|
|
if !bytes.Equal(alias.Name, starBytes) {
|
|
m.write(spaceBytes)
|
|
}
|
|
m.write(asSpaceBytes)
|
|
}
|
|
if alias.Binding != nil {
|
|
if alias.Binding[0] == '"' || alias.Binding[0] == '\'' {
|
|
m.write(minifyString(alias.Binding, false))
|
|
} else {
|
|
m.write(alias.Binding)
|
|
}
|
|
}
|
|
}
|
|
|
|
func (m *jsMinifier) minifyParams(params js.Params, removeUnused bool) {
|
|
// remove unused parameters from the end
|
|
j := len(params.List)
|
|
if removeUnused && params.Rest == nil {
|
|
for ; 0 < j; j-- {
|
|
if v, ok := params.List[j-1].Binding.(*js.Var); !ok || ok && 1 < v.Uses {
|
|
break
|
|
}
|
|
}
|
|
}
|
|
|
|
m.write(openParenBytes)
|
|
for i, item := range params.List[:j] {
|
|
if i != 0 {
|
|
m.write(commaBytes)
|
|
}
|
|
m.minifyBindingElement(item)
|
|
}
|
|
if params.Rest != nil {
|
|
if len(params.List) != 0 {
|
|
m.write(commaBytes)
|
|
}
|
|
m.write(ellipsisBytes)
|
|
m.minifyBinding(params.Rest)
|
|
}
|
|
m.write(closeParenBytes)
|
|
}
|
|
|
|
func (m *jsMinifier) minifyArguments(args js.Args) {
|
|
m.write(openParenBytes)
|
|
for i, item := range args.List {
|
|
if i != 0 {
|
|
m.write(commaBytes)
|
|
}
|
|
if item.Rest {
|
|
m.write(ellipsisBytes)
|
|
}
|
|
m.minifyExpr(item.Value, js.OpAssign)
|
|
}
|
|
m.write(closeParenBytes)
|
|
}
|
|
|
|
func (m *jsMinifier) minifyVarDecl(decl *js.VarDecl, onlyDefines bool) {
|
|
if len(decl.List) == 0 {
|
|
return
|
|
} else if decl.TokenType == js.ErrorToken {
|
|
// remove 'var' when hoisting variables
|
|
first := true
|
|
for _, item := range decl.List {
|
|
if item.Default != nil || !onlyDefines {
|
|
if !first {
|
|
m.write(commaBytes)
|
|
}
|
|
m.minifyBindingElement(item)
|
|
first = false
|
|
}
|
|
}
|
|
} else {
|
|
if decl.TokenType == js.VarToken && len(decl.List) <= 10000 {
|
|
// move single var decls forward and order for GZIP optimization
|
|
start := 0
|
|
if _, ok := decl.List[0].Binding.(*js.Var); !ok {
|
|
start++
|
|
}
|
|
for i := 0; i < len(decl.List); i++ {
|
|
item := decl.List[i]
|
|
if v, ok := item.Binding.(*js.Var); ok && item.Default == nil && len(v.Data) == 1 {
|
|
for j := start; j < len(decl.List); j++ {
|
|
if v2, ok := decl.List[j].Binding.(*js.Var); ok && decl.List[j].Default == nil && len(v2.Data) == 1 {
|
|
if m.renamer.identOrder[v2.Data[0]] < m.renamer.identOrder[v.Data[0]] {
|
|
continue
|
|
} else if m.renamer.identOrder[v2.Data[0]] == m.renamer.identOrder[v.Data[0]] {
|
|
break
|
|
}
|
|
}
|
|
decl.List = append(decl.List[:i], decl.List[i+1:]...)
|
|
decl.List = append(decl.List[:j], append([]js.BindingElement{item}, decl.List[j:]...)...)
|
|
break
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
m.write(decl.TokenType.Bytes())
|
|
m.writeSpaceBeforeIdent()
|
|
for i, item := range decl.List {
|
|
if i != 0 {
|
|
m.write(commaBytes)
|
|
}
|
|
m.minifyBindingElement(item)
|
|
}
|
|
}
|
|
}
|
|
|
|
func (m *jsMinifier) minifyFuncDecl(decl *js.FuncDecl, inExpr bool) {
|
|
parentRename := m.renamer.rename
|
|
m.renamer.rename = !decl.Body.Scope.HasWith && !m.o.KeepVarNames
|
|
m.hoistVars(&decl.Body)
|
|
decl.Body.List = optimizeStmtList(decl.Body.List, functionBlock)
|
|
|
|
if decl.Async {
|
|
m.write(asyncSpaceBytes)
|
|
}
|
|
m.write(functionBytes)
|
|
if decl.Generator {
|
|
m.write(starBytes)
|
|
}
|
|
|
|
// TODO: remove function name, really necessary?
|
|
//if decl.Name != nil && decl.Name.Uses == 1 {
|
|
// scope := decl.Body.Scope
|
|
// for i, vorig := range scope.Declared {
|
|
// if decl.Name == vorig {
|
|
// scope.Declared = append(scope.Declared[:i], scope.Declared[i+1:]...)
|
|
// }
|
|
// }
|
|
//}
|
|
|
|
if inExpr {
|
|
m.renamer.renameScope(decl.Body.Scope)
|
|
}
|
|
if decl.Name != nil && (!inExpr || 1 < decl.Name.Uses) {
|
|
if !decl.Generator {
|
|
m.write(spaceBytes)
|
|
}
|
|
m.write(decl.Name.Data)
|
|
}
|
|
if !inExpr {
|
|
m.renamer.renameScope(decl.Body.Scope)
|
|
}
|
|
|
|
m.minifyParams(decl.Params, true)
|
|
m.minifyBlockStmt(&decl.Body)
|
|
m.renamer.rename = parentRename
|
|
}
|
|
|
|
func (m *jsMinifier) minifyMethodDecl(decl *js.MethodDecl) {
|
|
parentRename := m.renamer.rename
|
|
m.renamer.rename = !decl.Body.Scope.HasWith && !m.o.KeepVarNames
|
|
m.hoistVars(&decl.Body)
|
|
decl.Body.List = optimizeStmtList(decl.Body.List, functionBlock)
|
|
|
|
if decl.Static {
|
|
m.write(staticBytes)
|
|
m.writeSpaceBeforeIdent()
|
|
}
|
|
if decl.Async {
|
|
m.write(asyncBytes)
|
|
if decl.Generator {
|
|
m.write(starBytes)
|
|
} else {
|
|
m.writeSpaceBeforeIdent()
|
|
}
|
|
} else if decl.Generator {
|
|
m.write(starBytes)
|
|
} else if decl.Get {
|
|
m.write(getBytes)
|
|
m.writeSpaceBeforeIdent()
|
|
} else if decl.Set {
|
|
m.write(setBytes)
|
|
m.writeSpaceBeforeIdent()
|
|
}
|
|
m.minifyPropertyName(decl.Name)
|
|
m.renamer.renameScope(decl.Body.Scope)
|
|
m.minifyParams(decl.Params, !decl.Set)
|
|
m.minifyBlockStmt(&decl.Body)
|
|
m.renamer.rename = parentRename
|
|
}
|
|
|
|
func (m *jsMinifier) minifyArrowFunc(decl *js.ArrowFunc) {
|
|
parentRename := m.renamer.rename
|
|
m.renamer.rename = !decl.Body.Scope.HasWith && !m.o.KeepVarNames
|
|
m.hoistVars(&decl.Body)
|
|
decl.Body.List = optimizeStmtList(decl.Body.List, functionBlock)
|
|
|
|
m.renamer.renameScope(decl.Body.Scope)
|
|
if decl.Async {
|
|
m.write(asyncBytes)
|
|
}
|
|
removeParens := false
|
|
if decl.Params.Rest == nil && len(decl.Params.List) == 1 && decl.Params.List[0].Default == nil {
|
|
if decl.Params.List[0].Binding == nil {
|
|
removeParens = true
|
|
} else if _, ok := decl.Params.List[0].Binding.(*js.Var); ok {
|
|
removeParens = true
|
|
}
|
|
}
|
|
if removeParens {
|
|
if decl.Async && decl.Params.List[0].Binding != nil {
|
|
// add space after async in: async a => ...
|
|
m.write(spaceBytes)
|
|
}
|
|
m.minifyBindingElement(decl.Params.List[0])
|
|
} else {
|
|
parentInFor := m.inFor
|
|
m.inFor = false
|
|
m.minifyParams(decl.Params, true)
|
|
m.inFor = parentInFor
|
|
}
|
|
m.write(arrowBytes)
|
|
removeBraces := false
|
|
if 0 < len(decl.Body.List) {
|
|
returnStmt, isReturn := decl.Body.List[len(decl.Body.List)-1].(*js.ReturnStmt)
|
|
if isReturn && returnStmt.Value != nil {
|
|
// merge expression statements to final return statement, remove function body braces
|
|
var list []js.IExpr
|
|
removeBraces = true
|
|
for _, item := range decl.Body.List[:len(decl.Body.List)-1] {
|
|
if expr, isExpr := item.(*js.ExprStmt); isExpr {
|
|
list = append(list, expr.Value)
|
|
} else {
|
|
removeBraces = false
|
|
break
|
|
}
|
|
}
|
|
if removeBraces {
|
|
list = append(list, returnStmt.Value)
|
|
expr := list[0]
|
|
if 0 < len(list) {
|
|
if 1 < len(list) {
|
|
expr = &js.CommaExpr{list}
|
|
}
|
|
expr = &js.GroupExpr{X: expr}
|
|
}
|
|
m.expectExpr = expectExprBody
|
|
m.minifyExpr(expr, js.OpAssign)
|
|
if m.groupedStmt {
|
|
m.write(closeParenBytes)
|
|
m.groupedStmt = false
|
|
}
|
|
}
|
|
} else if isReturn && returnStmt.Value == nil {
|
|
// remove empty return
|
|
decl.Body.List = decl.Body.List[:len(decl.Body.List)-1]
|
|
}
|
|
}
|
|
if !removeBraces {
|
|
m.minifyBlockStmt(&decl.Body)
|
|
}
|
|
m.renamer.rename = parentRename
|
|
}
|
|
|
|
func (m *jsMinifier) minifyClassDecl(decl *js.ClassDecl) {
|
|
m.write(classBytes)
|
|
if decl.Name != nil {
|
|
m.write(spaceBytes)
|
|
m.write(decl.Name.Data)
|
|
}
|
|
if decl.Extends != nil {
|
|
m.write(spaceExtendsBytes)
|
|
m.writeSpaceBeforeIdent()
|
|
m.minifyExpr(decl.Extends, js.OpLHS)
|
|
}
|
|
m.write(openBraceBytes)
|
|
m.needsSemicolon = false
|
|
for _, item := range decl.List {
|
|
m.writeSemicolon()
|
|
if item.StaticBlock != nil {
|
|
m.write(staticBytes)
|
|
m.minifyBlockStmt(item.StaticBlock)
|
|
} else if item.Method != nil {
|
|
m.minifyMethodDecl(item.Method)
|
|
} else {
|
|
if item.Static {
|
|
m.write(staticBytes)
|
|
if !item.Name.IsComputed() && item.Name.Literal.TokenType == js.IdentifierToken {
|
|
m.write(spaceBytes)
|
|
}
|
|
}
|
|
m.minifyPropertyName(item.Name)
|
|
if item.Init != nil {
|
|
m.write(equalBytes)
|
|
m.minifyExpr(item.Init, js.OpAssign)
|
|
}
|
|
m.requireSemicolon()
|
|
}
|
|
}
|
|
m.write(closeBraceBytes)
|
|
m.needsSemicolon = false
|
|
}
|
|
|
|
func (m *jsMinifier) minifyPropertyName(name js.PropertyName) {
|
|
if name.IsComputed() {
|
|
m.write(openBracketBytes)
|
|
m.minifyExpr(name.Computed, js.OpAssign)
|
|
m.write(closeBracketBytes)
|
|
} else if name.Literal.TokenType == js.StringToken {
|
|
m.write(minifyString(name.Literal.Data, false))
|
|
} else {
|
|
m.write(name.Literal.Data)
|
|
}
|
|
}
|
|
|
|
func (m *jsMinifier) minifyProperty(property js.Property) {
|
|
// property.Name is always set in ObjectLiteral
|
|
if property.Spread {
|
|
m.write(ellipsisBytes)
|
|
} else if v, ok := property.Value.(*js.Var); property.Name != nil && (!ok || !property.Name.IsIdent(v.Name())) {
|
|
// add 'old-name:' before BindingName as the latter will be renamed
|
|
m.minifyPropertyName(*property.Name)
|
|
m.write(colonBytes)
|
|
}
|
|
m.minifyExpr(property.Value, js.OpAssign)
|
|
if property.Init != nil {
|
|
m.write(equalBytes)
|
|
m.minifyExpr(property.Init, js.OpAssign)
|
|
}
|
|
}
|
|
|
|
func (m *jsMinifier) minifyBindingElement(element js.BindingElement) {
|
|
if element.Binding != nil {
|
|
parentInFor := m.inFor
|
|
m.inFor = false
|
|
m.minifyBinding(element.Binding)
|
|
m.inFor = parentInFor
|
|
if element.Default != nil {
|
|
m.write(equalBytes)
|
|
m.minifyExpr(element.Default, js.OpAssign)
|
|
}
|
|
}
|
|
}
|
|
|
|
func (m *jsMinifier) minifyBinding(ibinding js.IBinding) {
|
|
switch binding := ibinding.(type) {
|
|
case *js.Var:
|
|
m.write(binding.Data)
|
|
case *js.BindingArray:
|
|
m.write(openBracketBytes)
|
|
for i, item := range binding.List {
|
|
if i != 0 {
|
|
m.write(commaBytes)
|
|
}
|
|
m.minifyBindingElement(item)
|
|
}
|
|
if binding.Rest != nil {
|
|
if 0 < len(binding.List) {
|
|
m.write(commaBytes)
|
|
}
|
|
m.write(ellipsisBytes)
|
|
m.minifyBinding(binding.Rest)
|
|
}
|
|
m.write(closeBracketBytes)
|
|
case *js.BindingObject:
|
|
m.write(openBraceBytes)
|
|
for i, item := range binding.List {
|
|
if i != 0 {
|
|
m.write(commaBytes)
|
|
}
|
|
// item.Key is always set
|
|
if item.Key.IsComputed() {
|
|
m.minifyPropertyName(*item.Key)
|
|
m.write(colonBytes)
|
|
} else if v, ok := item.Value.Binding.(*js.Var); !ok || !item.Key.IsIdent(v.Data) {
|
|
// add 'old-name:' before BindingName as the latter will be renamed
|
|
m.minifyPropertyName(*item.Key)
|
|
m.write(colonBytes)
|
|
}
|
|
m.minifyBindingElement(item.Value)
|
|
}
|
|
if binding.Rest != nil {
|
|
if 0 < len(binding.List) {
|
|
m.write(commaBytes)
|
|
}
|
|
m.write(ellipsisBytes)
|
|
m.write(binding.Rest.Data)
|
|
}
|
|
m.write(closeBraceBytes)
|
|
}
|
|
}
|
|
|
|
func (m *jsMinifier) minifyExpr(i js.IExpr, prec js.OpPrec) {
|
|
if cond, ok := i.(*js.CondExpr); ok {
|
|
i = m.optimizeCondExpr(cond, prec)
|
|
} else if unary, ok := i.(*js.UnaryExpr); ok {
|
|
i = optimizeUnaryExpr(unary, prec)
|
|
}
|
|
|
|
switch expr := i.(type) {
|
|
case *js.Var:
|
|
for expr.Link != nil {
|
|
expr = expr.Link
|
|
}
|
|
data := expr.Data
|
|
if bytes.Equal(data, undefinedBytes) { // TODO: only if not defined
|
|
if js.OpUnary < prec {
|
|
m.write(groupedVoidZeroBytes)
|
|
} else {
|
|
m.write(voidZeroBytes)
|
|
}
|
|
} else if bytes.Equal(data, infinityBytes) { // TODO: only if not defined
|
|
if js.OpMul < prec {
|
|
m.write(groupedOneDivZeroBytes)
|
|
} else {
|
|
m.write(oneDivZeroBytes)
|
|
}
|
|
} else {
|
|
m.write(data)
|
|
}
|
|
case *js.LiteralExpr:
|
|
if expr.TokenType == js.DecimalToken {
|
|
m.write(decimalNumber(expr.Data, m.o.Precision))
|
|
} else if expr.TokenType == js.BinaryToken {
|
|
m.write(binaryNumber(expr.Data, m.o.Precision))
|
|
} else if expr.TokenType == js.OctalToken {
|
|
m.write(octalNumber(expr.Data, m.o.Precision))
|
|
} else if expr.TokenType == js.HexadecimalToken {
|
|
m.write(hexadecimalNumber(expr.Data, m.o.Precision))
|
|
} else if expr.TokenType == js.TrueToken {
|
|
if js.OpUnary < prec {
|
|
m.write(groupedNotZeroBytes)
|
|
} else {
|
|
m.write(notZeroBytes)
|
|
}
|
|
} else if expr.TokenType == js.FalseToken {
|
|
if js.OpUnary < prec {
|
|
m.write(groupedNotOneBytes)
|
|
} else {
|
|
m.write(notOneBytes)
|
|
}
|
|
} else if expr.TokenType == js.StringToken {
|
|
m.write(minifyString(expr.Data, true))
|
|
} else if expr.TokenType == js.RegExpToken {
|
|
// </script>/ => < /script>/
|
|
if 0 < len(m.prev) && m.prev[len(m.prev)-1] == '<' && bytes.HasPrefix(expr.Data, regExpScriptBytes) {
|
|
m.write(spaceBytes)
|
|
}
|
|
m.write(minifyRegExp(expr.Data))
|
|
} else {
|
|
m.write(expr.Data)
|
|
}
|
|
case *js.BinaryExpr:
|
|
mergeBinaryExpr(expr)
|
|
if expr.X == nil {
|
|
m.minifyExpr(expr.Y, prec)
|
|
break
|
|
}
|
|
|
|
precLeft := binaryLeftPrecMap[expr.Op]
|
|
// convert (a,b)&&c into a,b&&c but not a=(b,c)&&d into a=(b,c&&d)
|
|
if prec <= js.OpExpr {
|
|
if group, ok := expr.X.(*js.GroupExpr); ok {
|
|
if comma, ok := group.X.(*js.CommaExpr); ok && js.OpAnd <= exprPrec(comma.List[len(comma.List)-1]) {
|
|
expr.X = group.X
|
|
precLeft = js.OpExpr
|
|
}
|
|
}
|
|
}
|
|
if expr.Op == js.InstanceofToken || expr.Op == js.InToken {
|
|
group := expr.Op == js.InToken && m.inFor
|
|
if group {
|
|
m.write(openParenBytes)
|
|
}
|
|
m.minifyExpr(expr.X, precLeft)
|
|
m.writeSpaceAfterIdent()
|
|
m.write(expr.Op.Bytes())
|
|
m.writeSpaceBeforeIdent()
|
|
m.minifyExpr(expr.Y, binaryRightPrecMap[expr.Op])
|
|
if group {
|
|
m.write(closeParenBytes)
|
|
}
|
|
} else {
|
|
// TODO: has effect on GZIP?
|
|
//if expr.Op == js.EqEqToken || expr.Op == js.NotEqToken || expr.Op == js.EqEqEqToken || expr.Op == js.NotEqEqToken {
|
|
// // switch a==const for const==a, such as typeof a=="undefined" for "undefined"==typeof a (GZIP improvement)
|
|
// if _, ok := expr.Y.(*js.LiteralExpr); ok {
|
|
// expr.X, expr.Y = expr.Y, expr.X
|
|
// }
|
|
//}
|
|
|
|
if v, not, ok := isUndefinedOrNullVar(expr); ok {
|
|
// change a===null||a===undefined to a==null
|
|
op := js.EqEqToken
|
|
if not {
|
|
op = js.NotEqToken
|
|
}
|
|
expr = &js.BinaryExpr{op, v, &js.LiteralExpr{js.NullToken, nullBytes}}
|
|
}
|
|
|
|
m.minifyExpr(expr.X, precLeft)
|
|
if expr.Op == js.GtToken && m.prev[len(m.prev)-1] == '-' {
|
|
// 0 < len(m.prev) always
|
|
m.write(spaceBytes)
|
|
} else if expr.Op == js.EqEqEqToken || expr.Op == js.NotEqEqToken {
|
|
if left, ok := expr.X.(*js.UnaryExpr); ok && left.Op == js.TypeofToken {
|
|
if right, ok := expr.Y.(*js.LiteralExpr); ok && right.TokenType == js.StringToken {
|
|
if expr.Op == js.EqEqEqToken {
|
|
expr.Op = js.EqEqToken
|
|
} else {
|
|
expr.Op = js.NotEqToken
|
|
}
|
|
}
|
|
} else if right, ok := expr.Y.(*js.UnaryExpr); ok && right.Op == js.TypeofToken {
|
|
if left, ok := expr.X.(*js.LiteralExpr); ok && left.TokenType == js.StringToken {
|
|
if expr.Op == js.EqEqEqToken {
|
|
expr.Op = js.EqEqToken
|
|
} else {
|
|
expr.Op = js.NotEqToken
|
|
}
|
|
}
|
|
}
|
|
}
|
|
m.write(expr.Op.Bytes())
|
|
if expr.Op == js.AddToken {
|
|
// +++ => + ++
|
|
m.writeSpaceBefore('+')
|
|
} else if expr.Op == js.SubToken {
|
|
// --- => - --
|
|
m.writeSpaceBefore('-')
|
|
} else if expr.Op == js.DivToken {
|
|
// // => / /
|
|
m.writeSpaceBefore('/')
|
|
}
|
|
m.minifyExpr(expr.Y, binaryRightPrecMap[expr.Op])
|
|
}
|
|
case *js.UnaryExpr:
|
|
if expr.Op == js.PostIncrToken || expr.Op == js.PostDecrToken {
|
|
m.minifyExpr(expr.X, unaryPrecMap[expr.Op])
|
|
m.write(expr.Op.Bytes())
|
|
} else {
|
|
isLtNot := expr.Op == js.NotToken && 0 < len(m.prev) && m.prev[len(m.prev)-1] == '<'
|
|
m.write(expr.Op.Bytes())
|
|
if expr.Op == js.DeleteToken || expr.Op == js.VoidToken || expr.Op == js.TypeofToken || expr.Op == js.AwaitToken {
|
|
m.writeSpaceBeforeIdent()
|
|
} else if expr.Op == js.PosToken {
|
|
// +++ => + ++
|
|
m.writeSpaceBefore('+')
|
|
} else if expr.Op == js.NegToken || isLtNot {
|
|
// --- => - --
|
|
// <!-- => <! --
|
|
m.writeSpaceBefore('-')
|
|
} else if expr.Op == js.NotToken {
|
|
if lit, ok := expr.X.(*js.LiteralExpr); ok && (lit.TokenType == js.StringToken || lit.TokenType == js.RegExpToken) {
|
|
// !"string" => !1
|
|
m.write(oneBytes)
|
|
break
|
|
} else if ok && lit.TokenType == js.DecimalToken {
|
|
// !123 => !1 (except for !0)
|
|
if num := minify.Number(lit.Data, m.o.Precision); len(num) == 1 && num[0] == '0' {
|
|
m.write(zeroBytes)
|
|
} else {
|
|
m.write(oneBytes)
|
|
}
|
|
break
|
|
}
|
|
}
|
|
m.minifyExpr(expr.X, unaryPrecMap[expr.Op])
|
|
}
|
|
case *js.DotExpr:
|
|
if group, ok := expr.X.(*js.GroupExpr); ok {
|
|
if lit, ok := group.X.(*js.LiteralExpr); ok && lit.TokenType == js.DecimalToken {
|
|
num := minify.Number(lit.Data, m.o.Precision)
|
|
isInt := true
|
|
for _, c := range num {
|
|
if c == '.' || c == 'e' || c == 'E' {
|
|
isInt = false
|
|
break
|
|
}
|
|
}
|
|
if isInt {
|
|
m.write(num)
|
|
m.write(dotBytes)
|
|
} else {
|
|
m.write(num)
|
|
}
|
|
m.write(dotBytes)
|
|
m.write(expr.Y.Data)
|
|
break
|
|
}
|
|
}
|
|
if prec < js.OpMember {
|
|
m.minifyExpr(expr.X, js.OpCall)
|
|
} else {
|
|
m.minifyExpr(expr.X, js.OpMember)
|
|
}
|
|
if expr.Optional {
|
|
m.write(questionBytes)
|
|
} else if last := m.prev[len(m.prev)-1]; '0' <= last && last <= '9' {
|
|
// 0 < len(m.prev) always
|
|
isInteger := true
|
|
for _, c := range m.prev[:len(m.prev)-1] {
|
|
if c < '0' || '9' < c {
|
|
isInteger = false
|
|
break
|
|
}
|
|
}
|
|
if isInteger {
|
|
// prevent previous integer
|
|
m.write(dotBytes)
|
|
}
|
|
}
|
|
m.write(dotBytes)
|
|
m.write(expr.Y.Data)
|
|
case *js.GroupExpr:
|
|
if cond, ok := expr.X.(*js.CondExpr); ok {
|
|
expr.X = m.optimizeCondExpr(cond, js.OpExpr)
|
|
}
|
|
precInside := exprPrec(expr.X)
|
|
if prec <= precInside || precInside == js.OpCoalesce && prec == js.OpBitOr {
|
|
m.minifyExpr(expr.X, prec)
|
|
} else {
|
|
parentInFor := m.inFor
|
|
m.inFor = false
|
|
m.write(openParenBytes)
|
|
m.minifyExpr(expr.X, js.OpExpr)
|
|
m.write(closeParenBytes)
|
|
m.inFor = parentInFor
|
|
}
|
|
case *js.ArrayExpr:
|
|
parentInFor := m.inFor
|
|
m.inFor = false
|
|
m.write(openBracketBytes)
|
|
for i, item := range expr.List {
|
|
if i != 0 {
|
|
m.write(commaBytes)
|
|
}
|
|
if item.Spread {
|
|
m.write(ellipsisBytes)
|
|
}
|
|
m.minifyExpr(item.Value, js.OpAssign)
|
|
}
|
|
if 0 < len(expr.List) && expr.List[len(expr.List)-1].Value == nil {
|
|
m.write(commaBytes)
|
|
}
|
|
m.write(closeBracketBytes)
|
|
m.inFor = parentInFor
|
|
case *js.ObjectExpr:
|
|
parentInFor := m.inFor
|
|
m.inFor = false
|
|
groupedStmt := m.expectExpr != expectAny
|
|
if groupedStmt {
|
|
m.write(openParenBracketBytes)
|
|
} else {
|
|
m.write(openBraceBytes)
|
|
}
|
|
for i, item := range expr.List {
|
|
if i != 0 {
|
|
m.write(commaBytes)
|
|
}
|
|
m.minifyProperty(item)
|
|
}
|
|
m.write(closeBraceBytes)
|
|
if groupedStmt {
|
|
m.groupedStmt = true
|
|
}
|
|
m.inFor = parentInFor
|
|
case *js.TemplateExpr:
|
|
if expr.Tag != nil {
|
|
if prec < js.OpMember {
|
|
m.minifyExpr(expr.Tag, js.OpCall)
|
|
} else {
|
|
m.minifyExpr(expr.Tag, js.OpMember)
|
|
}
|
|
if expr.Optional {
|
|
m.write(optChainBytes)
|
|
}
|
|
}
|
|
parentInFor := m.inFor
|
|
m.inFor = false
|
|
for _, item := range expr.List {
|
|
m.write(replaceEscapes(item.Value, '`', 1, 2))
|
|
m.minifyExpr(item.Expr, js.OpExpr)
|
|
}
|
|
m.write(replaceEscapes(expr.Tail, '`', 1, 1))
|
|
m.inFor = parentInFor
|
|
case *js.NewExpr:
|
|
if expr.Args == nil && js.OpLHS < prec && prec != js.OpNew {
|
|
m.write(openNewBytes)
|
|
m.writeSpaceBeforeIdent()
|
|
m.minifyExpr(expr.X, js.OpNew)
|
|
m.write(closeParenBytes)
|
|
} else {
|
|
m.write(newBytes)
|
|
m.writeSpaceBeforeIdent()
|
|
if expr.Args != nil {
|
|
m.minifyExpr(expr.X, js.OpMember)
|
|
m.minifyArguments(*expr.Args)
|
|
} else {
|
|
m.minifyExpr(expr.X, js.OpNew)
|
|
}
|
|
}
|
|
case *js.NewTargetExpr:
|
|
m.write(newTargetBytes)
|
|
m.writeSpaceBeforeIdent()
|
|
case *js.ImportMetaExpr:
|
|
if m.expectExpr == expectExprStmt {
|
|
m.write(openParenBytes)
|
|
m.groupedStmt = true
|
|
}
|
|
m.write(importMetaBytes)
|
|
m.writeSpaceBeforeIdent()
|
|
case *js.YieldExpr:
|
|
m.write(yieldBytes)
|
|
m.writeSpaceBeforeIdent()
|
|
if expr.X != nil {
|
|
if expr.Generator {
|
|
m.write(starBytes)
|
|
m.minifyExpr(expr.X, js.OpAssign)
|
|
} else if v, ok := expr.X.(*js.Var); !ok || !bytes.Equal(v.Name(), undefinedBytes) { // TODO: only if not defined
|
|
m.minifyExpr(expr.X, js.OpAssign)
|
|
}
|
|
}
|
|
case *js.CallExpr:
|
|
m.minifyExpr(expr.X, js.OpCall)
|
|
parentInFor := m.inFor
|
|
m.inFor = false
|
|
if expr.Optional {
|
|
m.write(optChainBytes)
|
|
}
|
|
m.minifyArguments(expr.Args)
|
|
m.inFor = parentInFor
|
|
case *js.IndexExpr:
|
|
if m.expectExpr == expectExprStmt {
|
|
if v, ok := expr.X.(*js.Var); ok && bytes.Equal(v.Name(), letBytes) {
|
|
m.write(notBytes)
|
|
}
|
|
}
|
|
if prec < js.OpMember {
|
|
m.minifyExpr(expr.X, js.OpCall)
|
|
} else {
|
|
m.minifyExpr(expr.X, js.OpMember)
|
|
}
|
|
if expr.Optional {
|
|
m.write(optChainBytes)
|
|
}
|
|
if lit, ok := expr.Y.(*js.LiteralExpr); ok && lit.TokenType == js.StringToken && 2 < len(lit.Data) {
|
|
if isIdent := js.AsIdentifierName(lit.Data[1 : len(lit.Data)-1]); isIdent {
|
|
m.write(dotBytes)
|
|
m.write(lit.Data[1 : len(lit.Data)-1])
|
|
break
|
|
} else if isNum := js.AsDecimalLiteral(lit.Data[1 : len(lit.Data)-1]); isNum {
|
|
m.write(openBracketBytes)
|
|
m.write(minify.Number(lit.Data[1:len(lit.Data)-1], 0))
|
|
m.write(closeBracketBytes)
|
|
break
|
|
}
|
|
}
|
|
parentInFor := m.inFor
|
|
m.inFor = false
|
|
m.write(openBracketBytes)
|
|
m.minifyExpr(expr.Y, js.OpExpr)
|
|
m.write(closeBracketBytes)
|
|
m.inFor = parentInFor
|
|
case *js.CondExpr:
|
|
m.minifyExpr(expr.Cond, js.OpCoalesce)
|
|
m.write(questionBytes)
|
|
m.minifyExpr(expr.X, js.OpAssign)
|
|
m.write(colonBytes)
|
|
m.minifyExpr(expr.Y, js.OpAssign)
|
|
case *js.VarDecl:
|
|
m.minifyVarDecl(expr, true) // happens in for statement or when vars were hoisted
|
|
case *js.FuncDecl:
|
|
grouped := m.expectExpr == expectExprStmt && prec != js.OpExpr
|
|
if grouped {
|
|
m.write(openParenBytes)
|
|
} else if m.expectExpr == expectExprStmt {
|
|
m.write(notBytes)
|
|
}
|
|
parentInFor, parentGroupedStmt := m.inFor, m.groupedStmt
|
|
m.inFor, m.groupedStmt = false, false
|
|
m.minifyFuncDecl(expr, true)
|
|
m.inFor, m.groupedStmt = parentInFor, parentGroupedStmt
|
|
if grouped {
|
|
m.write(closeParenBytes)
|
|
}
|
|
case *js.ArrowFunc:
|
|
parentGroupedStmt := m.groupedStmt
|
|
m.groupedStmt = false
|
|
m.minifyArrowFunc(expr)
|
|
m.groupedStmt = parentGroupedStmt
|
|
case *js.MethodDecl:
|
|
parentGroupedStmt := m.groupedStmt
|
|
m.groupedStmt = false
|
|
m.minifyMethodDecl(expr) // only happens in object literal
|
|
m.groupedStmt = parentGroupedStmt
|
|
case *js.ClassDecl:
|
|
if m.expectExpr == expectExprStmt {
|
|
m.write(notBytes)
|
|
}
|
|
parentInFor, parentGroupedStmt := m.inFor, m.groupedStmt
|
|
m.inFor, m.groupedStmt = false, false
|
|
m.minifyClassDecl(expr)
|
|
m.inFor, m.groupedStmt = parentInFor, parentGroupedStmt
|
|
case *js.CommaExpr:
|
|
for i, item := range expr.List {
|
|
if i != 0 {
|
|
m.write(commaBytes)
|
|
}
|
|
m.minifyExpr(item, js.OpAssign)
|
|
}
|
|
}
|
|
}
|