Неудачная повторная отправка вложенного компонента с помощью JEST / ферментативного тестирования - PullRequest
0 голосов
/ 29 марта 2020

У меня есть родительский компонент App и вложенный компонент SearchByForm. SearchByForm содержит группу переключателей, и я пытаюсь изменить значение третьей кнопки, чтобы оно было проверено (true). Его можно определить по его className, .workstationRadioBtn.

Класс SearchByForm

import React from 'react';

class SearchByForm extends React.Component {

    constructor(props) {
      super(props);
      this.state = { which: 'order' };
    }

    buttonChange = (e) => {
      if (e.target.checked) {
        this.setState({ which: e.target.value });
        this.props.updateFromSearchByForm(e.target.value);
      }
    }

    render() {
      return (
        <form className="SearchByForm">
          <div className="form-check">
            <label>
              <input
                type="radio"
                name="searchBy"
                value="order"
                checked={this.state.which === 'order'}
                onChange={this.buttonChange}
                className="orderRadioBtn"
              />
              Order
              </label>
          </div>
          <div className="form-check">
            <label>
              <input
                type="radio"
                name="searchBy"
                value="employee"
                checked={this.state.which === 'employee'}
                onChange={this.buttonChange}
                className="employeeRadioBtn"
              />
              Employee
              </label>
          </div>
          <div className="form-check">
            <label>
              <input
                type="radio"
                name="searchBy"
                value="workstation"
                checked={this.state.which === 'workstation'}
                onChange={this.buttonChange}
                className="workstationRadioBtn"
              />
              Work Station
              </label>
          </div>
        </form>
      );
    }
  }

  export default SearchByForm;

Класс компонента поиска:

class Search extends React.Component {

  constructor(props) {
    super(props);
    this.state = { searchKey: '_id', searchBy: 'order', searchResults: null };
    this.updateFromSearchByForm = this.updateFromSearchByForm.bind(this);
    this.updateFromKeysForm = this.updateFromKeysForm.bind(this);
    this.clearSearchResults = this.clearSearchResults.bind(this);
    this.populateSearchResults = this.populateSearchResults.bind(this);
  }

  updateFromSearchByForm(value) {
    this.setState({ searchBy: value });
  }

  updateFromKeysForm(value) {
    this.setState({ searchKey: value });
  }

  keyToSearchBy = () => {
    switch (this.state.searchBy) {
      case 'order':
        return (
          <KeysForm which='order' updateFromKeysForm={this.updateFromKeysForm}></KeysForm>
        );
      case 'employee':
        return (
          <KeysForm which='employee' updateFromKeysForm={this.updateFromKeysForm}></KeysForm>
        );
      case 'workstation':
        return (
          <KeysForm which='workstation' updateFromKeysForm={this.updateFromKeysForm}></KeysForm>
        );
      default:
        return (
          <KeysForm which='order' updateFromKeysForm={this.updateFromKeysForm}></KeysForm>
        );
    }
  }

  clearSearchResults = () => {
    this.setState({ searchResults: null });
  }

  populateSearchResults = (results) => {
    this.setState({ searchResults: results });
  }

  displayOrderResults = () => {
    let list = [];
    let count = 0;
    list.push(<label> Orders found that match your search criteria:</label>)
    for (let result of this.state.searchResults) {
      list.push(
        <ListGroupItem key={count}>ID: {result._id}, OrderID: {result.OrderID}, CustomerID: {result.CustomerID}, DateRequired: {result.DateRequired}, Status: {result.Status}, PriorityFlag: {result.PriorityFlag}</ListGroupItem>
      )
      count++;
    }
    return list;
  }

  displayEmployeeResults = () => {
    let list = [];
    let count = 0;
    list.push(<label> Employees found that match your search criteria:</label>)
    for (let result of this.state.searchResults) {
      list.push(
        <ListGroupItem key={count}>ID: {result._id}, Staff Name: {result.WorkerName}</ListGroupItem>
      )
      count++;
    }
    return list;
  }

  displayWorkStationResults = () => {
    let list = [];
    let count = 0;
    list.push(<label> Stations found that match your search criteria:</label>)
    for (let result of this.state.searchResults) {
      list.push(
        <ListGroupItem key={count}>ID: {result._id}, Work Station Name: {result.StationName}</ListGroupItem>
      )
      count++;
    }
    return list;
  }

  showSearchResults = () => {
    //Know how to display based on searchBy type.
    switch (this.state.searchBy) {
      case 'order':
        return (this.displayOrderResults());
      case 'employee':
        return (this.displayEmployeeResults());
      case 'workstation':
        return (this.displayWorkStationResults());
      default:
        break;
    }
    for (let result of this.state.searchResults) {
      console.log(result);
    }
  }

  render() {
    return (
      <Jumbotron>
        <div className="App">
          <div className='header'>
            <div className='header-space1'>
              <div className='button-header'>
                <Link to='/menu'>
                  <Button id='menu'>Back to Menu</Button>
                </Link>
              </div>
            </div>
            <div className='header-space2'>
              <h2>Search</h2>
            </div>
            <div className='header-space3'>
              <div className='button-header'>
              </div>
            </div>
          </div>
          <div className='Search'>
            <div className='Row'>
              <div className='SearchForm'>
                <p>What would you like to search by?</p>
                <SearchByForm updateFromSearchByForm={this.updateFromSearchByForm}></SearchByForm>
                {this.keyToSearchBy()}
              </div>
            </div>
            <ValueForm searchKey={this.state.searchKey} searchBy={this.state.searchBy} clearSearchResults={this.clearSearchResults} populateSearchResults={this.populateSearchResults}></ValueForm>
            <br></br>
            <div className='Row'>
              <div className='SearchResults'>
                <p>Search Results</p>
                {this.state.searchResults ?
                  <ListGroup>
                    {this.showSearchResults()}
                  </ListGroup>
                  :
                  null}
              </div>
            </div>
          </div>
        </div>
      </Jumbotron>
    );
  }

}

