React - передать идентификатор из состояния на клиенте в axios.get на экспресс-сервере - PullRequest
0 голосов
/ 09 октября 2019

Я пытаюсь передать идентификатор из приложения React в axios.get на стороне сервера (Express), поэтому я получаю следующее:

    //get
app.get('/api/siteDashboard', (req, res, next) => {
    axios.get('https://****/api/site?id=3d5a6a8a-60d4-4488-1401-08d74bf02cbf')
    .then(response => res.json(response.data))
        .catch(err => next(err));
})

На стороне клиента у меня есть таблица данных и каждая строкаимеет ссылку на новую страницу. Данные для данных поступают из API. Я хочу иметь возможность щелкнуть ссылку для конкретной строки и получить ее идентификатор из API и открыть новую страницу на основе этого идентификатора.

    <Link to="/Sites/SiteDashboard">
        <MDBIcon icon="fa fa-external-link-alt" size="lg" className='green-text cursor-pointer' />
    </Link>

Я получаю маршрут с сервера, как это вмое приложение React на стороне клиентассылка на стороне клиента. Я не уверен, правильно ли я подхожу к этому, поэтому любой совет приветствуется. Спасибо.

Код реакции ...

import React from 'react'
import { Redirect } from "react-router-dom";
import styled from 'styled-components';
import DataTable, { memoize } from 'react-data-table-component'
import Moment from 'react-moment';
import Spinner from '../../Spinner'
import { BrowserRouter as Route, Link } from "react-router-dom";
import { MDBContainer, MDBCard, MDBCardBody, MDBRow, MDBCol, MDBBtn, MDBIcon } from 'mdbreact';

//import SiteDashboard from './SiteDashboard';
import SiteAdd from './SiteAdd'
import SiteDashboard from './SiteDashboard'

const TextField = styled.input`
  height: 32px;
  width: 300px;
  border-radius: 3px;
  border: 1px solid #e5e5e5;
  padding: 16px;

  &:hover {
    cursor: pointer;
  }
`;


const Filter = ({ onFilter }) => (
    <TextField id="search" type="search" role="search" placeholder="Search Title" onChange={e => onFilter(e.target.value)} />
);

const columns = memoize(clickHandler => [
    {
        name: 'Online',
        selector: 'cloudAccessEnabled',
        sortable: true,
        minWidth: '10px',
        center: true,
        cell: row => (
            <MDBIcon icon="circle"
                className={row.cloudAccessEnabled === true ? 'green-text' : 'red-text'} />
        )
    },
    {
        name: 'Site Ref.',
        selector: 'siteRef',
        sortable: true,
        wrap: true,
        minWidth: '40px',
    },
    {
        name: 'Name',
        selector: 'name',
        sortable: true,
        wrap: true,
        minWidth: '80px',
    },
    {
        name: 'Address Type',
        sortable: true,
        wrap: true,
        minWidth: '100px',
        //dont show this column
        hide: 6000,
        cell: row => (
            <div>
                {row.addresses && row.addresses.map(
                    ({ addressType, id }) => (
                        <div key={id}>
                            <ul>
                                <li>{addressType}</li>
                            </ul>
                        </div>
                    ))}
            </div>
        )
    },
    {
        name: 'Address',
        selector: 'address',
        wrap: true,
        minWidth: '100px',
        cell: row => (
            <div>
                {row.addresses && row.addresses.map(
                    ({ addressType, address1, address2, city, county, postCode, country, id }) => (
                        <div key={id}>
                            <ul>
                                <li><strong>{addressType}</strong></li>
                                <li>{address1}</li>
                                <li>{address2}</li>
                                <li>{city}</li>
                                <li>{county}</li>
                                <li>{postCode}</li>
                                <li>{country}</li>
                            </ul>
                        </div>
                    ))}
            </div>
        )
    },
    {
        name: 'Control Panel',
        selector: 'controlPanel',
        minWidth: '70px',
        wrap: true,
        cell: row => (
            <div>
                {(row.controlpanel == null) ?
                    <div>no CP assigned</div> : row.controlpanel}
            </div>
        )
    },
    {
        name: 'Last Maintenance',
        selector: 'lastMaintenance',
        defaultSortAsc: true,
        sortable: true,
        wrap: true,
        minWidth: '100px',
        //hide: 1030,
        cell: row => (
            <Moment format="DD/MM/YYYY HH:mm">{row.lastMaintenance}</Moment>
        )
    },
    {
        name: 'Next Maintenance',
        selector: 'nextMaintenance',
        sortable: true,
        wrap: true,
        minWidth: '100px',
        //hide: 1030,
        cell: row => (
            <Moment format="DD/MM/YYYY HH:mm">{row.nextMaintenance}</Moment>
        )
    },
    {
        name: 'Latest Result',
        selector: 'latestResult',
        sortable: true,
        minWidth: '90px',
    },
    {
        name: 'Site Dashboard',
        minWidth: '47px',
        center: true,
        cell: row => (
            <MDBIcon onClick={clickHandler} icon="fa fa-external-link-alt" size="lg" className='green-text cursor-pointer' />
        )
    },
]);

