globocom/megadraft-table-plugin

View on GitHub
tests/TableManagerModal_test.js

Summary

Maintainability
D
2 days
Test Coverage
/*
 * Copyright (c) 2017, Globo.com <https://github.com/globocom/megadraft-table-plugin>
 *
 * License: MIT
 */

import React from "react";
import {unmountComponentAtNode} from "react-dom";

import {mount, ReactWrapper} from "enzyme";
import chai from "chai";
import sinon from "sinon";

import TableManagerModal from "../src/TableManagerModal";
import {TableConfig} from "../src/TableConfig";
import {EmptyTableConfig, ValidTableConfig} from "./fixtures";

const expect = chai.expect;

describe("TableManagerModal", function () {

  beforeEach(function() {
    this.onSaveRequest = sinon.spy();
    this.tableManagerModal = mount(<TableManagerModal isOpen={true} onSaveRequest={(data) => {this.onSaveRequest(data);}} data={EmptyTableConfig}/>);
  });

  afterEach(function() {
    this.tableManagerModal = null;
    unmountComponentAtNode(document);
    document.body.innerHTML = "";
  });

  const findReactWrapper = function(find, tableManagerModal) {
    const dialog = tableManagerModal.find("Portal");
    return new ReactWrapper(dialog.getElement().props.children).find(find);
  };

  describe("isValid", function() {

    it("should be true when title is present", function(done) {
      this.tableManagerModal.setState({data: ValidTableConfig}, () => {
        expect(this.tableManagerModal.instance().isValid()).to.be.true;
        done();
      });
    });

    it("should be false when title is not present", function() {
      expect(this.tableManagerModal.instance().isValid()).to.be.false;
    });

  });

  describe("click add button", function() {

    it("save callback should be called when TableConfig is valid", function(done) {
      this.tableManagerModal.setState({data: ValidTableConfig}, () => {
        const addButton = document.querySelector(".table-manager-modal__add-button");
        addButton.click();
        expect(this.onSaveRequest.calledOnce).to.be.true;
        expect(this.onSaveRequest.getCall(0).args[0]).to.be.deep.equals(ValidTableConfig);
        done();
      });
    });

    it("save callback should not be called when TableConfig is invalid", function() {
      const addButton = document.querySelector(".table-manager-modal__add-button");
      addButton.click();
      expect(this.onSaveRequest.notCalled).to.be.true;
    });

  });

  describe("Title", function() {

    it("should update state when input changes", function() {
      expect(this.tableManagerModal.state().data.title).to.be.equals("");
      const titleInput = findReactWrapper("input[name=\"title\"]", this.tableManagerModal);
      titleInput.props().onChange({
        target: {
          name: "title",
          value: "Blo"
        }
      });
      expect(this.tableManagerModal.state().data.title).to.be.equals("Blo");
    });

  });

  describe("Header Style", function() {

    const getHeaderStyleObj = function() {
      return {
        top: false,
        bottom: false,
        right: false,
        left: false
      };
    };

    const testChangeHeaderStyle = function(position) {
      it(`should update state when header-style-${position} is clicked`, function() {
        expect(this.tableManagerModal.state().data.headerStyle[position]).to.be.false;
        const headeStyleCheckbox = findReactWrapper(`.bs-modal input[name="header-style"][value="${position}"]`, this.tableManagerModal);
        headeStyleCheckbox.props().onChange({
          target: {
            value: position,
            checked: true
          }
        });
        expect(this.tableManagerModal.state().data.headerStyle[position]).to.be.true;
      });

      it(`should check the checkbox-${position}`, function() {
        const newheaderStyle = Object.assign(getHeaderStyleObj(), {[position]: true});

        expect(findReactWrapper(`.bs-modal input[name="header-style"][value="${position}"]`, this.tableManagerModal).prop("checked")).to.be.false;
        this.tableManagerModal.setState({data: new TableConfig({headerStyle: newheaderStyle})});
        expect(findReactWrapper(`.bs-modal input[name="header-style"][value="${position}"]`, this.tableManagerModal).prop("checked")).to.be.true;
      });
    };

    testChangeHeaderStyle("top");
    testChangeHeaderStyle("bottom");
    testChangeHeaderStyle("right");
    testChangeHeaderStyle("left");

    it("should maintain the state of others options", function() {
      this.tableManagerModal.setState({data: ValidTableConfig});
      const headeStyleCheckbox = findReactWrapper(".bs-modal input[name=\"header-style\"][value=\"bottom\"]", this.tableManagerModal);
      headeStyleCheckbox.props().onChange({
        target: {
          value: "bottom",
          checked: true
        }
      });
      expect(this.tableManagerModal.state().data.headerStyle.top).to.be.true;
      expect(this.tableManagerModal.state().data.headerStyle.bottom).to.be.true;
      expect(this.tableManagerModal.state().data.headerStyle.right).to.be.false;
      expect(this.tableManagerModal.state().data.headerStyle.left).to.be.false;
    });

  });

  describe("Add or Remove Rows", function() {

    describe("Add", function() {

      beforeEach(function() {
        this.btnAddRow = document.querySelector(".bs-modal .add-remove-rows .btn-add");
      });

      it("should add a new row", function() {
        expect(this.tableManagerModal.state().data.rows).to.be.lengthOf(1);
        this.btnAddRow.click();
        expect(this.tableManagerModal.state().data.rows).to.be.lengthOf(2);
      });

      it("should add a new row after selected row cell", function() {
        const rows = [["A1", "B1"],["A2", "B2"]];
        const selectedCell = [1, 0];
        const data = Object.assign({}, ValidTableConfig, {rows});
        this.tableManagerModal.setState({data, selectedCell});
        this.btnAddRow.click();
        expect(this.tableManagerModal.state().data.rows).to.be.lengthOf(3);
        expect(this.tableManagerModal.state().data.rows[1]).to.not.deep.equals(rows[0]);
        expect(this.tableManagerModal.state().data.rows[1]).to.not.deep.equals(rows[1]);
      });

      it("should create a new row with the same quantity of columns when does have rows created previously", function() {
        const rows = [["A1", "B1"]];
        const data = Object.assign({}, ValidTableConfig, {rows});
        this.tableManagerModal.setState({data});
        this.btnAddRow.click();
        expect(this.tableManagerModal.state().data.rows[1]).to.deep.equals(["", ""]);
      });
    });

    describe("Remove", function() {

      beforeEach(function() {
        this.btnRemoveRow = document.querySelector(".bs-modal .add-remove-rows .btn-remove");
      });

      it("should keep row when does have only one row and one column", function() {
        const rows = [[""]];
        const data = Object.assign({}, ValidTableConfig, {rows});

        this.tableManagerModal.setState({data});
        this.btnRemoveRow.click();

        expect(this.tableManagerModal.state().data.rows).to.deep.equals(rows);
      });

      it("should remove the last row when does not have any selected cell", function() {
        const rows = [["A1", "B1"], ["A2", "B2"]];
        const data = Object.assign({}, ValidTableConfig, {rows});
        this.tableManagerModal.setState({data, selectedCell: []});

        this.btnRemoveRow.click();

        expect(this.tableManagerModal.state().data.rows).to.be.lengthOf(1);
        expect(this.tableManagerModal.state().data.rows[0]).to.deep.equals(["A1", "B1"]);
      });

      it("should remove the selected row when does have a selectedCell", function() {
        const rows = [["A1", "B1"], ["A2", "B2"]];
        const data = Object.assign({}, ValidTableConfig, {rows});
        this.tableManagerModal.setState({data, selectedCell: [0, 0]});

        this.btnRemoveRow.click();

        expect(this.tableManagerModal.state().data.rows).to.be.lengthOf(1);
        expect(this.tableManagerModal.state().data.rows[0]).to.deep.equals(["A2", "B2"]);
      });

    });

  });

  describe("Add or Remove Columns", function() {

    describe("Add", function() {

      beforeEach(function() {
        this.btnAddColumn = document.querySelector(".bs-modal .add-remove-columns .btn-add");
        const rows = [["A1", "B1"], ["A2", "B2"]];
        const data = Object.assign({}, ValidTableConfig, {rows});
        this.tableManagerModal.setState({data, selectedCell: []});
      });

      it("should add a new column", function() {
        const expected = [["A1", "B1", ""], ["A2", "B2", ""]];

        this.btnAddColumn.click();
        expect(this.tableManagerModal.state().data.rows).to.deep.equals(expected);
      });

      it("should add a new column after selected cell", function() {
        const expected = [["A1", "", "B1"], ["A2", "", "B2"]];
        this.tableManagerModal.setState({selectedCell: [0, 1]});

        this.btnAddColumn.click();
        expect(this.tableManagerModal.state().data.rows).to.deep.equals(expected);
      });

    });

    describe("Remove", function() {

      beforeEach(function() {
        this.btnRemoveColumn = document.querySelector(".bs-modal .add-remove-columns .btn-remove");
      });

      it("should keep the column when does have only one row and one column", function() {
        expect(this.tableManagerModal.state().data.rows).to.be.deep.equals([[""]]);

        this.btnRemoveColumn.click();

        expect(this.tableManagerModal.state().data.rows).to.be.deep.equals([[""]]);
      });

      it("should remove the last column when does not have any selected cell", function() {
        const rows = [["A1", "B1"], ["A2", "B2"]];
        const data = Object.assign({}, ValidTableConfig, {rows});
        this.tableManagerModal.setState({data, selectedCell: []});

        this.btnRemoveColumn.click();

        expect(this.tableManagerModal.state().data.rows).to.be.deep.equals([["A1"], ["A2"]]);
      });

      it("should remove the selected column when does have a selected cell", function() {
        const rows = [["A1", "B1"], ["A2", "B2"]];
        const data = Object.assign({}, ValidTableConfig, {rows});
        this.tableManagerModal.setState({data, selectedCell: [0, 1]});

        this.btnRemoveColumn.click();

        expect(this.tableManagerModal.state().data.rows).to.be.deep.equals([["B1"], ["B2"]]);
      });

    });
  });

  describe("Source", function() {
    it("should update state when input changes", function() {
      expect(this.tableManagerModal.state().data.title).to.be.equals("");
      const sourceInput = findReactWrapper(".bs-modal input[name=\"source\"]", this.tableManagerModal);
      sourceInput.props().onChange({
        target: {
          name: "source",
          value: "Blo"
        }
      });
      expect(this.tableManagerModal.state().data.source).to.be.equals("Blo");
    });
  });

  describe("TableView integration",function() {

    it("should use update cell value when a onEditTableCell is called", function() {

      const tableManager = this.tableManagerModal.instance();
      const newValue = "B78";
      tableManager.onEditTableCell(0,0, newValue);
      expect(this.tableManagerModal.state().data.rows[0][0]).to.equal(newValue);
    });

  });
});