Невозможно обновить таблицу всякий раз, когда происходят новые изменения, не перезагружая страницу с избыточным - PullRequest
0 голосов
/ 27 марта 2020

У меня есть 2 компонента таблицы на одной странице: Resolved Cases и Affected Cases. Мне нужно всякий раз, когда я обновляю строку в таблице Affectd Cases и проверяю ее, она будет отображаться в компоненте Resolved Cases.

ActedCases.tsx:

import React, { useEffect } from 'react';
import { useDispatch, useSelector, shallowEqual } from "react-redux";
import MaterialTable, { Column } from 'material-table';
import TableIcons from "../../common/TableIcons";
import {
  fetchAffectedCasesRequest,
  updateAffectedCaseRequest
} from "./affectedCasesSlice";

import { AppState } from "../../app/rootReducer";

interface Row {
  caseId: number;
  name: string;
  email: string;
  bikeFrameNumber: number;
  caseResolved: number;
}

const AffectedCasesList = () => {
  const dispatch = useDispatch();
  const { officerId, cases } = useSelector((state: AppState) => ({
    officerId: state.affectedCases.officerId,
    cases: state.affectedCases.cases
  }), shallowEqual)

  const data = cases.map(c => ({ ...c }));
  const columns: Column<Row>[] = [
    { title: 'Case Id', field: 'caseId', type: 'numeric', editable: 'never' },
    { title: 'Name', field: 'name', editable: 'never' },
    { title: 'Email', field: 'email', editable: 'never' },
    { title: 'Bike Frame Number', field: 'bikeFrameNumber', type: 'numeric', editable: 'never' },
    { title: 'Case Resolved', field: 'caseResolved', lookup: { 1: 'true', 0: 'false' } }
  ];

  useEffect(() => {
    dispatch(fetchAffectedCasesRequest(officerId))
  }, [officerId, dispatch])

  return (
    <MaterialTable
      icons={TableIcons}
      columns={columns}
      data={data}
      editable={{
        onRowUpdate: (newData, oldData) => new Promise((resolve, reject) => {
          setTimeout(() => {
            resolve();
            if (newData.caseResolved) {
              dispatch(updateAffectedCaseRequest({
                caseId: newData.caseId,
                officerId
              }))
            }
          }, 1000)
        }),
      }}
      options={{
        showTitle: false,
        rowStyle: {
          color: '#ffd600'
        }
      }}
      style={{
        width: '100%',
        backgroundColor: '#111010'
      }}
    />
  );
}

export default AffectedCasesList;

disabledCases.ts:

import { AppThunk } from "../../app/store";
import {
  ReturnedCase,
  resolveCaseApi,
  AffectedCaseToUpdate,
  affectedCasesApi,
} from "../../api/reportedCasesApi";

interface AffectedCasesState {
  officerId: number;
  loading: boolean;
  cases: ReturnedCase[];
  error: string | null;
}
const initialState: AffectedCasesState = {
  officerId: 1,
  loading: false,
  cases: [],
  error: null
}


const affectedCase = createSlice({
  name: 'affectedCase',
  initialState,
  reducers: {
    officerCaseStart: (state) => {
      state.error = null;
    },
    startAffectedCasesFetch: (state) => {
      state.loading = true
    },
    fetchAffectedCasesSuccess: (state, action: PayloadAction<ReturnedCase[]>) => {
      state.cases = action.payload;
      state.loading = false;
    },
    updateAffectedCaseSuccess: (state, action: PayloadAction<number>) => {
      state.cases = state.cases.filter(c => c.caseId !== action.payload)
    },
    updateAffectedCaseError: (state, action: PayloadAction<string>) => {
      state.error = action.payload;
    },
    fetchAffectedCaseError: (state, action: PayloadAction<string>) => {
      state.error = action.payload;
      state.loading = false;
    },
  }
})

export const {
  officerCaseStart,
  startAffectedCasesFetch,
  fetchAffectedCasesSuccess,
  fetchAffectedCaseError,
  updateAffectedCaseError,
  updateAffectedCaseSuccess
} = affectedCase.actions


export default affectedCase.reducer;


export const updateAffectedCaseRequest = (caseToUpdate: AffectedCaseToUpdate): AppThunk => async dispatch => {
  try {
    dispatch(officerCaseStart());
    const updatedCaseId = await resolveCaseApi(caseToUpdate);
    dispatch(updateAffectedCaseSuccess(updatedCaseId))

  } catch (error) {
    let errorMessage = "Internal Server Error";
    if (error.response) {
      errorMessage = error.response.data.message;
    }
    dispatch(updateAffectedCaseError(errorMessage))
  }
}

