fixes & improvements to [].join()

fixes
- [a].join() => "" + a
- ["a", , "b"].join() => "a,,b"
- ["a", null, "b"].join() => "a,,b"
- ["a", undefined, "b"].join() => "a,,b"

improvements
- ["a", "b"].join(null) => "anullb"
- ["a", "b"].join(undefined) => "a,b"
- [a + "b", c].join("") => a + "b" + c

closes #1453
This commit is contained in:
alexlamsl
2017-02-18 19:05:11 +08:00
parent 148047fbbf
commit 100307ab31
2 changed files with 92 additions and 23 deletions

View File

@@ -2290,39 +2290,57 @@ merge(Compressor.prototype, {
}).transform(compressor); }).transform(compressor);
} }
else if (exp instanceof AST_Dot && exp.expression instanceof AST_Array && exp.property == "join") EXIT: { else if (exp instanceof AST_Dot && exp.expression instanceof AST_Array && exp.property == "join") EXIT: {
var separator = self.args.length == 0 ? "," : self.args[0].evaluate(compressor)[1]; var separator;
if (separator == null) break EXIT; // not a constant if (self.args.length > 0) {
var elements = exp.expression.elements.reduce(function(a, el){ separator = self.args[0].evaluate(compressor);
if (separator.length < 2) break EXIT; // not a constant
separator = separator[1];
}
var elements = [];
var consts = [];
exp.expression.elements.forEach(function(el) {
el = el.evaluate(compressor); el = el.evaluate(compressor);
if (a.length == 0 || el.length == 1) { if (el.length > 1) {
a.push(el); consts.push(el[1]);
} else { } else {
var last = a[a.length - 1]; if (consts.length > 0) {
if (last.length == 2) { elements.push(make_node(AST_String, self, {
// it's a constant value: consts.join(separator)
var val = "" + last[1] + separator + el[1]; }));
a[a.length - 1] = [ make_node_from_constant(compressor, val, last[0]), val ]; consts.length = 0;
} else {
a.push(el);
} }
elements.push(el[0]);
}
});
if (consts.length > 0) {
elements.push(make_node(AST_String, self, {
value: consts.join(separator)
}));
} }
return a;
}, []);
if (elements.length == 0) return make_node(AST_String, self, { value: "" }); if (elements.length == 0) return make_node(AST_String, self, { value: "" });
if (elements.length == 1) return elements[0][0]; if (elements.length == 1) {
if (elements[0].is_string(compressor)) {
return elements[0];
}
return make_node(AST_Binary, elements[0], {
operator : "+",
left : make_node(AST_String, self, { value: "" }),
right : elements[0]
});
}
if (separator == "") { if (separator == "") {
var first; var first;
if (elements[0][0] instanceof AST_String if (elements[0].is_string(compressor)
|| elements[1][0] instanceof AST_String) { || elements[1].is_string(compressor)) {
first = elements.shift()[0]; first = elements.shift();
} else { } else {
first = make_node(AST_String, self, { value: "" }); first = make_node(AST_String, self, { value: "" });
} }
return elements.reduce(function(prev, el){ return elements.reduce(function(prev, el){
return make_node(AST_Binary, el[0], { return make_node(AST_Binary, el, {
operator : "+", operator : "+",
left : prev, left : prev,
right : el[0], right : el
}); });
}, first).transform(compressor); }, first).transform(compressor);
} }
@@ -2331,9 +2349,7 @@ merge(Compressor.prototype, {
var node = self.clone(); var node = self.clone();
node.expression = node.expression.clone(); node.expression = node.expression.clone();
node.expression.expression = node.expression.expression.clone(); node.expression.expression = node.expression.expression.clone();
node.expression.expression.elements = elements.map(function(el){ node.expression.expression.elements = elements;
return el[0];
});
return best_of(self, node); return best_of(self, node);
} }
} }

View File

