繁体   English   中英

在 React/Next.js 中,如何简单地将数据从一个页面的组件传递到另一个页面上的组件?

[英]How can I simply pass data from a component one page to a component on another page in React/Next.js?

总结

我有一个简单的 next.js 应用程序,它有两个页面,index.js 和 results.js。 索引页面有两个位置自动完成字段,在用户从推荐位置的下拉列表中选择位置后,它们会返回位置的地址和 LatLng。 返回后,此数据将存储在索引页面上称为markers的变量中。 当用户单击索引页面上的“获取结果”按钮时,我试图将地址和 LatLng 数据传递到结果页面,以便可以将所选位置显示为 map 上的标记。

问题

我一直在尝试断断续续地这样做大约一个星期,但我不知道如何实现我想要的。 我觉得应该有一个超级简单的方法来做到这一点,但我就是想不通。 我试过这样的事情:

  • 在单击按钮时通过请求参数传递数据,但链接是在页面呈现时根据变量值生成的,而不是在变量更新时更新
  • 将 map 与位置自动完成功能放在同一页面上,并尝试使用 state 自动重新加载 map,这对我来说似乎根本不起作用
  • 我还研究了 Redux 但这似乎太复杂了,无法简单地将对象/它的数据从一个页面传递到另一个页面

我理想的解决方案是将数据从 indext 传递到结果的简单方法。 可能还值得指出的是,我是 React/Next.js 的新手,所以我可能会以错误的方式处理所有这些问题,一旦它起作用,我还将添加一个 API 调用以获取两个 LatLng 点也将显示在结果页面上的 map 上。

代码

index.js(简化)

{/* Import used next and style objects */ }
import Head from 'next/head'
import styles from '../styles/Home.module.css'

{/* Import react-bootstrap */ }
import { Container, Row, Col } from 'react-bootstrap'

{/* Import geosuggest */ }
import Geosuggest from 'react-geosuggest'

{/* Import used components */ }
import NavigationBar from '../components/NavigationBar'
import Footer from '../components/Footer'

{/* Import and use runtime vars for Google API */ }
import getConfig from "next/config"
const { publicRuntimeConfig } = getConfig()
const GoogleApiUrl = `https://maps.googleapis.com/maps/api/js?key=${publicRuntimeConfig.MAPS_JAVASCRIPT_API_KEY}&libraries=places`

export default function Home() {

  const markers = {
    from: {description: "", lat: 0, lng: 0},
    to: {description: "", lat: 0, lng: 0}
  }

  const onFromSuggestSelect = (suggest) => {
    console.log("New FROM location selected: \"" + suggest.description + "\"");
    markers.from.description = suggest.description
    markers.from.lat = suggest.location.lat
    markers.from.lng = suggest.location.lng
  }

  const onToSuggestSelect = (suggest) => {
    console.log("New TO location selected: \"" + suggest.description + "\"");
    markers.to.description = suggest.description
    markers.to.lat = suggest.location.lat
    markers.to.lng = suggest.location.lng
  }

  return (
    <div>
      <Head>
        <title>MyApp</title>
        <meta name="description" content="MyApp" />
        <link rel="icon" href="/favicon.ico" />
        <script src={GoogleApiUrl}></script>
      </Head>

      <NavigationBar/>

      <div className={styles.container}>
        <main>
          <Container>
            <Row className="text-center">
              <Col md={1}></Col>
              <Col md={5}><Geosuggest data-test-id="autocomplete-start-test" placeholder="From..." onSuggestSelect={onFromSuggestSelect} country="gb"  minLength="3" /></Col>
              <Col md={5}><Geosuggest data-test-id="autocomplete-end-test" placeholder="To..." onSuggestSelect={onToSuggestSelect} country="gb"  minLength="3" /></Col>
              <Col md={1}></Col>
            </Row>
          </Container>
        </main>

        <Footer/>

      </div>
    </div>
  )
}

results.js(简化)

{/* Import used next and style objects */ }
import Head from 'next/head'
import styles from '../styles/Home.module.css'

{/* Import used react-bootstrap objects */ }
import { Container } from 'react-bootstrap'

{/* Import used components */ }
import NavigationBar from '../components/NavigationBar'
import Footer from '../components/Footer'
import Map from '../components/Map'

{/* Import and use runtime vars for Google API */ }
import getConfig from "next/config"
const { publicRuntimeConfig } = getConfig()
const GoogleApiUrl = `https://maps.googleapis.com/maps/api/js?key=${publicRuntimeConfig.MAPS_JAVASCRIPT_API_KEY}&libraries=places`

export default function Home() {

    const markers = {
        from: {description: "", lat: 0, lng: 0},
        to: {description: "", lat: 0, lng: 0}
      }

  return (
    <div>
      <Head>
        <title>MyApp</title>
        <meta name="description" content="MyApp" />
        <link rel="icon" href="/favicon.ico" />
        <script src={GoogleApiUrl}></script>
      </Head>

      <NavigationBar/>

      <div className={styles.container}>
        <main>
            <Container>
                {/* MARKERS NEEDS TO BE PASSED IN ON THE LINE BELOW */}
                <Map markers={markers}/> 
            </Container>
        </main>
      </div>
      
      <Footer/>

    </div>
  )
}

Map.js(组件)

import React from "react"
import { GoogleMap, Marker } from "@react-google-maps/api"

export default class Map extends React.Component {
    render() {
        // Set the map to fill its container
        const containerStyle = { width: '500px', height: '500px' }
          
        // Set the map to show the entire UK on initial load
        const center = { lat: 54.476422, lng: -3.397340 }
        const zoom = 6

        // Return the map with the above defaults
        return(
            <GoogleMap mapContainerStyle={containerStyle} center={center} zoom={zoom}>
                <Marker position={{ lat: this.props.markers.from.lat, lng: this.props.markers.from.lng }} />
                <Marker position={{ lat: this.props.markers.to.lat, lng: this.props.markers.to.lng }} />
            </GoogleMap>
        )
    }
}

我认为这里有两个问题:

  1. 如何在两个 NextJS 页面之间传递数据?
  2. 为什么更新suggest值时我的链接没有更新?

有几种不同的方法可以在页面之间传递数据,但查询字符串参数可能是我会使用的方法。 尽管如果可能的话,我不会使用查询字符串中的描述。 您可以通过使用NextJS <Link />组件来做到这一点

但是,您似乎已经尝试过类似的操作,并且链接没有更新。 原因是markers数据的设置方式。 您已将其设置为组件中的 const,它告诉 React 不要监视对此 object 的更改。 onFromSuggestSelect触发时,它会更新 object 属性,然后……什么都没有。 React 认为<Link />没有任何变化,因为 Parent 没有任何变化。

这就是为什么我们有 React State。 如果您将markers const 更改为使用useState挂钩,您应该会发现实现的<link />将在值更改时更新。

一句警告; 如果您之前没有使用过useState钩子,请仔细阅读文档,以便了解如何正确实现它。

我希望这能解决你的问题。

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM