React實(shí)現(xiàn)一個(gè)支持動(dòng)態(tài)插槽的Layout組件
目標(biāo)實(shí)現(xiàn)一個(gè)支持動(dòng)態(tài)注冊(cè)內(nèi)容的插槽組件,思路:提供一個(gè) Context,維護(hù)插槽中注冊(cè)的內(nèi)容,并將其渲染到對(duì)應(yīng)的位置。
思路
- 采用
Context
實(shí)現(xiàn)插槽的注冊(cè)和管理,并提供對(duì)應(yīng)的渲染。 - 采用
自定義Hook
將往插槽注冊(cè)內(nèi)容邏輯抽離封裝 - 提供對(duì)應(yīng)的渲染器將插槽內(nèi)容掛載到React組件樹上
代碼實(shí)現(xiàn)
一、定義 Context
import React, { createContext, useState } from "react"; export const SlotContext = createContext({ slots: { header: [], footer: [], }, registerSlot: (name: "header" | "footer", content: React.ReactNode) => undefined, }); const SlotProvider: React.FC<{ children: React.ReactNode }> = ({ children, }) => { const [slots, setSlots] = useState<{ header: React.ReactNode[]; footer: React.ReactNode[]; }>({ header: [], footer: [], }); // 定義注冊(cè)函數(shù) const register = (name: "header" | "footer", content: React.ReactNode) => { setSlots((pre) => { const newValue = { ...pre }; if (newValue[name]) { newValue[name] = [content]; return newValue; } else { return pre; } }); }; return ( <SlotContext.Provider value={{ slots, registerSlot: register }}> {children} </SlotContext.Provider> ); };
二、封裝 useSlotRegister 自定義 Hook
export function useSlotRegister( name: "header" | "footer", content: React.ReactNode ) { const { registerSlot } = React.useContext(SlotContext); React.useEffect(() => { registerSlot(name, content); // 清除 return () => { registerSlot(name, null); }; }, [name, content]); }
三、將插槽內(nèi)的注冊(cè)的內(nèi)容渲染到 React 組件樹上
渲染器可以根據(jù)自定義處理內(nèi)容進(jìn)行處理,如果您想讓插槽注冊(cè)的內(nèi)容按照優(yōu)先級(jí)進(jìn)行排序,修改注冊(cè)的內(nèi)容(標(biāo)記優(yōu)先級(jí)),渲染器中使用優(yōu)先級(jí)字段進(jìn)行排序然后渲染。以下只是一個(gè)簡(jiǎn)單的渲染實(shí)現(xiàn):
// export const HeaderSlotRenderer: React.FC = () => { const { slots } = React.useContext(SlotContext); return <>{slots.header}</>; }; export const FooterSlotRenderer = () => { const { slots } = React.useContext(SlotContext); return <>{slots.footer}</>; }; export const Layout: React.FC<{ children: React.ReactNode }> = ({ children, }) => { return ( <SlotProvider> {/* render header */} <div className="layout"> <div className="header"> <HeaderSlotRenderer /> </div> <div className="main">{children}</div> <div className="footer"> <FooterSlotRenderer /> </div> </div> </SlotProvider> ); };
/* src/components/Layout.css */ .layout { display: flex; flex-direction: column; min-height: 100vh; } .header { height: 100px; background-color: #333; color: white; display: flex; align-items: center; justify-content: center; } .main { flex: 1; background-color: #f4f4f4; padding: 20px; } .footer { height: 80px; background-color: #333; color: white; display: flex; align-items: center; justify-content: center; }
掛載
import ReactDOM from "react-dom/client"; import { Layout, SlotContext, useSlotRegister } from "./Layout"; const rootElement = document.getElementById("root")!; const root = ReactDOM.createRoot(rootElement); const HeaderTitle = () => { const { registerSlot } = React.useContext(SlotContext); // 手動(dòng) Register registerSlot( "header", <div> <h1>Header</h1> </div> ); return null; }; const FooterDate = () => { // use hook Register useSlotRegister("footer", <div>@2025</div>); return null; }; root.render( <React.StrictMode> <Layout> <HeaderTitle /> <FooterDate /> <div>Layout... main</div> </Layout> </React.StrictMode> );
到此這篇關(guān)于React實(shí)現(xiàn)一個(gè)支持動(dòng)態(tài)插槽的Layout組件的文章就介紹到這了,更多相關(guān)React動(dòng)態(tài)插槽組件內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
在?React?項(xiàng)目中全量使用?Hooks的方法
這篇文章主要介紹了在?React?項(xiàng)目中全量使用?Hooks,使用 Hooks 能為開發(fā)提升不少效率,但并不代表就要拋棄 Class Component,依舊還有很多場(chǎng)景我們還得用到它,本文給大家介紹的非常詳細(xì),需要的朋友可以參考下2022-10-10React?useEffect不支持async?function示例分析
這篇文章主要為大家介紹了React?useEffect不支持async?function示例分析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-07-07關(guān)于React中setState同步或異步問題的理解
相信很多小伙伴們都一直在疑惑,setState 到底是同步還是異步。本文就詳細(xì)的介紹一下React中setState同步或異步問題,感興趣的可以了解一下2021-11-11React 遞歸手寫流程圖展示樹形數(shù)據(jù)的操作方法
這篇文章主要介紹了React 遞歸手寫流程圖展示樹形數(shù)據(jù)的操作方法,本文通過示例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友參考下吧2023-11-11React Hooks 實(shí)現(xiàn)和由來以及解決的問題詳解
這篇文章主要介紹了React Hooks 實(shí)現(xiàn)和由來以及解決的問題詳解,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-01-01