使用 Ionic 7 和 React 18,如何访问历史记录对象以在页面之间导航?

Using Ionic 7 and React 18, how do I access the history object to navigate between pages?

提问人:Dave 提问时间:10/24/2023 最后编辑:Dave 更新时间:10/24/2023 访问量:37

问:

我正在使用 Ionic 7 和 React 18。我想构建一个下拉组件,以便在选择选项时,我的 URL 将更改为包含所选选项的 ID。我在我的 App.tsx 文件中设置了这个,

<IonHeader>
  <IonToolbar>
    <IonButtons slot="start">
      <IonButton>
        <IonIcon icon={logoReact} />
      </IonButton>
    </IonButtons>
    <IonTitle>My Title</IonTitle>
    <IonButtons slot="end">
      <IonButton>
        <IonIcon icon={search} />
      </IonButton>
      <IonButton>
        <IonIcon icon={personCircle} />
      </IonButton>
    </IonButtons>
  </IonToolbar>
  <IonToolbar>
    <CategorySelectComponent />
  </IonToolbar>
</IonHeader>
...
   <IonReactRouter>
      <IonRouterOutlet>
        <Route exact path="/home">
          <Home />
        </Route>
        <Route exact path="/">
          <Redirect to="/home" />
        </Route>
        <Route exact path="/cards/:categoryId">
          <CardSlideComponent />
        </Route>
      </IonRouterOutlet>

我的选择组件就是这样

import {
  IonContent,
  IonItem,
  IonLabel,
  IonPage,
  IonSelect,
  IonSelectOption,
} from "@ionic/react";
import React, { useEffect } from "react";
import { useState } from "react";
import axios from "axios";
import CardService from "../services/CardService";
import { useHistory } from "react-router-dom";

const CategorySelectComponent: React.FC = () => {
  const [selectedValue, setSelectedValue] = useState<string | undefined>(
    undefined
  );
  const [options, setOptions] = useState<Category[]>([]);

  const history = useHistory();
  console.log(history); // this is always undefined

  const handleSelectionChange = (event: CustomEvent) => {
    const selectedCategoryId = event.detail.value;
    setSelectedValue(selectedCategoryId);

    // Navigate to the selected category page
    console.log("handle change ...");
    console.log(history);
    history.push(`/cards/${selectedCategoryId}`);
  };

  useEffect(() => {
    CardService.getCategories(setOptions);
  }, []); // Empty dependency array means this effect runs once on component mount

  return (
    <IonPage>
      <IonContent>
        <IonItem>
          <IonLabel>Choose an option:</IonLabel>
          <IonSelect
            value={selectedValue}
            onIonChange={handleSelectionChange}
            interface="popover"
          >
            {options.map((category) => (
              <IonSelectOption key={category.id} value={category.id}>
                {category.name}
              </IonSelectOption>
            ))}
          </IonSelect>
        </IonItem>

        <IonItem>
          <IonLabel>Selected Value:</IonLabel>
          <IonLabel>{selectedValue}</IonLabel>
        </IonItem>
      </IonContent>
    </IonPage>
  );

但“历史”部分始终是“未定义”的。有没有一种更“离子”的方式来做到这一点,而不涉及历史对象?

reactjs ionic-framework react-router 历史

评论

0赞 Drew Reese 10/24/2023
我一直在我的 Ionic 应用程序中使用钩子。你到底想在你的应用中在哪里使用这个钩子?您能否编辑以包含一个完整的最小可重复示例useHistory
0赞 Dave 10/24/2023
我已经稍微缩小了问题范围。我编辑了我的问题以显示带有 useHistory 钩子的组件所在的位置(它位于我的 App.tsx 文件的标头元素中)。我注意到,如果我从“Route”元素中的组件调用“useHistory”钩子,它会加载一个实际对象。所以也许我需要找到另一种方法来重新加载我的 URL?

答:

0赞 Drew Reese 10/24/2023 #1

我最近遇到了同样的问题,当时需要使用钩子来检查是否可以进行后退导航。你只需要在 ReactTree 中将路由器组件(提供路由上下文,例如历史对象)提升到比使用它的任何组件更高的位置。useIonRouter

将组件塞入组件内,以便在其路由上下文中呈现。IonHeaderIonReactRouterCategorySelectComponent

<IonReactRouter>
  ...

  <IonHeader>
    <IonToolbar>
      <IonButtons slot="start">
        <IonButton>
          <IonIcon icon={logoReact} />
        </IonButton>
      </IonButtons>
      <IonTitle>My Title</IonTitle>
      <IonButtons slot="end">
        <IonButton>
          <IonIcon icon={search} />
        </IonButton>
        <IonButton>
          <IonIcon icon={personCircle} />
        </IonButton>
      </IonButtons>
    </IonToolbar>
    <IonToolbar>
      <CategorySelectComponent />
    </IonToolbar>
  </IonHeader>
  ...
   
  <IonRouterOutlet>
    <Route exact path="/home">
      <Home />
    </Route>
    <Route exact path="/">
      <Redirect to="/home" />
    </Route>
    <Route exact path="/cards/:categoryId">
      <CardSlideComponent />
    </Route>
  </IonRouterOutlet>
</IonReactRouter>