具有路由的React更新API调用

问题描述

我正在学习反应。我的第一个Web应用程序是一个简单页面,该页面请求有关美国的API Covid-19数据并将其显示为类别。我已使用React Router来显示同一卡组件,并使用dataFinal变量传递不同的数据,该变量保存API检索到的json。如果我要访问的类别之一不是家,例如/positiveIncrease,则会出现错误

Error in /~/index.js (46:255)
Cannot read property 'positiveIncrease' of undefined

我从错误中推断出,在API从json取回数据之前,渲染已发生,并且将其存储在对象中。为了解决该问题,我尝试在Card组件dataFinal.hospitalizedCurrently ? dataFinal.hospitalizedCurrently : "Loading..."

内容部分中使用JSX进行条件渲染。

但是,这似乎不起作用。我还尝试创建一个布尔变量,该变量保存API调用是否已完成,但也没有起作用。

Stackblitz上与该应用程序的链接是:

编辑者:https://stackblitz.com/edit/react-covid-19-us-tracker?file=index.js

查看:https://react-covid-19-us-tracker.stackblitz.io

显示文件:index.js

API.js

import React from 'react';
let loaded=false;
let dataFinal;
 fetch("https://api.covidtracking.com/v1/us/current.json")
 .then(data=>data.json())
 .then(data=>{
     dataFinal=data[0]
  })



export {dataFinal}

Card.js

import React from "react";
import { makeStyles } from "@material-ui/core/styles";
import Card from "@material-ui/core/Card";
import CardActions from "@material-ui/core/CardActions";
import CardContent from "@material-ui/core/CardContent";
import Button from "@material-ui/core/Button";
import Typography from "@material-ui/core/Typography";
import { title,content } from "./Routing.js";

export default function Card(props) {
  return (
    <Card className="card">
      <CardContent>
        <Typography color="textSecondary" gutterBottom>
          {props.title}
        </Typography>
        <Typography variant="h5" component="h2">
          {props.content}
        </Typography>
      </CardContent>
      <CardActions>
        <Button
          onClick={() => {
            window.location = "https://www.cdc.gov/covid-data-tracker/#cases";
          }}
          size="small"
        >
          Learn More
        </Button>
        <br />
      </CardActions>
     { typeof props.content === "string" || props.content instanceof String ? "" : <CardContent>
        <p style={{ color: "red" }}>{repeat(props.content)}</p>
      </CardContent>}
    </Card>
  );
}
export { Content,Title };

const repeat = num => {
  if (typeof num === "string" || num instanceof String) {
    return "";
  }
  if (num < 2000) {
    return "♥ ".repeat(num);
  } else {
    return "♥ ".repeat(2000) + " and " + (num - 2000) + " more";
  }
};

index.html

<title>Covid-19 USA Visualization</title>
<div id="root"></div>

index.js

import React,{ Component} from 'react';
import {render} from 'react-dom';
import './style.css';
import {dataFinal,loaded} from "./API.js"
import Routing from "./Routing.js"
import Card from "./Card.js"
import { browserRouter,Route,Switch } from 'react-router-dom';
class App extends Component {
    render() {
      return(
        <div> <Routing/>
        <main>
            <Switch>
             <Route exact path="/" render={()=> <Card title={''} content={"Navigate through the menus on the top left"} />}> 
                </Route>
                <Route exact path="/positive" render={()=><Card title={'Positive'} content={dataFinal.positive ? dataFinal.positive : "Loading..."} />}> 
                </Route>
                 <Route exact path="/positiveIncrease" render={()=> <Card title={'Positive increase'} content={dataFinal.positiveIncrease ? dataFinal.positiveIncrease : "Loading..."} />}> 
                </Route>
                <Route exact path="/hospitalized" render={()=> <Card title={'Hospitalized'} content={dataFinal.hospitalized ? dataFinal.hospitalized : "Loading..."}/>}> 
                </Route>
                 <Route exact path="/hospitalizedCurrently" render={()=> <Card title={'Hospitalized currently'} content={dataFinal.hospitalizedCurrently ? dataFinal.hospitalizedCurrently : "Loading..." }/>}> 
                </Route>
                  <Route exact path="/dead" render={()=> <Card title={'Total dead'} content={dataFinal.death ? dataFinal.death : "Loading..." }/>}> 
                </Route>
                 <Route exact path="/icu" render={()=> <Card title={'Currently in intensive care unit'} content={dataFinal.inIcuCurrently ? dataFinal.inIcuCurrently : "Loading..." }/>}> 
                                 </Route>
                 <Route exact path="/culminativeicu" render={()=> <Card title={'Culminative in intensive care unit'} content={dataFinal.inIcuCumulative ? dataFinal.inIcuCumulative : "Loading..." }/>}> 
                </Route>
                 <Route exact path="/onventillatorsculminative" render={()=> <Card title={'Culminative on ventillators'} content={dataFinal.onVentilatorCumulative ? dataFinal.onVentilatorCumulative : "Loading..." }/>}> 
                </Route>
                 <Route exact path="/onventillatorscurrently" render={()=> <Card title={'Currently on ventillators'} content={dataFinal.onVentilatorCurrently ? dataFinal.onVentilatorCurrently : "Loading..." }/>}> 
                </Route>
                 <Route exact path="/recovered" render={()=> <Card title={'Recovered'} content={dataFinal.recovered ? dataFinal.recovered : "Loading..." }/>}> 
                </Route>
                 <Route exact path="/deadIncrease" render={()=> <Card title={'Dead Increase'} content={dataFinal.deathIncrease ? dataFinal.deathIncrease : "Loading..." }/>}> 
                 </Route>
                  <Route exact path="/hospitalizedIncrease" render={()=> <Card title={'Hospitalized increase'} content={dataFinal.deathIncrease ? dataFinal.deathIncrease : "Loading..." }/>}> 
                 </Route>
            </Switch>
        </main>
        </div>
        )  
    }
}


render( <browserRouter>
        <App /> 
    </browserRouter>,document.getElementById('root'));

Routing.js

import React from 'react';
import clsx from 'clsx';
import { makeStyles,useTheme } from '@material-ui/core/styles';
import Drawer from '@material-ui/core/Drawer';
import CssBaseline from '@material-ui/core/CssBaseline';
import AppBar from '@material-ui/core/AppBar';
import Toolbar from '@material-ui/core/Toolbar';
import List from '@material-ui/core/List';
import Typography from '@material-ui/core/Typography';
import Divider from '@material-ui/core/Divider';
import IconButton from '@material-ui/core/IconButton';
import MenuIcon from '@material-ui/icons/Menu';
import ChevronLeftIcon from '@material-ui/icons/ChevronLeft';
import ChevronRightIcon from '@material-ui/icons/ChevronRight';
import ListItem from '@material-ui/core/ListItem';
import ListItemIcon from '@material-ui/core/ListItemIcon';
import ListItemText from '@material-ui/core/ListItemText';
import {Content,Title} from "./Card.js"
import { browserRouter,Switch,Link } from 'react-router-dom';
import ReplayIcon from '@material-ui/icons/Replay';
import Box from '@material-ui/core/Box';
import {retrieve} from "./API"
import 'fontsource-roboto';
const drawerWidth = 200;

