diff --git a/src/components/toast/img.json b/src/components/toast/img.json
new file mode 100644
index 0000000..cfc1e87
--- /dev/null
+++ b/src/components/toast/img.json
@@ -0,0 +1,8 @@
+{
+ "error":
+ "",
+ "success":
+ "",
+ "loading":
+ ""
+}
diff --git a/src/components/toast/index.jsx b/src/components/toast/index.jsx
new file mode 100644
index 0000000..34123b5
--- /dev/null
+++ b/src/components/toast/index.jsx
@@ -0,0 +1,185 @@
+import classNames from 'classnames'
+import statusImg from './img.json'
+
+export default {
+ name: 'Toast',
+ props: {
+ text: {
+ type: String,
+ default: '',
+ },
+ icon: {
+ type: String,
+ default: '',
+ },
+ hasMask: {
+ type: Boolean,
+ default: false,
+ },
+ image: {
+ type: String,
+ default: '',
+ },
+ isOpened: {
+ type: Boolean,
+ default: false,
+ },
+ duration: {
+ type: Number,
+ default: 3000,
+ },
+ status: {
+ type: String,
+ default: '',
+ validator: (val) => ['', 'error', 'loading', 'success'].includes(val),
+ },
+ onClick: {
+ type: Function,
+ default: () => () => {},
+ },
+ onClose: {
+ type: Function,
+ default: () => () => {},
+ },
+ className: {
+ type: [Object, String],
+ default: () => '',
+ },
+ },
+ data() {
+ return {
+ state: {
+ _timer: null,
+ _isOpened: this.isOpened,
+ },
+ }
+ },
+ watch: {
+ isOpened: {
+ immediate: true,
+ handler() {
+ this.handleChange()
+ },
+ },
+ duration() {
+ this.handleChange()
+ },
+ },
+ methods: {
+ setState(newState, fn) {
+ const ks = Object.keys(newState)
+ if (Array.isArray(ks)) {
+ ks.forEach((k) => {
+ if (k in this.state) {
+ this.state[k] = newState[k]
+ }
+ })
+ }
+ typeof fn === 'function' && fn.call(this)
+ },
+ clearTimmer() {
+ if (this._timer) {
+ clearTimeout(this._timer)
+ this._timer = null
+ }
+ },
+
+ makeTimer(duration) {
+ if (duration === 0) {
+ return
+ }
+ this._timer = setTimeout(() => {
+ this.close()
+ }, +duration)
+ },
+
+ close() {
+ const { _isOpened } = this.state
+ if (_isOpened) {
+ this.setState(
+ {
+ _isOpened: false,
+ },
+ this.handleClose
+ )
+ this.clearTimmer()
+ }
+ },
+
+ handleClose(event) {
+ if (typeof this.onClose === 'function') {
+ this.onClose(event)
+ }
+ },
+
+ handleClick(event) {
+ const { onClick, status } = this
+ if (status === 'loading') {
+ return
+ }
+ if (onClick) {
+ return onClick(event)
+ }
+ this.close()
+ },
+
+ handleChange() {
+ const { isOpened, duration } = this
+ if (!isOpened) {
+ this.close()
+ return
+ }
+
+ if (!this.state._isOpened) {
+ this.setState({
+ _isOpened: true,
+ })
+ } else {
+ this.clearTimmer()
+ }
+ this.makeTimer(duration || 0)
+ },
+ },
+ render() {
+ const { _isOpened } = this.state
+ const { customStyle, text, icon, status, image, hasMask } = this
+
+ const realImg = image || statusImg[status] || null
+ const isRenderIcon = !!(icon && !(image || statusImg[status]))
+
+ const bodyClass = classNames('toast-body', {
+ 'at-toast__body--custom-image': image,
+ 'toast-body--text': !realImg && !icon,
+ [`at-toast__body--${status}`]: !!status,
+ })
+
+ const iconClass = classNames('at-icon', {
+ [`at-icon-${icon}`]: icon,
+ })
+
+ return _isOpened ? (
+
+ {hasMask && }
+
+
+ {realImg ? (
+
+
+
+ ) : null}
+ {isRenderIcon && (
+
+
+
+ )}
+ {text && (
+
+ {text}
+
+ )}
+
+
+
+ ) : null
+ },
+}
diff --git a/src/pages/index/index.vue b/src/pages/index/index.vue
index 478d9d6..a57679c 100644
--- a/src/pages/index/index.vue
+++ b/src/pages/index/index.vue
@@ -2,6 +2,11 @@
+
标题
@@ -142,6 +147,7 @@ import Modal from '../../components/modal/index.jsx'
import ModalHeader from '../../components/modal/header/index.jsx'
import ModalContent from '../../components/modal/content/index.jsx'
import ModalAction from '../../components/modal/action/index.jsx'
+import Toast from '../../components/toast/index.jsx'
export default {
name: 'Index',
@@ -165,6 +171,7 @@ export default {
ModalHeader,
ModalContent,
ModalAction,
+ Toast,
},
data() {
return {