class Sites extends React.Component {
    constructor(props) {
        super(props);
        this.state = {
            loading: false,
            sites: [],
            toggledClearRows: false
        }
    }

    // handler for link to Site Dashboard
    handleButtonClick = () => {
        console.log("clicked");
        //<Redirect to={"/Sites/SiteDashboard" + this.state.sites.id} />
    };

    componentDidMount() {

        this.setState({ loading: true });

        //const { match: { params } } =this.props

        fetch('/api/site')
            .then(res => res.json())
            .then(site => {
                this.setState({
                    sites: site.array,
                    //cahnge to true to debug
                    loading: false
                });
                console.log(this.state.sites)
            })
            .catch(error => {
                if (error.response) {
                    console.log(error.responderEnd);
                }
            });
    }

    render() {

        // The row data is composed into your custom expandable component via the data prop
        const ExpandedSection = ({ data }) =>
            <MDBContainer fluid>
                <p className="small mt-2">
                    <strong>Notes: </strong>
                    {data.notes}
                </p>
            </MDBContainer>

        //Spinner
        const CustomLoader = () => (<div><Spinner /></div>);
        const { loading } = this.state;

        return (

            <div>
                <MDBCard className="full-width sites-dt">
                    <MDBCardBody>
                        <MDBRow className="my-4 ml-1">
                            <MDBCol sm="12" md="4" className="text-center text-md-left">
                                <h3>Sites</h3>
                            </MDBCol>
                            <MDBCol sm="12" md="8" className="text-center text-md-right">
                                <Route path="/Sites/SiteAdd" component={SiteAdd} />
                                <Link to={`/Sites/SiteAdd`}>
                                    <MDBBtn outline color="primary">
                                        <MDBIcon icon="plus" /> Add a new Site
                            </MDBBtn>
                                </Link>
                            </MDBCol>
                        </MDBRow>
                        <Link to={`/Sites/SiteDashboard`}>temp link to Site Dashboard <MDBIcon icon="fa fa-external-link-alt" size="lg" className='green-text' />
                        </Link>
                        <MDBRow>
                            <MDBCol>
                                <DataTable className="responsive-table site-headers pagination-align-left"
                                    columns={columns(this.handleButtonClick)}
                                    data={this.state.sites}
                                    defaultSortField="siteRef"
                                    defaultSortAsc={false}
                                    subHeader
                                    //subHeaderComponent={<Filter onFilter={value => setFilterText(value)} />}
                                    compact
                                    pagination
                                    highlightOnHover
                                    expandableRows
                                    expandableRowsComponent={<ExpandedSection />}
                                    progressPending={loading}
                                    progressComponent={<CustomLoader />}
                                />
                            </MDBCol>
                        </MDBRow>
                    </MDBCardBody>
                </MDBCard>

            </div>
        );
    }
}


export default Sites

enter image description here

Ответы [ 2 ]

1 голос
/ 09 октября 2019

вы можете написать отдельную функцию для вызова API, откуда вы получаете идентификаторы / маршруты:

 ComponentDidMount(){
      app.get('/api/siteDashboard', (req, res, next) => {
            // your call

        })
    }

Здесь вы должны внести изменения в маршрут, а также добавить другой маршрут на клиентесо стороны, поэтому, чтобы сделать URL для каждого элемента в таблице.

<Route path='/site/siteDashboard/:id' component={youComponent} />
      //youComponent which shows the details of the item

и где вы отображаете свои элементы в таблице, просто оберните каждый элемент с помощью

<Link to=`/sites/siteDashboard/${this.state.id} />

Теперь, когдав компоненте, где вы показываете детали элемента:

var id = props.match.params.id; // this will get your id from URL
or 
var id = this.props.match.params.id;
axios.get('https://pathc/id?=${id}') //like this

и передайте его вызову API для получения его данных.

0 голосов
/ 09 октября 2019

Можете ли вы также поделиться кодом React? Отредактирую мой ответ с соответствующим кодом

Что вам нужно сделать:

  • Получить все данные таблицы со стороны сервера
  • Создать компонент React для таблицыкоторая будет содержать несколько записей таблицы (еще один компонент React), каждая с идентификатором prop
  • обработчик onClick для каждой кнопки каждой записи таблицы
  • onClick будет обрабатываться компонентом элемента таблицы, который имеетid реквизит, отправка запроса axios для извлечения данных
  • изменение состояния после того, как axios вернет данные элемента
...