123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313 |
- <template>
- <div class="vent-modal">
- <Modal v-bind="getBindValue" @cancel="handleCancel">
- <template #closeIcon v-if="!$slots.closeIcon">
- <ModalClose :canFullscreen="getProps.canFullscreen" :fullScreen="fullScreenRef" :commentSpan="commentSpan" :enableComment="getProps.enableComment" @comment="handleComment" @cancel="handleCancel" @fullscreen="handleFullScreen" />
- </template>
- <template #title v-if="!$slots.title">
- <ModalHeader :helpMessage="getProps.helpMessage" :title="getMergeProps.title" @dblclick="handleTitleDbClick" />
- </template>
- <template #title v-if="!isNoTitle">
- <ModalHeader :helpMessage="getProps.helpMessage" :title="getMergeProps.title" @dblclick="handleTitleDbClick" />
- </template>
- <template #footer v-if="!$slots.footer">
- <ModalFooter v-bind="getBindValue" @ok="handleOk" @cancel="handleCancel">
- <template #[item]="data" v-for="item in Object.keys($slots)">
- <slot :name="item" v-bind="data || {}"></slot>
- </template>
- </ModalFooter>
- </template>
- <!-- update-begin-author:taoyan date:2022-7-18 for: modal弹窗 支持评论 slot -->
- <a-row v-if="getProps.enableComment" class="jeecg-modal-wrapper">
- <a-col :span="24-commentSpan" class="jeecg-modal-content">
- <ModalWrapper
- :useWrapper="getProps.useWrapper"
- :footerOffset="wrapperFooterOffset"
- :fullScreen="fullScreenRef"
- ref="modalWrapperRef"
- :loading="getProps.loading"
- :loading-tip="getProps.loadingTip"
- :minHeight="getProps.minHeight"
- :height="getWrapperHeight"
- :visible="visibleRef"
- :modalFooterHeight="footer !== undefined && !footer ? 0 : undefined"
- v-bind="omit(getProps.wrapperProps, 'visible', 'height', 'modalFooterHeight')"
- @ext-height="handleExtHeight"
- @height-change="handleHeightChange">
- <slot></slot>
- </ModalWrapper>
- </a-col>
- </a-row>
- <!-- update-begin-author:taoyan date:2022-7-18 for: modal弹窗 支持评论 slot -->
- <a-row class="jeecg-modal-wrapper">
- <a-col :span="24-commentSpan" class="jeecg-modal-content">
- <ModalWrapper
- :useWrapper="getProps.useWrapper"
- :footerOffset="wrapperFooterOffset"
- :fullScreen="fullScreenRef"
- ref="modalWrapperRef"
- :loading="getProps.loading"
- :loading-tip="getProps.loadingTip"
- :minHeight="getProps.minHeight"
- :height="getWrapperHeight"
- :visible="visibleRef"
- :modalFooterHeight="footer !== undefined && !footer ? 0 : undefined"
- v-bind="omit(getProps.wrapperProps, 'visible', 'height', 'modalFooterHeight')"
- @ext-height="handleExtHeight"
- @height-change="handleHeightChange">
- <slot></slot>
- </ModalWrapper>
- </a-col>
-
- <a-col :span="commentSpan" class="jeecg-comment-outer">
- <slot name="comment"></slot>
- </a-col>
- </a-row>
- <!-- update-end-author:taoyan date:2022-7-18 for: modal弹窗 支持评论 slot -->
- <template #[item]="data" v-for="item in Object.keys(omit($slots, 'default'))">
- <slot :name="item" v-bind="data || {}"></slot>
- </template>
- </Modal>
- </div>
- </template>
- <script lang="ts">
- import type { ModalProps, ModalMethods } from './typing';
- import { defineComponent, computed, ref, watch, unref, watchEffect, toRef, getCurrentInstance, nextTick } from 'vue';
- import Modal from './components/Modal';
- import ModalWrapper from './components/ModalWrapper.vue';
- import ModalClose from './components/ModalClose.vue';
- import ModalFooter from './components/ModalFooter.vue';
- import ModalHeader from './components/ModalHeader.vue';
- import { isFunction } from '/@/utils/is';
- import { deepMerge } from '/@/utils';
- import { basicProps } from './props';
- import { useFullScreen } from './hooks/useModalFullScreen';
- import { omit } from 'lodash-es';
- import { useDesign } from '/@/hooks/web/useDesign';
- export default defineComponent({
- name: 'BasicModal',
- components: { Modal, ModalWrapper, ModalClose, ModalFooter, ModalHeader },
- inheritAttrs: false,
- props: basicProps,
- emits: ['visible-change', 'height-change', 'cancel', 'ok', 'register', 'update:visible', 'fullScreen'],
- setup(props, { emit, attrs , slots}) {
- const visibleRef = ref(false);
- const propsRef = ref<Partial<ModalProps> | null>(null);
- const modalWrapperRef = ref<any>(null);
- const { prefixCls } = useDesign('basic-modal');
- // modal Bottom and top height
- const extHeightRef = ref(0);
- const modalMethods: ModalMethods = {
- setModalProps,
- emitVisible: undefined,
- redoModalHeight: () => {
- nextTick(() => {
- if (unref(modalWrapperRef)) {
- (unref(modalWrapperRef) as any).setModalHeight();
- }
- });
- },
- };
- const instance = getCurrentInstance();
- if (instance) {
- emit('register', modalMethods, instance.uid);
- }
- // Custom title component: get title
- const getMergeProps = computed((): Recordable => {
- return {
- ...props,
- ...(unref(propsRef) as any),
- };
- });
- //update-begin-author:liusq date:2023-05-25 for:【issues/4856】Modal控件设置 :title = null 无效
- //是否未设置标题
- const isNoTitle = computed(() => {
- //标题为空并且不含有标题插槽
- return !unref(getMergeProps).title && !slots.title;
- });
- //update-end-author:liusq date:2023-05-25 for:【issues/4856】Modal控件设置 :title = null 无效
- const { handleFullScreen, getWrapClassName, fullScreenRef } = useFullScreen({
- modalWrapperRef,
- extHeightRef,
- wrapClassName: toRef(getMergeProps.value, 'wrapClassName'),
- });
- // modal component does not need title and origin buttons
- const getProps = computed((): Recordable => {
- const opt = {
- ...unref(getMergeProps),
- visible: unref(visibleRef),
- okButtonProps: undefined,
- cancelButtonProps: undefined,
- title: undefined,
- };
- return {
- ...opt,
- wrapClassName: unref(getWrapClassName),
- };
- });
- const getBindValue = computed((): Recordable => {
- const attr = {
- ...attrs,
- ...unref(getMergeProps),
- visible: unref(visibleRef),
- wrapClassName: unref(getWrapClassName),
- };
- if (unref(fullScreenRef)) {
- return omit(attr, ['height', 'title']);
- }
- return omit(attr, 'title');
- });
- const getWrapperHeight = computed(() => {
- if (unref(fullScreenRef)) return undefined;
- return unref(getProps).height;
- });
- watchEffect(() => {
- fullScreenRef.value = !!props.defaultFullscreen;
- });
- watchEffect(() => {
- visibleRef.value = !!props.visible;
- });
- watch(
- () => unref(visibleRef),
- (v) => {
- emit('visible-change', v);
- emit('update:visible', v);
- instance && modalMethods.emitVisible?.(v, instance.uid);
- nextTick(() => {
- if (props.scrollTop && v && unref(modalWrapperRef)) {
- (unref(modalWrapperRef) as any).scrollTop();
- }
- });
- },
- {
- immediate: false,
- }
- );
- // 取消事件
- async function handleCancel(e: Event) {
- e?.stopPropagation();
- // 过滤自定义关闭按钮的空白区域
- if ((e.target as HTMLElement)?.classList?.contains(prefixCls + '-close--custom')) return;
- if (props.closeFunc && isFunction(props.closeFunc)) {
- const isClose: boolean = await props.closeFunc();
- visibleRef.value = !isClose;
- return;
- }
- visibleRef.value = false;
- emit('cancel', e);
- }
- /**
- * @description: 设置modal参数
- */
- function setModalProps(props: Partial<ModalProps>): void {
- // Keep the last setModalProps
- propsRef.value = deepMerge(unref(propsRef) || ({} as any), props);
- if (Reflect.has(props, 'visible')) {
- visibleRef.value = !!props.visible;
- }
- if (Reflect.has(props, 'defaultFullscreen')) {
- fullScreenRef.value = !!props.defaultFullscreen;
- }
- }
- function handleOk(e: Event) {
- emit('ok', e);
- }
- function handleHeightChange(height: string) {
- emit('height-change', height);
- }
- function handleExtHeight(height: number) {
- extHeightRef.value = height;
- }
- function handleTitleDbClick(e) {
- if (!props.canFullscreen) return;
- e.stopPropagation();
- handleFullScreen(e);
- }
- //update-begin-author:taoyan date:2022-7-18 for: modal支持评论 slot
- const commentSpan = ref(0);
- watch(()=>props.enableComment, (flag)=>{
- handleComment(flag)
- }, {immediate:true});
- function handleComment(flag){
- if(flag=== true){
- commentSpan.value = 6
- }else{
- commentSpan.value = 0
- }
- }
- //update-end-author:taoyan date:2022-7-18 for: modal支持评论 slot
- // update-begin--author:liaozhiyang---date:20230804---for:【QQYUN-5866】放大行数自适应
- watch(fullScreenRef,(val)=>{
- emit('fullScreen',val);
- });
- // update-begin--author:liaozhiyang---date:20230804---for:【QQYUN-5866】放大行数自适应
- return {
- handleCancel,
- getBindValue,
- getProps,
- handleFullScreen,
- fullScreenRef,
- getMergeProps,
- handleOk,
- visibleRef,
- omit,
- modalWrapperRef,
- handleExtHeight,
- handleHeightChange,
- handleTitleDbClick,
- getWrapperHeight,
- commentSpan,
- handleComment,
- isNoTitle
- };
- },
- });
- </script>
- <style lang="less">
- @ventSpace: zxm;
- /*update-begin-author:taoyan date:2022-7-27 for:modal评论区域样式*/
- .jeecg-comment-outer {
- border-left: 1px solid #f0f0f0;
- .@{ventSpace}-tabs-nav-wrap {
- /* text-align: center;*/
- }
- }
- .jeecg-modal-content {
- > .scroll-container {
- padding: 14px;
- }
- }
- /*update-end-author:taoyan date:2022-7-27 for:modal评论区域样式*/
- // wrapper设为100%,兼容之前写过的弹窗自定义样式
- .jeecg-modal-wrapper,
- .jeecg-modal-content {
- height: 100%;
- }
- </style>
|