export const fetchAffectedCasesRequest = (officerId: number): AppThunk => async dispatch => {
  try {
    dispatch(startAffectedCasesFetch())
    const affectedCasesList = await affectedCasesApi(officerId);
    dispatch(fetchAffectedCasesSuccess(affectedCasesList))
  } catch (error) {
    let errorMessage = "Internal Server Error";
    if (error.response) {
      errorMessage = error.response.data.message;
    }
    dispatch(fetchAffectedCaseError(errorMessage))
  }
}

ResolvedCases.tsx:

import React, { useEffect } from 'react';
import { useDispatch, useSelector, shallowEqual } from "react-redux";
import MaterialTable, { Column } from 'material-table';
import TableIcons from "../../common/TableIcons";

import {
  fetchResolvedCasesRequest
} from "./resolvedCasesSlice";

import { AppState } from "../../app/rootReducer";

interface Row {
  caseId: number;
  name: string;
  email: string;
  bikeFrameNumber: number;
}

const ResolvedCasesList = () => {
  const dispatch = useDispatch();
  const { officerId, cases } = useSelector((state: AppState) => ({
    officerId: state.resolvedCases.officerId,
    cases: state.resolvedCases.cases
  }), shallowEqual);

  const data = cases.map(c => ({ ...c }));
  const columns: Column<Row>[] = [
    { title: 'Case Id', field: 'caseId', type: 'numeric' },
    { title: 'Name', field: 'name' },
    { title: 'Email', field: 'email' },
    { title: 'Bike Frame Number', field: 'bikeFrameNumber', type: 'numeric' },
  ];

  useEffect(() => {
    dispatch(fetchResolvedCasesRequest(officerId))
  }, [officerId, dispatch])

  return (
    <MaterialTable
      icons={TableIcons}
      columns={columns}
      data={data}
      options={{
        showTitle: false,
        rowStyle: {
          color: '#ffd600'
        }
      }}
      style={{
        width: '100%',
        backgroundColor: '#111010'
      }}
    />
  );
}

export default ResolvedCasesList;

resolvedCasesSlice.ts:

import { createSlice, PayloadAction } from "@reduxjs/toolkit";
import { AppThunk } from "../../app/store";
import {
  BasicResult,
  resolvedCasesApi
} from "../../api/reportedCasesApi";

interface CasesState {
  officerId: number;
  loading: boolean;
  cases: BasicResult[];
  error: string | null;
}
const initialState: CasesState = {
  officerId: 1,
  loading: false,
  cases: [],
  error: null
}


const resolvedCases = createSlice({
  name: 'resolvedCases',
  initialState,
  reducers: {
    startResolvedCasesFetch: (state) => {
      state.loading = true
    },
    fetchResolvedCasesSuccess: (state, action: PayloadAction<BasicResult[]>) => {
      state.cases = action.payload;
      state.loading = false;
    },
    fetchResolvedCaseError: (state, action: PayloadAction<string>) => {
      state.error = action.payload;
      state.loading = false;
    }
  }
})

export const {
  startResolvedCasesFetch,
  fetchResolvedCasesSuccess,
  fetchResolvedCaseError,
} = resolvedCases.actions


export default resolvedCases.reducer;


export const fetchResolvedCasesRequest = (officerId: number): AppThunk => async dispatch => {
  try {
    dispatch(startResolvedCasesFetch())
    const resolvedCasesList = await resolvedCasesApi(officerId);
    dispatch(fetchResolvedCasesSuccess(resolvedCasesList));
  } catch (error) {
    let errorMessage = "Internal Server Error";
    if (error.response) {
      errorMessage = error.response.data.message;
    }
    dispatch(fetchResolvedCaseError(errorMessage))
  }
}

Компонент ResolvedCases обновляется только при обновлении страницы sh и это из-за:

useEffect(() => {
    dispatch(fetchResolvedCasesRequest(officerId))
  }, [officerId, dispatch])

Есть ли какое-либо решение обновить таблицу ResolvedCases без обновления sh страницы?

1 Ответ

0 голосов
/ 27 марта 2020

useEffect с диспетчеризацией должно запускаться каждый раз, когда обновляется строка в затронутых случаях. Так, например, вы можете отправить действие с некоторым revisionId, когда это произойдет, и переписать свой useEffect следующим образом:

  const revisionIdPrev = useRef(props.revisionId)

  useEffect(() => {
    if (revisionId !== revisionIdPrev.current) {
      dispatch(fetchResolvedCasesRequest(officerId))
      revisionIdPrev.current = revisionId
    }
  }, [officerId, dispatch, revisionId])
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...