React實現(xiàn)一個支持動態(tài)插槽的Layout組件
目標(biāo)實現(xiàn)一個支持動態(tài)注冊內(nèi)容的插槽組件,思路:提供一個 Context,維護(hù)插槽中注冊的內(nèi)容,并將其渲染到對應(yīng)的位置。
思路
- 采用
Context實現(xiàn)插槽的注冊和管理,并提供對應(yīng)的渲染。 - 采用
自定義Hook將往插槽注冊內(nèi)容邏輯抽離封裝 - 提供對應(yīng)的渲染器將插槽內(nèi)容掛載到React組件樹上
代碼實現(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: [],
});
// 定義注冊函數(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)的注冊的內(nèi)容渲染到 React 組件樹上
渲染器可以根據(jù)自定義處理內(nèi)容進(jìn)行處理,如果您想讓插槽注冊的內(nèi)容按照優(yōu)先級進(jìn)行排序,修改注冊的內(nèi)容(標(biāo)記優(yōu)先級),渲染器中使用優(yōu)先級字段進(jìn)行排序然后渲染。以下只是一個簡單的渲染實現(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);
// 手動 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實現(xiàn)一個支持動態(tài)插槽的Layout組件的文章就介紹到這了,更多相關(guān)React動態(tài)插槽組件內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
React?useEffect不支持async?function示例分析
這篇文章主要為大家介紹了React?useEffect不支持async?function示例分析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-07-07
關(guān)于React中setState同步或異步問題的理解
相信很多小伙伴們都一直在疑惑,setState 到底是同步還是異步。本文就詳細(xì)的介紹一下React中setState同步或異步問題,感興趣的可以了解一下2021-11-11
React 遞歸手寫流程圖展示樹形數(shù)據(jù)的操作方法
這篇文章主要介紹了React 遞歸手寫流程圖展示樹形數(shù)據(jù)的操作方法,本文通過示例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友參考下吧2023-11-11
React Hooks 實現(xiàn)和由來以及解決的問題詳解
這篇文章主要介紹了React Hooks 實現(xiàn)和由來以及解決的問題詳解,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-01-01

