React實現全局組件:Toast輕提示

Toast是經常使用的輕提示彈框,經常使用於頁面loading和提示語彈窗。
本例基於React實現一個隨時可調用且不隨頁面渲染的全局組件。

如何使用

首先引入css

import Toast from './components/toast'

JSX中事件調用:react

<button onClick={() => { Toast.info('普通提示') }}>普通提示</button>

JS中方法調用:ios

Toast.info('普通提示')

回調方法:web

const hideLoading = Toast.loading('加載中...', 0, () => {
    Toast.success('加載完成')
})
setTimeout(hideLoading, 2000)

調用規則:
3個參數:ajax

  • content 提示內容 string(loading方法爲可選)
  • duration 提示持續時間 number,單位ms(可選)
  • onClose 提示關閉時的回調函數(可選)
Toast.info("普通",2000)
Toast.success("成功",1000,() => {
    console.log('回調方法')
}))
Toast.error("錯誤")
Toast.loading()

代碼實現

目錄結構:
圖片描述axios

  • index.js:對外export接口,設置默認的參數值,全局建立或銷燬Toast的DIV。
  • toast.js:Toast具體顯示的內容及屢次調用Toast時的狀態管理。
  • toast.css:Toast的樣式,費話很少說。

index.js:api

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 ToastBox

toast.css:app

.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); } }

應用例子:axios全局請求加loading特效

import axios from 'axios';
import Toast from '../routes/components/toast';
const mAxios = axios.create({
    baseURL:"/api/h5/",
    timeout: 10000,
    withCredentials: true,
    headers: {
        'X-Requested-With': 'XMLHttpRequest',
        'Content-Type': 'application/x-www-form-urlencoded'
    },
})
window.ajaxNum = 0
mAxios.interceptors.request.use(config => {
    window.ajaxNum ++
    window.ToastLoding = Toast.loading()
    return config
}, error => {
    console.error(`error reject`)
    return Promise.reject(error)
})

mAxios.interceptors.response.use(res => {
    window.ajaxNum--
    checkAjaxConnet()
    let status = res.status
    let data = res.data
    if(status >= 200 && status < 400 && data.code === '0'){
        return data.data
    }
}, error => {
    window.ajaxNum--
    Toast.error('網絡異常',3000)
    return Promise.reject(error)
})

//全部請求完成以後才能夠關閉彈窗,防止屢次請求彈窗抖動
function checkAjaxConnet() {
    if(window.ajaxNum < 1) {
        setTimeout(window.ToastLoding)
    }
}

export default mAxios;
相關文章
相關標籤/搜索