@@ -21,10 +21,19 @@ constant_join: {
input: { input: {
var a = [ "foo", "bar", "baz" ].join(""); var a = [ "foo", "bar", "baz" ].join("");
var a1 = [ "foo", "bar", "baz" ].join(); var a1 = [ "foo", "bar", "baz" ].join();
var a2 = [ "foo", "bar", "baz" ].join(null);
var a3 = [ "foo", "bar", "baz" ].join(void 0);
var a4 = [ "foo", , "baz" ].join();
var a5 = [ "foo", null, "baz" ].join();
var a6 = [ "foo", void 0, "baz" ].join();
var b = [ "foo", 1, 2, 3, "bar" ].join(""); var b = [ "foo", 1, 2, 3, "bar" ].join("");
var c = [ boo(), "foo", 1, 2, 3, "bar", bar() ].join(""); var c = [ boo(), "foo", 1, 2, 3, "bar", bar() ].join("");
var c1 = [ boo(), bar(), "foo", 1, 2, 3, "bar", bar() ].join(""); var c1 = [ boo(), bar(), "foo", 1, 2, 3, "bar", bar() ].join("");
var c2 = [ 1, 2, "foo", "bar", baz() ].join(""); var c2 = [ 1, 2, "foo", "bar", baz() ].join("");
var c3 = [ boo() + bar() + "foo", 1, 2, 3, "bar", bar() + "foo" ].join("");
var c4 = [ 1, 2, null, undefined, "foo", "bar", baz() ].join("");
var c5 = [ boo() + bar() + "foo", 1, 2, 3, "bar", bar() + "foo" ].join();
var c6 = [ 1, 2, null, undefined, "foo", "bar", baz() ].join();
var d = [ "foo", 1 + 2 + "bar", "baz" ].join("-"); var d = [ "foo", 1 + 2 + "bar", "baz" ].join("-");
var e = [].join(foo + bar); var e = [].join(foo + bar);
var f = [].join(""); var f = [].join("");
@@ -33,10 +42,19 @@ constant_join: {
expect: { expect: {
var a = "foobarbaz"; var a = "foobarbaz";
var a1 = "foo,bar,baz"; var a1 = "foo,bar,baz";
var a2 = "foonullbarnullbaz";
var a3 = "foo,bar,baz";
var a4 = "foo,,baz";
var a5 = "foo,,baz";
var a6 = "foo,,baz";
var b = "foo123bar"; var b = "foo123bar";
var c = boo() + "foo123bar" + bar(); var c = boo() + "foo123bar" + bar();
var c1 = "" + boo() + bar() + "foo123bar" + bar(); var c1 = "" + boo() + bar() + "foo123bar" + bar();
var c2 = "12foobar" + baz(); var c2 = "12foobar" + baz();
var c3 = boo() + bar() + "foo123bar" + (bar() + "foo");
var c4 = "12foobar" + baz();
var c5 = [ boo() + bar() + "foo", 1, 2, 3, "bar", bar() + "foo" ].join();
var c6 = [ "1,2,,,foo,bar", baz() ].join();
var d = "foo-3bar-baz"; var d = "foo-3bar-baz";
var e = [].join(foo + bar); var e = [].join(foo + bar);
var f = ""; var f = "";
@@ -73,6 +91,41 @@ constant_join_2: {
} }
} }
constant_join_3: {
options = {
unsafe: true,
evaluate: true,
};
input: {
var a = [ null ].join();
var b = [ , ].join();
var c = [ , 1, , 3 ].join();
var d = [ foo ].join();
var e = [ foo, null, undefined, bar ].join("-");
var f = [ foo, bar ].join("");
var g = [ null, "foo", null, bar + "baz" ].join("");
var h = [ null, "foo", null, bar + "baz" ].join("-");
var i = [ "foo" + bar, null, baz + "moo" ].join("");
var j = [ foo + "bar", baz ].join("");
var k = [ foo, "bar" + baz ].join("");
var l = [ foo, bar + "baz" ].join("");
}
expect: {
var a = "";
var b = "";
var c = ",1,,3";
var d = "" + foo;
var e = [ foo, "-", bar ].join("-");
var f = "" + foo + bar;
var g = "foo" + (bar + "baz");
var h = [ "-foo-", bar + "baz" ].join("-");
var i = "foo" + bar + (baz + "moo");
var j = foo + "bar" + baz;
var k = foo + ("bar" + baz);
var l = foo + (bar + "baz");
}
}
for_loop: { for_loop: {
options = { options = {
unsafe : true, unsafe : true,