const useStyles = makeStyles((theme) => ({
  root: {
    display: 'flex',},appBar: {
    transition: theme.transitions.create(['margin','width'],{
      easing: theme.transitions.easing.sharp,duration: theme.transitions.duration.leavingScreen,}),appBarShift: {
    width: `calc(100% - ${drawerWidth}px)`,marginLeft: drawerWidth,transition: theme.transitions.create(['margin',{
      easing: theme.transitions.easing.eaSEOut,duration: theme.transitions.duration.enteringScreen,menuButton: {
    marginRight: theme.spacing(2),hide: {
    display: 'none',drawer: {
    width: drawerWidth,flexShrink: 0,drawerPaper: {
    width: drawerWidth,drawerHeader: {
    display: 'flex',alignItems: 'center',padding: theme.spacing(0,1),// necessary for content to be below app bar
    ...theme.mixins.toolbar,justifyContent: 'flex-end',content: {
    flexGrow: 1,padding: theme.spacing(3),transition: theme.transitions.create('margin',marginLeft: -drawerWidth,contentShift: {
    transition: theme.transitions.create('margin',marginLeft: 0,}));

export default function PersistentDrawerLeft() {
  const classes = useStyles();
  const theme = useTheme();
  const [open,setopen] = React.useState(false);

  const handleDrawerOpen = () => {
    setopen(true);
  };

  const handleDrawerClose = () => {
    setopen(false);
  };

  return (
    <div className={classes.root}>
      <CssBaseline />
      <AppBar
        position="fixed"
        className={clsx(classes.appBar,{
          [classes.appBarShift]: open,})}
      >
        <Toolbar>
          <IconButton
            color="inherit"
            aria-label="Open drawer"
            onClick={handleDrawerOpen}
            edge="start"
            className={clsx(classes.menuButton,open && classes.hide)}
          >
            <MenuIcon />
          </IconButton>
          <Typography variant="h6" Nowrap>
USA Covid-19 Visualization          
</Typography>
        </Toolbar>
      </AppBar>
      <Drawer
        className={classes.drawer}
        anchor="bottom"
        open={open}
        classes={{
          paper: classes.drawerPaper,}}
      >
        <div className={classes.drawerHeader}>
          <IconButton onClick={handleDrawerClose}>
            {theme.direction === 'ltr' ? <ChevronLeftIcon /> : <ChevronRightIcon />}
          </IconButton>
        </div>
        <Divider />
        <List>
        <Link onClick={()=>setopen(false)} style={{textdecoration:"none",color:"black"}} to="/">  <ListItem  button key={"Home"}>
              <ListItemText  primary={"Home"} />
            </ListItem> </Link>
            <Divider />
          <Link onClick={()=>setopen(false)}  style={{textdecoration:"none",color:"black"}} to="/positive">  <ListItem  button key={"Positive"}>
              <ListItemText  primary={"Positive"} />
            </ListItem> </Link>
             <Link onClick={()=>setopen(false)} style={{textdecoration:"none",color:"black"}} to="/positiveIncrease">  <ListItem  button key={"PositiveIncrease"}>
              <ListItemText  primary={"Positive increase"} />
            </ListItem> </Link>
            <Divider />
         <Link onClick={()=>setopen(false)}  style={{textdecoration:"none",color:"black"}}  to="/hospitalized">   <ListItem  button key={"Hospitalized"}>
              <ListItemText primary={"Hospitalized culminative"} />
            </ListItem></Link>
            <Link onClick={()=>setopen(false)}  style={{textdecoration:"none",color:"black"}}  to="/hospitalizedCurrently">  <ListItem  button key={"hospitalizedCurrently"}>
              <ListItemText primary={"Hospitalized currently"} />
            </ListItem></Link>
             <Link onClick={()=>setopen(false)}  style={{textdecoration:"none",color:"black"}}  to="/hospitalizedIncrease">  <ListItem  button key={"hospitalizedIncrease"}>
              <ListItemText primary={"Hospitalized increase"} />
            </ListItem></Link>
                        <Divider />

           <Link onClick={()=>setopen(false)}  style={{textdecoration:"none",color:"black"}}  to="/dead">  <ListItem  button key={"Dead"}>
              <ListItemText primary={"Dead Culminative"} />
            </ListItem></Link>
             <Link onClick={()=>setopen(false)}  style={{textdecoration:"none",color:"black"}}  to="/deadIncrease">  <ListItem  button key={"DeadIncrease"}>
              <ListItemText primary={"Dead increase"} />
            </ListItem></Link>
                        <Divider />
             <Link onClick={()=>setopen(false)}  style={{textdecoration:"none",color:"black"}}  to="/icu">  <ListItem  button key={"Icu"}>
              <ListItemText primary={"Currently in ICU"} />
            </ListItem></Link>
             <Link onClick={()=>setopen(false)}  style={{textdecoration:"none",color:"black"}}  to="/culminativeicu">  <ListItem  button key={"culminativeIcu"}>
              <ListItemText primary={"In ICU culminative"} />
            </ListItem></Link>
                        <Divider />
             <Link onClick={()=>setopen(false)}  style={{textdecoration:"none",color:"black"}}  to="/onventillatorsculminative">  <ListItem  button key={"onventillatorsculminative"}>
              <ListItemText primary={"On ventillators culminative"} />
            </ListItem></Link>
             <Link onClick={()=>setopen(false)}  style={{textdecoration:"none",color:"black"}}  to="/onventillatorscurrently">  <ListItem  button key={"onventillatorscurrently"}>
              <ListItemText primary={"On ventillators currently"} />
            </ListItem></Link>
                        <Divider />
             <Link onClick={()=>setopen(false)}  style={{textdecoration:"none",color:"black"}}  to="/recovered">  <ListItem  button key={"recovered"}>
              <ListItemText primary={"Recovered"} />
            </ListItem></Link>
            <Divider />
            <center><Typography  style={{color:"grey",fontSize:"1vmax",marginTop:"1.5vmax",marginBottom:"1vmax"}} >Created by atanas Bobev</Typography></center>
        </List>
      </Drawer>
      <main
        className={clsx(classes.content,{
          [classes.contentShift]: open,})}
      >
   
      </main>
    </div>
  );
  
}

package.json

{
  "name": "react","version": "0.0.0","private": true,"dependencies": {
    "@fortawesome/fontawesome-svg-core": "1.2.30","@fortawesome/react-fontawesome": "0.1.11","@material-ui/core": "4.11.0","@material-ui/icons": "4.9.1","@types/react": "16.9.46","clsx": "1.1.1","font-awesome": "4.7.0","fontsource-roboto": "^3.0.3","material-ui": "0.20.2","react": "16.13.1","react-dom": "16.13.1","react-router-dom": "5.2.0"
  },"scripts": {
    "start": "react-scripts start","build": "react-scripts build","test": "react-scripts test --env=jsdom","eject": "react-scripts eject"
  },"devDependencies": {
    "react-scripts": "latest"
  }
}

style.css

h1,p {
  font-family: Lato;
}
.card{
  width:90vw;
  margin-left:5vw;
  margin-top:5vw;
}
Link{
  color:black;
  text-decoration:none;
}

如何解决此路由(API)问题?我很高兴您可以建议我在此应用中错过的一些更好的做法。预先谢谢你。

Working section Card element with API data after click

Non-working after reload

解决方法

Ciao,为避免该错误,您可以这样做:

<Route exact path="/positiveIncrease" render={()=> <Card title={'Positive increase'} content={dataFinal ? dataFinal.positiveIncrease ? dataFinal.positiveIncrease : "Loading..." : "Loading..."} />}> 
</Route>

出现该错误是因为在重新加载页面上丢失了dataFinal。将此更改应用到所有路线,应该解决问题。

Here您的代码已修改。

编辑

正如您在评论中所说,这是部分修复(因为如果您重新加载页面,“ Loading ...”仍然停留在该页面,并且不会重新加载数据)。

为了完全解决您的问题,您必须更改一点解决方案:

  1. componentDidMount:例如,当您从“正增长”路线重新加载页面时,您的dataFinal将丢失,并且页面无法显示所需的数据。但是我们知道,每次页面重新加载时,都会触发componentDidMount函数。好的,所以我们可以使用它:

    class App extends Component {
     ...
     componentDidMount(){
       // ok here we can do some to reload data we need
     }
    
  2. API.js:您的解决方案现在仅从dataFinal返回API.js。到现在为止还挺好。但是可以说我想每次都想重新获取数据,而用您的解决方案是不可能的。因此,我们可以这样修改API.js

    export default function(){
      return fetch("https://api.covidtracking.com/v1/us/current.json")
      .then(data=>data.json())
      .then(data=>{
         return data[0];
       })
    } 
    

如您所见,现在API.js导出了一个函数,该函数返回获取的值(不仅是以前实现中获取的值)。

  1. this.state.dataFinal:好的,现在我们有了所需的所有元素。让我们继续使用App.js文件并使用它们。首先,将dataFinal置于App.js状态:

    class App extends Component {
     constructor(props) {
       super(props);
       this.state = {
         dataFinal: {}
       }
     }
     ...
    

然后,使用我们从API.js导入的函数来在每次重新加载组件时更新App.js状态:

componentDidMount(){
  fetchCovidData().then(data => {
    this.setState({dataFinal: data});
  })
}

就是这样!现在,每次用户重新加载页面时,无论用户身在何路,dataFinal都将被重新获取并正确显示。

在此实现中,您还可以避免编写以前建议的条件(因为我将dataFinal初始化为无效对象{})。

Here您的代码已修改。