В настоящее время тест создает полную копию дерева и успешно проверяет проверенное значение .workstationRadioBtn на true. Он также успешно меняет поле состояния «который» в SearchByForm на «рабочая станция». Текущее состояние моего теста выглядит следующим образом:

describe('Work Station Search', () => {
    it(' successfully selects order radio button', () => {
        const wrapper = mount(
            <Router>
                <Search />
            </Router>
        );
        console.log(wrapper.debug());
        const wrapperSearchByForm = wrapper.find(SearchByForm);
        wrapperSearchByForm.find('input').filter('.workstationRadioBtn').simulate('change', {target: {checked: true}});
        wrapperSearchByForm.setState({ which: 'workstation' });
        wrapper.update();
        console.log(wrapper.debug());
    })
});

Код отладки теста показывает, что вышеприведенное является верным, однако update (), похоже, не выполнил повторную визуализацию второго вложенного компонента, KeysForm, чей возвращаемый рендер зависит от SearchByForm. Он по-прежнему работает так, как будто установлен переключатель «Порядок».

Вот полный вывод отладочной информации теста:

console.log src/__tests__/Search/Search.test.js:84
      <BrowserRouter>
        <Router history={{...}}>
          <Search>
            <Jumbotron fluid={false}>
              <div className="jumbotron">
                <div className="App">
                  <div className="header">
                    <div className="header-space1">
                      <div className="button-header">
                        <Link to="/menu">
                          <LinkAnchor href="/menu" navigate={[Function: navigate]}>
                            <a href="/menu" onClick={[Function: onClick]}>
                              <Button id="menu" variant="primary" active={false} disabled={false} type="button">
                                <button id="menu" disabled={false} type="button" className="btn btn-primary">
                                  Back to Menu
                                </button>
                              </Button>
                            </a>
                          </LinkAnchor>
                        </Link>
                      </div>
                    </div>
                    <div className="header-space2">
                      <h2>
                        Search
                      </h2>
                    </div>
                    <div className="header-space3">
                      <div className="button-header" />
                    </div>
                  </div>
                  <div className="Search">
                    <div className="Row">
                      <div className="SearchForm">
                        <p>
                          What would you like to search by?
                        </p>
                        <SearchByForm updateFromSearchByForm={[Function: bound updateFromSearchByForm]}>
                          <form className="SearchByForm">
                            <div className="form-check">
                              <label>
                                <input type="radio" name="searchBy" value="order" checked={true} onChange={[Function]} className="orderRadioBtn" />
                                Order
                              </label>
                            </div>
                            <div className="form-check">
                              <label>
                                <input type="radio" name="searchBy" value="employee" checked={false} onChange={[Function]} className="employeeRadioBtn" />
                                Employee
                              </label>
                            </div>
                            <div className="form-check">
                              <label>
                                <input type="radio" name="searchBy" value="workstation" checked={false} onChange={[Function]} className="workstationRadioBtn" />
                                Work Station
                              </label>
                            </div>
                          </form>
                        </SearchByForm>
                        <KeysForm which="order" updateFromKeysForm={[Function: bound updateFromKeysForm]}>
                          <div className="KeysForm">
                            <div className="OrderKeys">
                              <p>
                                Which Order Key would you like to search by?
                              </p>
                              <form>
                                <div className="form-check">
                                  <label>
                                    <input type="radio" name="orderKey" value="_id" checked={true} onChange={[Function]} className="form-check-input" />
                                    Id
                                  </label>
                                </div>
                                <div className="form-check">
                                  <label>
                                    <input type="radio" name="orderKey" value="DateRequired" checked={false} onChange={[Function]} className="form-check-input" />
                                    Date Required
                                  </label>
                                </div>
                                <div className="form-check">
                                  <label>
                                    <input type="radio" name="orderKey" value="Status" checked={false} onChange={[Function]} className="form-check-input" />
                                    Status
                                  </label>
                                </div>
                                <div className="form-check">
                                  <label>
                                    <input type="radio" name="orderKey" value="PriorityFlag" checked={false} onChange={[Function]} className="form-check-input" />
                                    Priority Flag
                                  </label>
                                </div>
                              </form>
                            </div>
                          </div>
                        </KeysForm>
                      </div>
                    </div>
                    <ValueForm searchKey="_id" searchBy="order" clearSearchResults={[Function: bound ]} populateSearchResults={[Function: bound ]}>
                      <form className="ValueForm">
                        <label>
                          <input type="text" name="value" value="" onChange={[Function]} />
                        </label>
                        <Button onClick={[Function]} variant="primary" active={false} disabled={false} type="button">
                          <button onClick={[Function]} disabled={false} type="button" className="btn btn-primary">
                            Search
                          </button>
                        </Button>
                      </form>
                    </ValueForm>
                    <br />
                    <div className="Row">
                      <div className="SearchResults">
                        <p>
                          Search Results
                        </p>
                      </div>
                    </div>
                  </div>
                </div>
              </div>
            </Jumbotron>
          </Search>
        </Router>
      </BrowserRouter>
...