为 React Apps 设计和实现 Ant Design Global App Tour。
用户导览是网络应用程序的一项宝贵的可用性功能。它们可让您有效地引导新用户,提供分步指南以帮助他们了解软件。导览还可作为重复任务或高级功能的快速参考。
目标:跨页面导览解决方案
我们的目标是创建一个解决方案,让您能够在 React 应用程序中创建跨多个页面的入门体验。它看起来是这样的:
Ant Design Tour:本地解决方案
Ant Design 提供了 **Tour** 组件来创建交互式导览,但它有一些限制:
以下是官方文档中的一个示例,演示了基本的本地实现:
import React, { useRef, useState } from 'react'; import { EllipsisOutlined } from '@ant-design/icons'; import { Button, Divider, Space, Tour } from 'antd'; const App = () => { const ref1 = useRef(null); const ref2 = useRef(null); const ref3 = useRef(null); const [open, setOpen] = useState(false); const steps = [ { title: 'Upload File', description: 'Put your files here.', target: () => ref1.current }, { title: 'Save', description: 'Save your changes.', target: () => ref2.current }, { title: 'Other Actions', description: 'Click to see other actions.', target: () => ref3.current }, ]; return ( <>} /> setOpen(false)} steps={steps} /> > ); }; export default App;
虽然此实现对于单个页面效果很好,但在 React 应用程序中的跨页面游览场景中它就显得不足了。
以下是我们实现的方法:
预步骤,app.jsx,routes.jsx,routesNames.js:
import { RouterProvider } from "react-router-dom"; import AppRouter from "./routes"; export default function App() { return; }
export const ROUTE_NAMES = { HOME: "/", ABOUT: "/about", };
import AppLayout from "./AppLayout"; import { createBrowserRouter } from "react-router-dom"; import { ROUTE_NAMES } from "./routeNames"; import { Home } from "./components/Home"; import { About } from "./components/About"; import { Result } from "antd"; import {TourProvider} from "./TourContext"; const GetItem = (label, key, icon, to, children = [], type) => { return !to ? { key, icon, children, label, type, } : { key, icon, to, label, }; }; const GetRoute = (path, element, params = null) => { return { path, element, }; }; const WithAppLayout = (Component) =>; export const routeItems = [ GetItem("Home", "home", null, ROUTE_NAMES.HOME), GetItem("About", "about", null, ROUTE_NAMES.ABOUT), ]; const AppRouter = createBrowserRouter([ GetRoute(ROUTE_NAMES.HOME, WithAppLayout( {Component} )), GetRoute(ROUTE_NAMES.ABOUT, WithAppLayout( )), GetRoute( "*", ), ]); export default AppRouter;
步骤 1:设置全球旅游背景
我们使用 React Context 来管理旅游的全局状态,包括活动的旅游步骤。
import React, { createContext, useState, useEffect } from "react"; import { useNavigate } from "react-router-dom"; import { APP_TOURS } from "./steps"; const TourContext = createContext(); export const TourProvider = ({ children }) => { const [isTourActive, setTourActive] = useState(false); const navigate = useNavigate(); useEffect(() => { if (isTourActive) { navigate("/home"); // Redirect to the starting point of the tour } }, [isTourActive, navigate]); return ({children} ); }; export default TourContext;
第 2 步:定义全球游览步骤
我们不使用 React refs,而是使用 `querySelector` 通过自定义 `data-tour-id` 属性动态获取元素。
const getTourStepElement = (id) => document.querySelector(`[data-tour-id="${id}"]`); export const APP_TOURS = { "/home": [ { title: "Upload File", description: "Put your files here.", target: () => getTourStepElement("upload") }, { title: "Save", description: "Save your changes.", target: () => getTourStepElement("save") }, { type: "navigate", to: "/about", title: "About Us", description: "Learn more about us." }, ], "/about": [ { title: "About Us", description: "Here's what we are all about.", target: () => getTourStepElement("about") }, ], };
步骤 3:创建全球游览组件
该组件动态处理跨页面的导航和步骤。
import React, { useContext } from "react"; import { Tour } from "antd"; import { useNavigate } from "react-router-dom"; import TourContext from "./TourContext"; export const GlobalTour = () => { const { isTourActive, steps, setTourActive } = useContext(TourContext); const navigate = useNavigate(); return (setTourActive(false)} steps={steps} onChange={(current) => { const step = steps[current]; if (step.type === "navigate") { navigate(step.to); } }} /> ); };
步骤 4:集成到应用程序布局
游览与布局无缝集成,可从任何页面访问。
import React, { useContext } from "react"; import { Layout, Button } from "antd"; import { Link } from "react-router-dom"; import TourContext from "./TourContext"; import { GlobalTour } from "./GlobalTour"; const { Header, Content, Footer } = Layout; const AppLayout = ({ children }) => { const { setTourActive } = useContext(TourContext); return (); }; export default AppLayout; Home About {children}
第 5 步:添加步骤游览 ID
由于我们的游览跨越多个页面,我们将为想要在步骤中突出显示的每个组件分配 data-tour-id
import { Button, Space } from "antd"; import { EllipsisOutlined } from "@ant-design/icons"; export const Home = () => { return ( <> } /> > ); };
export const About = () => { returnAbout; };