195 lines
		
	
	
		
			6.5 KiB
		
	
	
	
		
			Vue
		
	
	
			
		
		
	
	
			195 lines
		
	
	
		
			6.5 KiB
		
	
	
	
		
			Vue
		
	
	
| import type { CSSProperties, ExtractPropTypes, PropType } from 'vue';
 | |
| import { defineComponent, ref } from 'vue';
 | |
| import CloseOutlined from '@ant-design/icons-vue/CloseOutlined';
 | |
| import CheckCircleOutlined from '@ant-design/icons-vue/CheckCircleOutlined';
 | |
| import ExclamationCircleOutlined from '@ant-design/icons-vue/ExclamationCircleOutlined';
 | |
| import InfoCircleOutlined from '@ant-design/icons-vue/InfoCircleOutlined';
 | |
| import CloseCircleOutlined from '@ant-design/icons-vue/CloseCircleOutlined';
 | |
| import CheckCircleFilled from '@ant-design/icons-vue/CheckCircleFilled';
 | |
| import ExclamationCircleFilled from '@ant-design/icons-vue/ExclamationCircleFilled';
 | |
| import InfoCircleFilled from '@ant-design/icons-vue/InfoCircleFilled';
 | |
| import CloseCircleFilled from '@ant-design/icons-vue/CloseCircleFilled';
 | |
| import classNames from '../_util/classNames';
 | |
| import PropTypes from '../_util/vue-types';
 | |
| import { getTransitionProps, Transition } from '../_util/transition';
 | |
| import { isValidElement, getPropsSlot } from '../_util/props-util';
 | |
| import { tuple, withInstall } from '../_util/type';
 | |
| import { cloneElement } from '../_util/vnode';
 | |
| import type { NodeMouseEventHandler } from '../vc-tree/contextTypes';
 | |
| import useConfigInject from '../_util/hooks/useConfigInject';
 | |
| 
 | |
| function noop() {}
 | |
| 
 | |
| const iconMapFilled = {
 | |
|   success: CheckCircleFilled,
 | |
|   info: InfoCircleFilled,
 | |
|   error: CloseCircleFilled,
 | |
|   warning: ExclamationCircleFilled,
 | |
| };
 | |
| 
 | |
| const iconMapOutlined = {
 | |
|   success: CheckCircleOutlined,
 | |
|   info: InfoCircleOutlined,
 | |
|   error: CloseCircleOutlined,
 | |
|   warning: ExclamationCircleOutlined,
 | |
| };
 | |
| 
 | |
| const AlertTypes = tuple('success', 'info', 'warning', 'error');
 | |
| 
 | |
| export type AlertType = typeof AlertTypes[number];
 | |
| 
 | |
| export const alertProps = () => ({
 | |
|   /**
 | |
|    * Type of Alert styles, options: `success`, `info`, `warning`, `error`
 | |
|    */
 | |
|   type: PropTypes.oneOf(AlertTypes),
 | |
|   /** Whether Alert can be closed */
 | |
|   closable: PropTypes.looseBool,
 | |
|   /** Close text to show */
 | |
|   closeText: PropTypes.any,
 | |
|   /** Content of Alert */
 | |
|   message: PropTypes.any,
 | |
|   /** Additional content of Alert */
 | |
|   description: PropTypes.any,
 | |
|   /** Trigger when animation ending of Alert */
 | |
|   afterClose: PropTypes.func.def(noop),
 | |
|   /** Whether to show icon */
 | |
|   showIcon: PropTypes.looseBool,
 | |
|   prefixCls: PropTypes.string,
 | |
|   banner: PropTypes.looseBool,
 | |
|   icon: PropTypes.any,
 | |
|   closeIcon: PropTypes.any,
 | |
|   onClose: Function as PropType<NodeMouseEventHandler>,
 | |
| });
 | |
| 
 | |
| export type AlertProps = Partial<ExtractPropTypes<ReturnType<typeof alertProps>>>;
 | |
| 
 | |
