
Toast是常用的轻提示弹框,常用于页面loading和提示语弹窗。
本例基于React实现一个随时可调用且不随页面渲染的全局组件。
需求分析
如何使用
首先引入
import Toast from './components/toast'
JSX中事件调用:
<button onClick={() => { Toast.info('普通提示') }}>普通提示</button>JS中方法调用:
Toast.info('普通提示')回调方法:
const hideLoading = Toast.loading('加载中...', 0, () => {
 Toast.success('加载完成')
})
setTimeout(hideLoading, 2000)调用规则:
3个参数:
Toast.info("普通",2000)
Toast.success("成功",1000,() => {
 console.log('回调方法')
}))
Toast.error("错误")
Toast.loading()代码实现
目录结构:

index.js:
import React from 'react'
import ReactDOM from 'react-dom'
import Toast from './toast'
import './toast.css'
function createNotification() {
 const div = document.createElement('div')
 document.body.appendChild(div)
 const notification = ReactDOM.render(<Toast />, div)
 return {
 addNotice(notice) {
 return notification.addNotice(notice)
 },
 destroy() {
 ReactDOM.unmountComponentAtNode(div)
 document.body.removeChild(div)
 }
 }
}
let notification
const notice = (type, content, duration = 2000, onClose) => {
 if (!notification) notification = createNotification()
 return notification.addNotice({ type, content, duration, onClose })
}
export default {
 info(content, duration, onClose) {
 return notice('info', content, duration, onClose)
 },
 success(content = '操作成功', duration, onClose) {
 return notice('success', content, duration, onClose)
 },
 error(content, duration , onClose) {
 return notice('error', content, duration, onClose)
 },
 loading(content = '加载中...', duration = 0, onClose) {
 return notice('loading', content, duration, onClose)
 }
}toast.js:
import React, { Component } from 'react'
class ToastBox extends Component {
 constructor() {
 super()
 this.transitionTime = 300
 this.state = { notices: [] }
 this.removeNotice = this.removeNotice.bind(this)
 }
 getNoticeKey() {
 const { notices } = this.state
 return `notice-${new Date().getTime()}-${notices.length}`
 }
 addNotice(notice) {
 const { notices } = this.state
 notice.key = this.getNoticeKey()
 // notices.push(notice);//展示所有的提示
 notices[0] = notice;//仅展示最后一个提示
 
 this.setState({ notices })
 if (notice.duration > 0) {
 setTimeout(() => {
 this.removeNotice(notice.key)
 }, notice.duration)
 }
 return () => { this.removeNotice(notice.key) }
 }
 removeNotice(key) {
 const { notices } = this.state
 this.setState({
 notices: notices.filter((notice) => {
 if (notice.key === key) {
 if (notice.onClose) setTimeout(notice.onClose, this.transitionTime)
 return false
 }
 return true
 })
 })
 }
 render() {
 const { notices } = this.state
 const icons = {
 info: 'toast_info',
 success: 'toast_success',
 error: 'toast_error',
 loading: 'toast_loading'
 }
 return (
 <div className="toast">
 {
 notices.map(notice => (
 <div className="toast_bg" key={notice.key}>
 <div className='toast_box'>
 <div className={`toast_icon ${icons[notice.type]}`}></div>
 <div className='toast_text'>{notice.content}</div> 
 </div>
 </div>
 ))
 }
 </div>
 )
 }
}
export default ToastBoxtoast.css:
.toast {
 position: fixed;
 left: 0;
 top: 0;
 z-index: 999;
 display: flex;
 flex-direction: column; }
 .toast_bg {
 position: fixed;
 width: 100%;
 height: 100%;
 left: 0;
 top: 0; }
 .toast_box {
 position: relative;
 left: 50%;
 top: 50%;
 width: 2.8rem;
 height: 2rem;
 margin: -1rem -1.4rem;
 background: rgba(0, 0, 0, 0.65);
 border-radius: .1rem;
 color: #fff; }
 .toast_text {
 position: absolute;
 bottom: 16%;
 text-align: center;
 width: 90%;
 margin: 0 5%;
 height: .28rem;
 overflow: hidden;
 text-overflow: ellipsis;
 white-space: nowrap; }
 .toast_icon {
 position: relative;
 left: 50%;
 top: 15%;
 margin: -.4rem;
 width: .8rem;
 height: .8rem; }
 .toast_loading {
 -webkit-animation: loading 1s steps(12, end) infinite;
 animation: loading 1s steps(12, end) infinite;
 background: url("") no-repeat;
 background-size: 100%; }
 .toast_success {
 background: url("") no-repeat;
 background-size: 100%; }
 .toast_error {
 background: url("") no-repeat;
 background-size: 100%; }
 .toast_info {
 background: url("") no-repeat;
 background-size: 100%; }
@-webkit-keyframes loading {
 0% {
 -webkit-transform: rotate3d(0, 0, 1, 0deg);
 transform: rotate3d(0, 0, 1, 0deg); }
 100% {
 -webkit-transform: rotate3d(0, 0, 1, 360deg);
 transform: rotate3d(0, 0, 1, 360deg); } }
@keyframes loading {
 0% {
 -webkit-transform: rotate3d(0, 0, 1, 0deg);
 transform: rotate3d(0, 0, 1, 0deg); }
 100% {
 -webkit-transform: rotate3d(0, 0, 1, 360deg);
 transform: rotate3d(0, 0, 1, 360deg); } }