React-Redux реквизиты не обновляются - PullRequest
0 голосов
/ 07 ноября 2019

Я пытаюсь объединить реакцию с d3 для создания графика, чтобы в итоге реализовать визуализатор алгоритма сортировки. У меня есть код return <svg ref={node=>this.node=node}/> в функции рендеринга, и я вызываю узел в функции с именем renderSvg(), которую я вызываю в componentDidUpdate. Похоже, что реквизиты не обновляются.

Я провел небольшое исследование и понимаю, что наиболее распространенной причиной этого является то, что состояние при обновлении состояния. Тем не менее, я просмотрел свой код, и, похоже, я все равно не изменяю исходное состояние. У меня нет объектов ни в одной из моих частей состояния, но там, где у меня есть массивы, я использую оператор распространения, чтобы убедиться, что я не изменяю оригинал. Мне интересно, связано ли это с тем, что я использую Redux.combineReducers. Я не собираюсь проверять реальные алгоритмы сортировки, да, я все еще пытаюсь получить случайно сгенерированный массив для визуальной визуализации в SVG. Обратите внимание, что я также использую reduxThunk, если это имеет какое-то значение.

Кто-нибудь может помочь? Если это функция combineReducers, есть ли способ обойти это? Вот ссылка на мой проект и мой код ниже:

