/*
 * Copyright 2018 The Closure Compiler Authors.
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
package com.google.javascript.jscomp;

import org.junit.Before;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.junit.runners.JUnit4;

@RunWith(JUnit4.class)
public final class Es6SplitVariableDeclarationsTest extends CompilerTestCase {

  public Es6SplitVariableDeclarationsTest() {
    super();
  }

  @Override
  @Before
  public void setUp() throws Exception {
    super.setUp();
    enableTypeCheck();
    enableTypeInfoValidation();

    // there are a lot of 'property x never defined on ?' warnings caused by object destructuring
    ignoreWarnings(TypeCheck.POSSIBLE_INEXISTENT_PROPERTY);
    ignoreWarnings(TypeCheck.INEXISTENT_PROPERTY);
    ignoreWarnings(TypeValidator.TYPE_MISMATCH_WARNING);
  }

  @Override
  protected CompilerPass getProcessor(final Compiler compiler) {
    return new Es6SplitVariableDeclarations(compiler);
  }

  @Test
  public void testSplitArrayDestructuring() {
    test("var   [a] = [], b = 3;", "var [a] = []; var b = 3;");
    test("let   [a] = [], b = 3;", "let [a] = []; let b = 3;");
    test("const [a] = [], b = 3;", "const [a] = []; const b = 3;");
  }

  @Test
  public void testSplitObjectDestructuring() {
    test("var   {a} = {}, b = 3;", "var {a} = {}; var b = 3;");
    test("let   {a} = {}, b = 3;", "let {a} = {}; let b = 3;");
    test("const {a} = {}, b = 3;", "const {a} = {}; const b = 3;");
  }

  @Test
  public void testSplitObjectDestructuringInsideBlock() {
    test("{ var   {a} = {}, b = 3; }", "{ var {a} = {}; var b = 3; }");
    test("{ let   {a} = {}, b = 3; }", "{ let {a} = {}; let b = 3; }");
    test("{ const {a} = {}, b = 3; }", "{ const {a} = {}; const b = 3; }");
  }

  @Test
  public void testCannotSplitInForLoopInitializer() {
    testError("for (var   [a] = [], b = 3;;) {}", Es6ToEs3Util.CANNOT_CONVERT_YET);
    testError("for (let   [a] = [], b = 3;;) {}", Es6ToEs3Util.CANNOT_CONVERT_YET);
    testError("for (const [a] = [], b = 3;;) {}", Es6ToEs3Util.CANNOT_CONVERT_YET);
  }

  @Test
  public void testCannotSplitLabeledDeclaration() {
    testError("label: var   [a] = [], b = 3;", Es6ToEs3Util.CANNOT_CONVERT_YET);
    // `label: let   [a] = [], b = 3;` and `label: const [a] = [], b = 3;` are syntax errors
  }

  @Test
  public void testNothingToSplit() {
    testSame("var [a] = arr;");
    testSame("var {a} = obj;");
    testSame("var a = 1, b = 2;");
    testSame("for (var a = 1, b = 2;;) {}");
    testSame("for (let [x] = arr;;) {}");
    testSame("for (let {length: x} in object) {}");
  }
}
