一个弹窗是不是也挺复杂的,是的,往往看似简单的东西真正做起来很复杂。只有多试验才能出真知。光看不练肯定是不行的。下面我们来把前几节的内容整合一下就大功告成了。
_Model.jsx
/** @jsxImportSource @emotion/react */
import { css, jsx, keyframes } from '@emotion/react'
import React, { useState, useRef, useEffect, useCallback } from 'react';import ModelHeader from './_ModelHeader';
import ModelMask from './_ModelMask';
import ModelContent from './_ModelContent';
import ModelActions from './_ModelActions';
import ModelContainer from './_ModelContianer';
import Draggable from './_Draggable';
import { ModelContext } from './_useModel';
import SThemeProvider from '../STheme/SThemeProvider';
import { useSTheme } from '../STheme/useToggleThemeHook';function Model(props) {const {sizeMode = "sm", //弹窗的大小level = "default", // 弹窗的类型(主要是颜色类型),选项有:normal, error, warning, success, infotitle = "提示", //标题isDark,onClose, //关闭弹窗后的回调enableDragging = true,enableController = true, //是否显示控制按钮content = "暂无弹窗内容", //弹窗内容actions = [ //操作按钮{title: "确定", //按钮标题attention: false, //是否为操作按钮onClick: (setLoading, setTitle, setDisable, onClose) => { onClose(); } //按钮回调},],//功能按钮} = props;const [stateMode, setStateMode] = useState(1); // 弹窗的状态,0: 最小化, 1: 正常, 2: 最大化const theme = useSTheme(); //获取主题console.log(`theme => ${theme}`);return (<SThemeProvider isDark={isDark}><ModelContext.Provider value={{stateMode, // 弹窗的状态,0: 最小化, 1: 正常, 2: 最大化setStateMode, // 设置弹窗的状态sizeMode, //弹窗最大宽度onClose, //关闭弹窗的回调isDark, //是否是暗黑模式level, // 弹窗的类型(主要是颜色类型),选项有:normal, error, warning, success, info}}><ModelMask><DraggableenableDragging={enableDragging && stateMode !== 2 }enableHandler={true}stateMode={stateMode}><ModelContainer><ModelHeaderclassName=".model-handler"title={title}level={level}onClose={onClose}enableController={enableController}/>{content &&<ModelContent>{content}</ModelContent>}{actions && actions.length > 0 &&<ModelActions actions={actions} onClose={onClose} />}</ModelContainer></Draggable></ModelMask></ModelContext.Provider></SThemeProvider>);
};export default Model;
相关的内容前几节已经讲得很通透了。
下面我们来测试吧:
PopModelTest.jsx
import React from 'react';
import Stack from '@mui/material/Stack';
import Button from '@mui/material/Button';
import useModel from '../../framework-kakaer/SModel/_useModel';
import Model from '../../framework-kakaer/SModel/_Model';const longContent = `碧玉妆成一树高”,写柳树给人的总体印象。柳树的形象美在于它那曼长披拂的枝条,一年一度,它长出了嫩绿的新叶,丝丝下垂,在春风吹拂中,有着一种迷人的意态。这里的“碧玉”既可指真实的玉,又暗含“碧玉小家女”(《碧玉歌》)中“碧玉”之意,指小户人家出身的年轻秀美的女子。古典诗词常借用柳树的形象美来形容美人苗条的身段、婀娜的腰肢,但此诗别出新意,翻转过来,将柳树化身为美人。用“碧玉”来比柳实际上有两层意思:一是“碧玉”和柳的颜色有关,“碧”和下句的“绿”是互相生发、互为补充的;二是“碧玉”这个人在人们头脑中留下的是年轻的印象,在古代文学作品里,“碧玉”几乎成了年轻貌美的女子的泛称。用“碧玉”来比柳,人们就会想象到这美人还未到丰容盛鬋的年华,这柳也还是早春稚柳,没有到密叶藏鸦的时候,同时和下文的“细叶”“二月春风”又是有联系的。
“万条垂下绿丝绦”,具体描写那茂密并轻柔下垂的柳枝,它是柳树最具代表性的部分。有了上句的铺垫,这千条万缕的垂丝,也随之变成了美人的裙带。上句的“高”字,衬托出美人婷婷袅袅的风姿;下句的“垂”字,暗示出纤腰在风中款摆。诗中没有“杨柳”和“腰肢”字样,然而这早春的垂柳以及柳树化身的美人,却给写活了。《南史》说刘悛之为益州刺史,献蜀柳数株,“条甚长,状若丝缕”。齐武帝把这些杨柳种植在太昌云和殿前,玩赏不置,说它“风流可爱”。这里把柳条说成“绿丝绦”,可能是暗用这个关于杨柳的典故。但这里的化用,几乎看不出一点痕迹。
“不知细叶谁裁出,二月春风似剪刀。”这两句进一步细描细绘,刻画柳树的嫩叶。每一片树叶都造型别致,纹理细腻,仿佛都是精心裁剪而出。诗人由于惊叹不禁发问:这满树的细叶到底出自哪位高明的裁缝之手?接着找到了答案:原来是大自然的杰作,她手持二月春风这把大剪刀裁出了满树春色。绿叶好比美人衣裙上的花纹和图案,至此,那位美人便形神毕现地跃然纸上了。“二月春风似剪刀”这一新巧的比喻,把视之无形又不可捉摸的春风形象化地描绘出来。春风和剪刀,本来全不相干,它们的相同处只存在于诗人的想象之中。因此,“二月春风似剪刀”既新奇,又能唤起人们丰富的联想。
这首诗立意高远,比喻巧妙,先从大处着眼,然后分部描述,越写越细,把柳树的形神栩栩如生地表现了出来。题目是咏柳,但又不仅仅是咏柳,更是咏春,歌咏自然造化。全诗由“碧玉妆成”引出了“绿丝绦”,“绿丝绦”引出了“谁裁出”,最后,那视之无形的不可捉摸的“春风”,也被用“似剪刀”形象化地描绘了出来。这“剪刀”裁制出嫩绿鲜红的花花草草,给大地换上了新妆,它正是自然活力的象征,是春给予人们美的启示。从“碧玉妆成”到“剪刀”,可以看出诗人一系列艺术构思的过程。诗歌里出现的一连串的形象,是一环紧扣一环的。`;const normalContent = "唐玄宗天宝三载(744),贺知章奉诏告老回乡,百官送行。他坐船经南京、杭州,顺萧绍官河到达萧山县城,越州官员到驿站相迎,然后再坐船去南门外潘水河边的旧宅。此时正是二月早春,柳芽初发,春意盎然,微风拂面。贺知章如脱笼之鸟回到家乡,心情自然格外高兴,即景写下了这首诗。";function PopModelTest() { const alertDefault = useModel({title: "默认提示",content: normalContent,enableController: false,});const alertIno = useModel({title: "信息提示",level: "info",content: normalContent,enableController: false,});const alertSuccess = useModel({title: "成功提示",level: "success",content: normalContent,enableController: false,});const alertWarning = useModel({title: "警告提示",level: "warning",content: normalContent,enableController: false,});const alertError = useModel({title: "错误提示",level: "error",content: normalContent,enableController: false,});const alertLong = useModel({title: "长内容提示",content: longContent,enableController: false,});let initStatus = true;const alertLoading = useModel({title: "异步提交提示",content: normalContent,enableController: false,actions: [{title: "提交",attention: false,onClick: (setLoading, setTitle, setDisable, onClose) => {if (initStatus) {setLoading(true);setTimeout(() => {setLoading(false);setTitle("提交成功, 可关闭");initStatus = false;}, 3000);} else {initStatus = true;onClose();}}}]});const alertMaxMin = useModel({title: "最大化最小化测试",level: "warning",content: normalContent,enableController: true,enableDragging: true,});const alertDrag = useModel({title: "拖动测试",level: "error",content: normalContent,enableController: false,enableDragging: true,});const morActionsAlert = useModel({title: "confirm测试",level: "info",content: normalContent,enableController: false,actions: [{title: "取消",attention: false,onClick: (setLoading, setTitle, setDisable, onClose) => {onClose();}},{title: "确定",attention: true,onClick: (setLoading, setTitle, setDisable, onClose) => {setLoading(true);setTimeout(() => {onClose();}, 3000);}}]});return (<Stack spacing={2}><Buttonvariant='contained'onClick={() => { alertDefault(Model) }}>default 弹窗</Button><Buttonvariant='contained'onClick={() => { alertIno(Model) }}>info 弹窗</Button><Buttonvariant='contained'onClick={() => { alertError(Model) }}>error 弹窗</Button><Buttonvariant='contained'onClick={() => { alertWarning(Model) }}>warning 弹窗</Button><Buttonvariant='contained'onClick={() => { alertSuccess(Model) }}>success 弹窗</Button><Buttonvariant='contained'onClick={() => { alertLong(Model) }}>长文 弹窗</Button><Buttonvariant='contained'onClick={() => { alertLoading(Model) }}>loading 弹窗</Button><Buttonvariant='contained'onClick={() => { morActionsAlert(Model) }}>多Action 弹窗</Button><Buttonvariant='contained'onClick={() => { alertDrag(Model) }}>可拖动弹窗</Button><Buttonvariant='contained'onClick={() => { alertMaxMin(Model) }}>可最大化最小化弹窗</Button></Stack>)
}export default PopModelTest;
本示例中所有的引入目录的位置请根据自己的项目做适当的调整。是不是很不错? 本弹窗系列到此完结。