$(function(){
  //REDUC
  //array action types
  const GENERATE_ARRAY = 'GENERATE_ARRAY';
  const SORT_ARRAY = 'SORT_ARRAY';

  //accessCount action types
  const INCREMENT_ACCESS_COUNTER = 'INCREMENT_ACCESS_COUNTER';
  const RESET_ACCESS_COUNTER = 'RESET_ACCESS_COUNTER';

  //indexing action types
  const ACCESSING_INDICES = 'ACCESSING_INDICES'
  const MIN_INDEX = 'MIN_INDEX';
  const MAX_INDEX = 'MAX_INDEX';
  const SORTED_INDICES = 'SORTED_INDICES';

  //array action generators
  function generateArray(value){
    return {type: GENERATE_ARRAY, value};
  }
  function sortArray(algorithm){
    return {type: SORT_ARRAY, algorithm};
  }

  //accessCounter action generators
  function incrementAccessCounter(){
    return {type: INCREMENT_ACCESS_COUNTER};
  }
  function resetAccessCounter(){
    return {type: RESET_ACCESS_COUNTER};
  }

  //sortedIndices and accessingIndices action generators
  function updateSortedIndices(value){
    return {type: SORTED_INDICES, value};
  }
  function updateAccessingIndices(value){
    return {type: ACCESSING_INDICES, value};
  }

  //max and min index action generators
  function setMinIndex(index){
    return {type: MIN_INDEX, index};
  }
  function setMaxIndex(index){
    return {type: MAX_INDEX, index};
  }

  //initial values
  const INITIAL_ARRAY = [];
  const INITIAL_INDEX = -1;
  const INITIAL_ACCESS_COUNTER = 0;
  const INITIAL_STATE = {
    array: [],
    accessCounter: INITIAL_ACCESS_COUNTER,
    sortedIndices: [],
    accessingIndices: [],
    minIndex: INITIAL_INDEX,
    maxIndex: INITIAL_INDEX 
  }

  //state reducers
  function arrayReducer(state = [], action){
    switch(action.type){
      case GENERATE_ARRAY:
        if (Array.isArray(action.value))
          return action.value;
        let result = [];
        for (let i = 0; i < action.value; i++)
          result.push(Math.random());
        return result;
      default: return [...state];
    }
  }

  function sortedIndicesReducer(state = [], action){
    switch(action.type){
      case SORTED_INDICES:
        if (Array.isArray(action.value))
          return [...action.value];
        return [...state, action.value];
      default: return [...state];
    }
  }

  function accessingIndicesReducer(state = [], action){
    switch(action.type){
      case ACCESSING_INDICES:
        if (Array.isArray(action.value))
          return [...action.value];
        return [...state, action.value];
      default: return [...state];
    }
  }

  function accessCounterReducer(state = INITIAL_ACCESS_COUNTER, action){
    switch(action.type){
      case INCREMENT_ACCESS_COUNTER: return state + 1;
      case RESET_ACCESS_COUNTER: return 0;
      default: return state;
    }
  }

  function minIndexReducer(state = INITIAL_INDEX, action){
    switch(action.type){
      case MIN_INDEX: return action.index;
      default: return state;
    }
  }

  function maxIndexReducer(state = INITIAL_INDEX, action){
    switch(action.type){
      case MAX_INDEX: return action.index;
      default: return state;
    }
  }

  const rootReducer = Redux.combineReducers({
    array: arrayReducer,
    accessCounter: accessCounterReducer,
    sortedIndices: sortedIndicesReducer,
    accessingIndices: accessingIndicesReducer,
    minIndex: minIndexReducer,
    maxIndex: maxIndexReducer
  });

  const store = Redux.createStore(rootReducer, Object.assign({},INITIAL_STATE), Redux.applyMiddleware(ReduxThunk.default));

  //sorting algorithms
  const sortingAlgorithms = [
    {
      name: 'Selection Sort',
      algorithm: function(){
        return function(dispatch, getState){
          dispatch(resetAccessCounter());
          dispatch(updateSortedIndices([]));
          let array = [...getState().array];
          let sorted = [...getState().sortedIndices];
          for (let i = 0; i < array.length; i++){
            let minIndex = i;
            for (let j = 0; j < array.length; j++){
              let accessing = j;
              minIndex = array[j] < array[minIndex]
              ? j : minIndex;
              dispatch(incrementAccessCounter());
              dispatch(updateAccessingIndices([accessing]));
              if (minIndex != j)
                dispatch(setMinIndex(minIndex));
            }
            dispatch(updateAccessingIndices([]));
            let tmp = array[i];
            array.splice(i, 1, array[minIndex]);
            array.splice(minIndex, 1, tmp);
            dispatch(setMinIndex(-1));
            dispatch(generateArray(array));
            dispatch(updateSortedIndices(i));
          }
        }
      }
    },
    {
      name: 'Bubble Sort',
      algorithm: function(){
        return function(dispatch, getState){

        };
      }
    }
  ];



  //React
  class Form extends React.Component{
    constructor(props){
      super(props);
      this.state = {
        length: '1000',
        algorithm: sortingAlgorithms[0].algorithm
      }
      this.algorithmHandler = this.algorithmHandler.bind(this);
      this.lengthHandler = this.lengthHandler.bind(this);
      this.generateHandler = this.generateHandler.bind(this);
      this.sortHandler = this.sortHandler.bind(this);
    }
    algorithmHandler(event){
      this.setState({algorithm: event.target.value});
    }
    lengthHandler(event){
      this.setState({length: event.targer.value});
    }
    generateHandler(event){
      event.preventDefault();
      console.log('before:', this.props, store.getState());
      this.props.generateArray(parseInt(this.state.length));
      console.log('after:', this.props, store.getState());
    }
    sortHandler(event){
      event.preventDefault();
      this.props.sort(this.state.algorithm);
    }
    render(){
      return(
        <div>
          <h1 class='text-center'>SortingAlgorithms</h1>
          <form>
            <label for='algorithm'>
              Algorithm: 
              <select
                id='algorithm'
                class='form-control'
                value={this.state.algorithm}
                onChange={this.algorithmHander}>
                {sortingAlgorithms.map(function(e){
                  return (
                    <option value={e.algorithm}>
                      {e.name}
                    </option>
                  );
                })}
              </select>
            </label>
            <label for='array-length'>
              <input
                id='array-length'
                class='form-control'
                type='number'
                min='10'
                max='10000'
                value={this.state.length}
                onChange={this.lenghtHandler}/>
            </label>
            <button
              id='generate'
              class='btn btn-primary'
              onClick={this.generateHandler}>
              Generate Random Array
            </button>
            <button
              id='sort'
              class='btn btn-primary'
              onClick={this.sortHandler}>
              Sort Array
            </button>
          </form>
        </div>
      )
    }
  }

  class Svg extends React.Component{
    constructor(props){
      super(props);
      this.renderSvg = this.renderSvg.bind(this);
    }
    componenDidUpdate(){
      this.renderSvg();
    }
    componentDidMount(){
      this.renderSvg();
    }
    renderSvg(){
      const svgPadding = 50;
      const svgWidth = 900;
      const svgHeight = 540;

      var svg = d3.select(this.node)
      .attr('width', svgWidth)
      .attr('height', svgHeight);

      //scales
      const xScale = d3.scaleBand()
      .domain([0, this.props.data.length])
      .range([svgPadding, svgWidth-svgPadding]);
      const yScale = d3.scaleLinear()
      .domain(d3.extent(this.props.data))
      .range([svg-svgHeight, svgPadding]);

      svg.selectAll('.bar')
      .data(this.props.data)
      .enter().append('rect')
      .attr('class', 'bar')
      .attr('x', (d, i)=>xScale(i))
      .attr('y', d=>yScale(d))
      .attr('fill', 'black');
    }
    render(){
      return <svg ref={node=>this.node=node}/>
    }
  }

  //ReactRedux
  const Provider = ReactRedux.Provider;
  const connect = ReactRedux.connect;

  function mapStateToProps(state){
    return {
      data: state.array,
      accessCounter: state.accessCounter,
      sortedIndices: state.sortedIndices,
      accessingIndices: state.accessingIndices,
      minIndex: state.minIndex,
      maxIndex: state.maxIndex
    };
  }

  function mapDispatchToProps(dispatch){
    return {
      generateArray: length => dispatch(generateArray(length)),
      sort: algorithm => dispatch(algorithm())
    };
  }

  const FormConnection = connect(mapStateToProps, mapDispatchToProps)(Form);
  const SvgConnection = connect(mapStateToProps, null)(Svg);

  class Wrapper extends React.Component{
    render(){
      return(
        <Provider store={store}>
          <FormConnection/>
          <SvgConnection/>
        </Provider>
      );
    }
  }

  ReactDOM.render(<Wrapper/>, $('body')[0]);
})
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...