import { PropType, reactive, defineComponent } from 'vue'; // Utils import { callInterceptor, Interceptor } from '../utils/interceptor'; import { pick, extend, addUnit, truthProp, isFunction, unknownProp, createNamespace, } from '../utils'; import { BORDER_TOP, BORDER_LEFT } from '../utils/constant'; import { popupSharedProps, popupSharedPropKeys } from '../popup/shared'; // Components import { Popup } from '../popup'; import { Button } from '../button'; import { ActionBar } from '../action-bar'; import { ActionBarButton } from '../action-bar-button'; const [name, bem, t] = createNamespace('dialog'); export type DialogTheme = 'default' | 'round-button'; export type DialogAction = 'confirm' | 'cancel'; export type DialogMessage = string | (() => JSX.Element); export type DialogMessageAlign = 'left' | 'center' | 'right'; const popupKeys = [ ...popupSharedPropKeys, 'transition', 'closeOnPopstate', ] as const; export default defineComponent({ name, props: extend({}, popupSharedProps, { title: String, theme: String as PropType, width: [Number, String], message: [String, Function] as PropType, callback: Function as PropType<(action?: DialogAction) => void>, allowHtml: Boolean, className: unknownProp, beforeClose: Function as PropType, messageAlign: String as PropType, closeOnPopstate: truthProp, showCancelButton: Boolean, cancelButtonText: String, cancelButtonColor: String, confirmButtonText: String, confirmButtonColor: String, showConfirmButton: truthProp, closeOnClickOverlay: Boolean, transition: { type: String, default: 'van-dialog-bounce', }, }), emits: ['confirm', 'cancel', 'update:show'], setup(props, { emit, slots }) { const loading = reactive({ confirm: false, cancel: false, }); const updateShow = (value: boolean) => emit('update:show', value); const close = (action: DialogAction) => { updateShow(false); if (props.callback) { props.callback(action); } }; const getActionHandler = (action: DialogAction) => () => { // should not trigger close event when hidden if (!props.show) { return; } emit(action); if (props.beforeClose) { loading[action] = true; callInterceptor({ interceptor: props.beforeClose, args: [action], done() { close(action); loading[action] = false; }, canceled() { loading[action] = false; }, }); } else { close(action); } }; const onCancel = getActionHandler('cancel'); const onConfirm = getActionHandler('confirm'); const renderTitle = () => { const title = slots.title ? slots.title() : props.title; if (title) { return (
{title}
); } }; const renderMessage = (hasTitle: boolean) => { const { message, allowHtml, messageAlign } = props; const classNames = bem('message', { 'has-title': hasTitle, [messageAlign as string]: messageAlign, }); if (allowHtml && typeof message === 'string') { return
; } return (
{isFunction(message) ? message() : message}
); }; const renderContent = () => { if (slots.default) { return
{slots.default()}
; } const { title, message, allowHtml } = props; if (message) { const hasTitle = !!(title || slots.title); return (
{renderMessage(hasTitle)}
); } }; const renderButtons = () => (
{props.showCancelButton && (
); const renderRoundButtons = () => ( {props.showCancelButton && ( )} {props.showConfirmButton && ( )} ); const renderFooter = () => { if (slots.footer) { return slots.footer(); } return props.theme === 'round-button' ? renderRoundButtons() : renderButtons(); }; return () => { const { width, title, theme, message, className } = props; return ( {renderTitle()} {renderContent()} {renderFooter()} ); }; }, });