| const Alert = defineComponent({
 | |
|   name: 'AAlert',
 | |
|   inheritAttrs: false,
 | |
|   props: alertProps(),
 | |
|   setup(props, { slots, emit, attrs, expose }) {
 | |
|     const { prefixCls, direction } = useConfigInject('alert', props);
 | |
|     const closing = ref(false);
 | |
|     const closed = ref(false);
 | |
|     const alertNode = ref();
 | |
| 
 | |
|     const handleClose = (e: MouseEvent) => {
 | |
|       e.preventDefault();
 | |
| 
 | |
|       const dom = alertNode.value;
 | |
| 
 | |
|       dom.style.height = `${dom.offsetHeight}px`;
 | |
|       // Magic code
 | |
|       // 重复一次后才能正确设置 height
 | |
|       dom.style.height = `${dom.offsetHeight}px`;
 | |
| 
 | |
|       closing.value = true;
 | |
|       emit('close', e);
 | |
|     };
 | |
| 
 | |
|     const animationEnd = () => {
 | |
|       closing.value = false;
 | |
|       closed.value = true;
 | |
|       props.afterClose?.();
 | |
|     };
 | |
| 
 | |
|     expose({ animationEnd });
 | |
|     const motionStyle = ref<CSSProperties>({});
 | |
|     return () => {
 | |
|       const { banner, closeIcon: customCloseIcon = slots.closeIcon?.() } = props;
 | |
| 
 | |
|       let { closable, type, showIcon } = props;
 | |
| 
 | |
|       const closeText = getPropsSlot(slots, props, 'closeText');
 | |
|       const description = getPropsSlot(slots, props, 'description');
 | |
|       const message = getPropsSlot(slots, props, 'message');
 | |
|       const icon = getPropsSlot(slots, props, 'icon');
 | |
| 
 | |
|       // banner模式默认有 Icon
 | |
|       showIcon = banner && showIcon === undefined ? true : showIcon;
 | |
|       // banner模式默认为警告
 | |
|       type = banner && type === undefined ? 'warning' : type || 'info';
 | |
| 
 | |
|       const IconType = (description ? iconMapOutlined : iconMapFilled)[type] || null;
 | |
| 
 | |
|       // closeable when closeText is assigned
 | |
|       if (closeText) {
 | |
|         closable = true;
 | |
|       }
 | |
|       const prefixClsValue = prefixCls.value;
 | |
|       const alertCls = classNames(prefixClsValue, {
 | |
|         [`${prefixClsValue}-${type}`]: true,
 | |
|         [`${prefixClsValue}-closing`]: closing.value,
 | |
|         [`${prefixClsValue}-with-description`]: !!description,
 | |
|         [`${prefixClsValue}-no-icon`]: !showIcon,
 | |
|         [`${prefixClsValue}-banner`]: !!banner,
 | |
|         [`${prefixClsValue}-closable`]: closable,
 | |
|         [`${prefixClsValue}-rtl`]: direction.value === 'rtl',
 | |
|       });
 | |
| 
 | |
|       const closeIcon = closable ? (
 | |
|         <button
 | |
|           type="button"
 | |
|           onClick={handleClose}
 | |
|           class={`${prefixClsValue}-close-icon`}
 | |
|           tabindex={0}
 | |
|         >
 | |
|           {closeText ? (
 | |
|             <span class={`${prefixClsValue}-close-text`}>{closeText}</span>
 | |
|           ) : customCloseIcon === undefined ? (
 | |
|             <CloseOutlined />
 | |
|           ) : (
 | |
|             customCloseIcon
 | |
|           )}
 | |
|         </button>
 | |
|       ) : null;
 | |
| 
 | |
|       const iconNode = (icon &&
 | |
|         (isValidElement(icon) ? (
 | |
|           cloneElement(icon, {
 | |
|             class: `${prefixClsValue}-icon`,
 | |
|           })
 | |
|         ) : (
 | |
|           <span class={`${prefixClsValue}-icon`}>{icon}</span>
 | |
|         ))) || <IconType class={`${prefixClsValue}-icon`} />;
 | |
| 
 | |
|       const transitionProps = getTransitionProps(`${prefixClsValue}-motion`, {
 | |
|         appear: false,
 | |
|         css: true,
 | |
|         onAfterLeave: animationEnd,
 | |
|         onBeforeLeave: (node: HTMLDivElement) => {
 | |
|           node.style.maxHeight = `${node.offsetHeight}px`;
 | |
|         },
 | |
|         onLeave: (node: HTMLDivElement) => {
 | |
|           node.style.maxHeight = '0px';
 | |
|         },
 | |
|       });
 | |
|       return closed.value ? null : (
 | |
|         <Transition {...transitionProps}>
 | |
|           <div
 | |
|             role="alert"
 | |
|             {...attrs}
 | |
|             style={{ ...(attrs.style as Object), ...motionStyle.value }}
 | |
|             v-show={!closing.value}
 | |
|             class={[attrs.class, alertCls]}
 | |
|             data-show={!closing.value}
 | |
|             ref={alertNode}
 | |
|           >
 | |
|             {showIcon ? iconNode : null}
 | |
|             <div class={`${prefixClsValue}-content`}>
 | |
|               {message ? <div class={`${prefixClsValue}-message`}>{message}</div> : null}
 | |
|               {description ? (
 | |
|                 <div class={`${prefixClsValue}-description`}>{description}</div>
 | |
|               ) : null}
 | |
|             </div>
 | |
|             {closeIcon}
 | |
|           </div>
 | |
|         </Transition>
 | |
|       );
 | |
|     };
 | |
|   },
 | |
| });
 | |
| 
 | |
| export default withInstall(Alert);
 |