Heim >Web-Frontend >View.js >Lassen Sie uns darüber sprechen, wie Sie mit Vue3+Hook Popup-Komponenten schneller und effizienter schreiben können

Lassen Sie uns darüber sprechen, wie Sie mit Vue3+Hook Popup-Komponenten schneller und effizienter schreiben können

青灯夜游
青灯夜游nach vorne
2022-12-28 20:55:273171Durchsuche

Lassen Sie uns darüber sprechen, wie Sie mit Vue3+Hook Popup-Komponenten schneller und effizienter schreiben können

Warum haben Sie diese Idee?

Im Entwicklungsprozess des Management-Backends sind zu viele Pop-up-Business-Pop-ups beteiligt, von denen die häufigsten „XX-Daten hinzufügen“ und „XX-Daten bearbeiten“ sind „, „XX detaillierte Daten anzeigen“ und andere Popup-Fenstertypen sind am häufigsten. [Verwandte Empfehlungen: vuejs-Video-Tutorial, Web-Front-End-Entwicklung]

Viele der Codes für diese Popup-Komponenten sind gleich, z. B. Komponentenstatus, Methoden im Zusammenhang mit Formularkomponenten ...

Also , I einfach Die sekundäre Kapselung und Hooks der Dialog-Komponente reduzieren einige wiederholte CodesDialog组件进行的二次封装和hooks,减少了一些重复的代码

要封装什么

如果是普通弹窗使用的话,直接使用el-dialog组件已经足够了

但我还是一个比较爱折腾的人,我们先看看官方dialog文档有什么可以添加的功能

...

大概看了一下,我打算封装一下功能

  • 提供全屏操作按钮(右上角)
  • 默认提供“确认”,“关闭”按钮
  • 内部添加Loading效果

封装Dialog

确定了要封装的功能之后,先来一个简单的dialog组件。

把双向绑定处理一下,这样外部就可以直接通过v-model直接控制弹窗了。

<template>
    <el-dialog :model-value="props.modelValue"></el-dialog>
</template>
<script setup>
interface PropsType {
  modelValue?: boolean;
}

const props = withDefaults(defineProps<PropsType>(), {
  modelValue: false,
});

const emits = defineEmits<{
  (e: "update:modelValue"): void;
}>();
</script>

header

这里使用到图标库@element-plus/icons-vue

如没有安装,请执行npm install @element-plus/icons-vue

使用el-dialog提供的header插槽,将全屏图表和关闭图标放置到右上角中。给el-dialog传递show-close属性关闭默认图标。

<template>
  <el-dialog :model-value="props.modelValue" :show-close="false">
    <template #header>
      <div>
        <span>{{ props.title }}</span>
      </div>
      <div>
        <el-icon><FullScreen /></el-icon>
        <el-icon><Close /></el-icon>
      </div>
    </template>
  </el-dialog>
</template>
<script setup>
import { FullScreen, Close } from "@element-plus/icons-vue";
</script>
<style scoped>
// 处理样式
:deep(.el-dialog__header) {
  border-bottom: 1px solid #eee;
  display: flex;
  padding: 12px 16px;
  align-items: center;
  justify-content: space-between;
  margin: 0;
}
.dialog-title {
  line-height: 24px;
  font-size: 18px;
  color: #303133;
}
.btns {
  display: flex;
  align-items: center;
  i {
    margin-right: 8px;

    font-size: 16px;
    cursor: pointer;
  }
  i:last-child {
    margin-right: 0;
  }
}
</style>

弹窗的标题文字内容通过props进行传递,默认为空(''

<script lang="ts" setup>
interface PropsType {
  // 忽略之前的代码
  title?: string;
}

const props = withDefaults(defineProps<PropsType>(), {
  title: "",
});

</script>

我们看看现在头部的效果(这里没传入标题,默认为''

Lassen Sie uns darüber sprechen, wie Sie mit Vue3+Hook Popup-Komponenten schneller und effizienter schreiben können

现在这个按钮只有样式效果,还没有写上对应的功能 ~

给他们先绑定上对应的事件和指令

<template>
    <el-dialog
    :model-value="props.modelValue"
    :show-close="false"
    :fullscreen="attrs?.fullscreen ?? isFullscreen"
    >
        <template #header>
        <div>
            <span class="dialog-title">{{ props.title }}</span>
        </div>
        <div class="btns">
            <el-icon v-if="isFullScreenBtn" @click="handleFullscreen"
            ><FullScreen
            /></el-icon>
            <el-icon @click="handleClose"><Close /></el-icon>
        </div>
        </template>
    </el-dialog>
</template>
<script setup lang="ts">
import { FullScreen, Close } from "@element-plus/icons-vue";

interface PropsType {
  title?: string;
  modelValue?: boolean;
  hiddenFullBtn?: boolean;
}

const props = withDefaults(defineProps<PropsType>(), {
  title: "",
  modelValue: false,
  hiddenFullBtn: false,
});

const emits = defineEmits<{
  (e: "update:modelValue"): void;
  (e: "close"): void;
}>();

// 当前是否处于全屏状态
const isFullscreen = ref(false);
// 是否显示全屏效果图标
const isFullScreenBtn = computed(() => {
  if (props.hiddenFullBtn) return false;
  if (attrs?.fullscreen) return false;
  return true;
});

// 开启、关闭全屏效果
const handleFullscreen = () => {
  if (attrs?.fullscreen) return;
  isFullscreen.value = !isFullscreen.value;
};

// 关闭弹窗时向外部发送close事件
const handleClose = () => {
  emits("close");
};
</script>

再点击下全屏图标看看效果怎么样

Lassen Sie uns darüber sprechen, wie Sie mit Vue3+Hook Popup-Komponenten schneller und effizienter schreiben können

NICE 头部功能也就完成了

Footer

接下来,再处理下底部内容,默认提供两个按钮,分别是“确定”和“关闭”,这个名称也是可以通过props属性修改的。

两个按钮绑定点击事件,向外发送不同的事件。

<template>
  <div class="">
    <el-dialog
      v-bind="attrs"
      :model-value="props.modelValue"
      :show-close="false"
      :fullscreen="attrs?.fullscreen ?? isFullscreen"
    >
      <template #footer>
        <!-- 如果没有提供其他footer插槽,就使用默认的 -->
        <span v-if="!slots.footer" class="dialog-footer">
          <el-button type="primary" @click="handleConfirm">{{
            props.confirmText
          }}</el-button>
          <el-button @click="handleClose">{{ props.cancelText }}</el-button>
        </span>
        <!-- 使用传入进来的插槽 -->
        <slot v-else name="footer"></slot>
      </template>
    </el-dialog>
  </div>
</template>
<script setup lang="ts">
import { useSlots } from "vue";
// 获取插槽
const slots = useSlots();
interface PropsType {
    title?: string;
    width?: string | number;
    isDraggable?: boolean;
    modelValue?: boolean;
    hiddenFullBtn?: boolean;
    confirmText?: string;
    cancelText?: string;
}

const props = withDefaults(defineProps<PropsType>(), {
    title: "",
    isDraggable: false,
    modelValue: false,
    hiddenFullBtn: false,
    confirmText: "确认",
    cancelText: "关闭",
});
const handleClose = () => {
    emits("close");
};
const handleConfirm = () => {
    emits("confirm");
};
</script>

Lassen Sie uns darüber sprechen, wie Sie mit Vue3+Hook Popup-Komponenten schneller und effizienter schreiben können

又搞定了一部分了,就剩下Content了 ~

Content

弹窗内容通过默认插槽的方式传入进来,在外层的div元素上添加v-loading标签,实现加载态。

如果你想整个弹窗实现loading效果,请把v-loading移到最外层元素即可。 注意不能是el-dialog元素上,否则无法实现 可能是el-dialog使用了teleport组件,导致v-loading无法正常工作。 等有空研究一下 ~

<template>
  <div class="">
    <el-dialog
      v-bind="attrs"
      :model-value="props.modelValue"
      :show-close="false"
      :fullscreen="attrs?.fullscreen ?? isFullscreen"
    >
        <div class="content" v-loading="props.loading">
            <slot></slot>
        </div>
    </el-dialog>
  </div>
</template>
<script lang="ts" setup>
interface PropsType {
  loading?: boolean;
}

const props = withDefaults(defineProps<PropsType>(), {
  loading: false,
});

</script>

试试看中间的loading效果

Lassen Sie uns darüber sprechen, wie Sie mit Vue3+Hook Popup-Komponenten schneller und effizienter schreiben können

剩下一些细节处理

el-dialog组件提供了很多个props属性供用户选择,但我们现在封装的dialog组件只使用到了一小部分props属性。当用户想要使用其他的props属性时该怎么办?

例如使用width属性时,难道要在我们封装的组件中接收props.width再传递给<el-dialog :width="props.width"></el-dialog>组件吗?

不不不,还有另外一种方法,还记得刚刚在做全屏操作的时候使用到的useAttrs辅助函数吗

它可以获取当前组件传递进来的属性。有了这个方法之后,再配合并即可将外部传递进来的函数再传递到el-dialog组件上面啦

<el-dialog
    v-bind="attrs"
    :model-value="props.modelValue"
    :show-close="false"
    :fullscreen="attrs?.fullscreen ?? isFullscreen"
    :before-close="handleClose"
>
    <!-- 忽略其他代码 -->
</el-dialog>

为了避免内部传递的props被覆盖掉,v-bind="attrs"需要放在最前面

在使用时,可能会给before-close属性传递一个函数,但到了后面被内部的handleClose

Was zu kapseln ist🎜Wenn Wenn Es wird für gewöhnliche Popup-Fenster verwendet. Es reicht aus, die el-dialog-Komponente direkt zu verwenden. Aber ich bin immer noch eine Person, die gerne herumspielt. Werfen wir einen Blick auf die offizielle dialog Dokumentieren Sie zuerst, welche Funktionen hinzugefügt werden können rechte Ecke)

  • Stellen Sie standardmäßig die Schaltfläche „Bestätigen“ bereit.
  • Fügen Sie den Ladeeffekt intern hinzu
  • Encapsulate Dialog🎜🎜Confirm to encapsulate Nach der Funktion beginnen wir mit einer einfachen dialog-Komponente. 🎜🎜 Verarbeiten Sie die bidirektionale Bindung so, dass die Außenseite das Popup-Fenster direkt über v-model steuern kann. 🎜
    const handleClose = () => {
      if (
        Reflect.has(attrs, "before-close") &&
        typeof attrs["before-close"] === "function"
      ) {
        attrs["before-close"]();
      }
      emits("close");
    };

    header🎜
    🎜Hier wird die Icon-Bibliothek @element-plus/icons-vue verwendet🎜
    🎜Wenn diese nicht installiert ist, bitte Führen Sie npm install @element-plus/icons-vue aus🎜
    🎜Verwenden Sie den von el-dialog bereitgestellten header-Slot, um das Vollbilddiagramm und das Schließsymbol zu platzieren in der oberen rechten Ecke. Übergeben Sie das Attribut show-close an el-dialog, um das Standardsymbol zu schließen. 🎜
    import { ref } from "vue";
    
    export default function useDialog() {
      const visible = ref(false);
      const loading = ref(false);
      const openDialog = () => (visible.value = true);
      const closeDialog = () => (visible.value = false);
      const openLoading = () => (loading.value = true);
      const closeLoading = () => (loading.value = false);
      return {
        visible,
        loading,
        openDialog,
        closeDialog,
        openLoading,
        closeLoading,
      };
    }
    🎜Der Inhalt des Titeltextes des Popup-Fensters wird über props weitergeleitet und der Standardwert ist leer ('') 🎜
    <template>
    <el-button @click="openDialog1">普通弹窗</el-button>
    <DialogCmp
      title="DialogCmp1"
      :hiddenFullBtn="true"
      v-model="visible1"
      @confirm="handleConfirm"
      @close="handleClose"
    >
      <h3>DialogCmp1</h3>
    </DialogCmp>
    </template>
    <script setup lang="ts">
    import useDialog from "./components/useDialog";
    import DialogCmp from "./components/Dialog.vue";
    
    const {
      visible: visible1,
      openDialog: openDialog1,
      closeDialog: closeDialog1,
    } = useDialog();
    </script>
    🎜Werfen wir einen Blick darauf Wirkung des aktuellen Kopfes (hier nicht übergeben) Geben Sie den Titel ein, der Standard ist '')🎜🎜Lassen Sie uns darüber sprechen, wie Sie mit Vue3+Hook Popup-Komponenten schneller und effizienter schreiben können🎜🎜Jetzt hat diese Schaltfläche nur noch Stileffekte und die entsprechende Funktion wurde noch nicht geschrieben~🎜🎜Binden Sie die entsprechenden Ereignisse und Anweisungen für sie zuerst🎜
    export enum MODE {
      ADD,  EDIT,
    }
    🎜 Klicken Sie dann auf das Vollbildsymbol. Mal sehen, wie es funktioniert🎜🎜Lassen Sie uns darüber sprechen, wie Sie mit Vue3+Hook Popup-Komponenten schneller und effizienter schreiben können🎜🎜NICE Die Header-Funktion ist jetzt abgeschlossen🎜

    Footer🎜🎜Als nächstes verarbeiten Sie den unteren Inhalt werden standardmäßig bereitgestellt, nämlich „OK“ und „Close“. Dieser Name kann auch über das Attribut props geändert werden. 🎜🎜Zwei Schaltflächen binden Klickereignisse und senden verschiedene Ereignisse aus. 🎜
    import { ref } from "vue";
    import { MODE } from "./types";
    export default function useDialogState() {
      const mode = ref<MODE>(MODE.ADD);
      const visible = ref(false);
      const updateMode = (target: MODE) => {
        mode.value = target;
      };
      return { mode, visible, updateMode };
    }
    🎜Lassen Sie uns darüber sprechen, wie Sie mit Vue3+Hook Popup-Komponenten schneller und effizienter schreiben können🎜🎜 Ein weiterer Teil ist erledigt, nur der Inhalt bleibt übrig~🎜

    Content🎜🎜Der Popup-Inhalt wird über den Standard-Slot übergeben und der äußere div hinzugefügt <code>v-loading-Tag zum Code>-Element hinzufügen, um den Ladezustand zu erreichen. 🎜
    🎜Wenn Sie möchten, dass das gesamte Popup-Fenster den Ladeeffekt erzielt, verschieben Sie das V-Laden bitte auf das äußerste Element. Beachten Sie, dass es sich nicht auf dem El-Dialog-Element befinden darf, da es sonst nicht implementiert werden kann. Es kann sein, dass el-dialog die Teleport-Komponente verwendet, was dazu führt, dass das Laden von V-Dateien nicht richtig funktioniert. Studieren Sie es, wenn Sie Zeit haben~🎜
    import { FormInstance } from "element-plus";
    import { Ref, ref } from "vue";
    import { MODE } from "./types";
    import useDialogState from "./useDialogState";
    
    export default function useDialogFn(
      formInstance: Ref<FormInstance>
    ) {
      const { visible, mode, updateMode } = useDialogState();
    
      const closeDialog = () => {
        formInstance.value.resetFields();
        visible.value = false;
      };
      const openDialog = (target: MODE) => {
        updateMode(target);
        visible.value = true;
      };
      return { visible, mode, openDialog, closeDialog };
    }
    🎜Versuchen Sie, den loading-Effekt in der Mitte zu sehen🎜🎜Lassen Sie uns darüber sprechen, wie Sie mit Vue3+Hook Popup-Komponenten schneller und effizienter schreiben können🎜

    Es sind noch einige Details zu klären 🎜🎜 in el-dialogDie Komponente stellt viele props-Attribute zur Auswahl für Benutzer bereit, aber die dialog-Komponente, die wir derzeit kapseln, verwendet nur einen kleinen Teil davon die Eigenschaft props. Was sollte der Benutzer tun, wenn er andere props-Eigenschaften verwenden möchte? 🎜🎜Müssen wir beispielsweise bei Verwendung des width-Attributs props.width in unserer gekapselten Komponente empfangen und es dann an <el-dialog :width="props.width"></el-dialog> Komponente?🎜🎜Nein, nein, es gibt einen anderen Weg. Erinnern Sie sich an die Hilfsfunktion useAttrs, die Sie gerade beim Vollbildbetrieb verwendet haben?🎜🎜Sie kann den Strom abrufen Komponentenübertragung Kommen Sie in Eigenschaften. Mit dieser Methode können Sie dann zusammenarbeiten und die von außen übergebenen Funktionen an die el-dialog-Komponente übergeben🎜
    <template>
      <Dialog
        :before-close="customClose"
        @confirm="confirm"
        v-model="visible"
        :title="mode == MODE.ADD ? &#39;添加数据&#39; : &#39;编辑信息&#39;"
        :confirm-text="mode == MODE.ADD ? &#39;添加&#39; : &#39;修改&#39;"
      >
        <el-form
          label-width="100px"
          :model="formData"
          ref="formDataRef"
          style="max-width: 460px"
          :rules="rules"
        >
          <el-form-item label="姓名" prop="name">
            <el-input v-model="formData.name" />
          </el-form-item>
          <el-form-item label="年龄" prop="age">
            <el-input v-model="formData.age" />
          </el-form-item>
          <el-form-item label="手机号码" prop="mobile">
            <el-input v-model="formData.mobile" />
          </el-form-item>
        </el-form>
      </Dialog>
    </template>
    <script setup>
    import { ElMessage, FormInstance } from "element-plus";
    import { Ref, ref } from "vue";
    import Dialog from "./Dialog.vue";
    import { MODE } from "./types";
    import useDialogWithForm from "./useDialogWithForm";
    
    const rules = {
      name: {
        type: "string",
        required: true,
        pattern: /^[a-z]+$/,
        trigger: "change",
        message: "只能是英文名称哦",
        transform(value: string) {
          return value.trim();
        },
      },
      age: {
        type: "string",
        required: true,
        pattern: /^[0-9]+$/,
        trigger: "change",
        message: "年龄只能是数字哦",
        transform(value: string) {
          return value.trim();
        },
      },
      mobile: {
        type: "string",
        required: true,
        pattern:
          /^(?:(?:\+|00)86)?1(?:(?:3[\d])|(?:4[5-79])|(?:5[0-35-9])|(?:6[5-7])|(?:7[0-8])|(?:8[\d])|(?:9[189]))\d{8}$/,
        trigger: "change",
        message: "请输入正确的手机号码",
        transform(value: string) {
          return value.trim();
        },
      },
    };
    
    interface FromDataType {
      name: string;
      age: string;
      mobile: string;
    }
    
    const formDataRef = ref<FormInstance | null>(null);
    
    let formData = ref<FromDataType>({
      name: "",
      age: "",
      mobile: "",
    });
    
    const { visible, closeDialog, openDialog, mode } = useDialogWithForm(
      formDataRef as Ref<FormInstance>
    );
    const confirm = () => {
      if (!formDataRef.value) return;
      formDataRef.value.validate((valid) => {
        if (valid) {
          console.log("confirm");
          ElMessage({
            message: "提交成功",
            type: "success",
          });
          closeDialog();
        }
      });
    };
    
    const customClose = () => {
      ElMessage({
        message: "取消提交",
        type: "info",
      });
      closeDialog();
    };
    defineExpose({
      closeDialog,
      openDialog,
    });
    </script>
    <style scoped></style>
    🎜Um zu verhindern, dass die intern übergebenen Requisiten überschrieben werden, v-bind="attrs" muss vorne platziert werden🎜
    🎜Bei Verwendung kann eine Funktion an das Attribut before-close übergeben werden, dies wird jedoch der Fall sein Die interne Methode handleClose wird überschrieben. 🎜

    解决方案是在handleClose函数中,获取attrs.['before-close']属性,如果类型是函数函数,先执行它。

    const handleClose = () => {
      if (
        Reflect.has(attrs, "before-close") &&
        typeof attrs["before-close"] === "function"
      ) {
        attrs["before-close"]();
      }
      emits("close");
    };

    有关于el-dialog组件的封装就到这里了

    封装hooks

    利用Vue composition Api再封装一下在使用el-dialog组件状态的管理hook

    useDialog

    简单处理显示和加载态开关的hook

    import { ref } from "vue";
    
    export default function useDialog() {
      const visible = ref(false);
      const loading = ref(false);
      const openDialog = () => (visible.value = true);
      const closeDialog = () => (visible.value = false);
      const openLoading = () => (loading.value = true);
      const closeLoading = () => (loading.value = false);
      return {
        visible,
        loading,
        openDialog,
        closeDialog,
        openLoading,
        closeLoading,
      };
    }

    useDialog Demo

    Lassen Sie uns darüber sprechen, wie Sie mit Vue3+Hook Popup-Komponenten schneller und effizienter schreiben können

    <template>
    <el-button @click="openDialog1">普通弹窗</el-button>
    <DialogCmp
      title="DialogCmp1"
      :hiddenFullBtn="true"
      v-model="visible1"
      @confirm="handleConfirm"
      @close="handleClose"
    >
      <h3>DialogCmp1</h3>
    </DialogCmp>
    </template>
    <script setup lang="ts">
    import useDialog from "./components/useDialog";
    import DialogCmp from "./components/Dialog.vue";
    
    const {
      visible: visible1,
      openDialog: openDialog1,
      closeDialog: closeDialog1,
    } = useDialog();
    </script>

    useDialogState 和 useDialogWithForm

    useDialogState

    针对开发管理后台弹窗状态封装的一个hook,搭配下面的useDialogWithForm使用。

    export enum MODE {
      ADD,  EDIT,
    }
    import { ref } from "vue";
    import { MODE } from "./types";
    export default function useDialogState() {
      const mode = ref<MODE>(MODE.ADD);
      const visible = ref(false);
      const updateMode = (target: MODE) => {
        mode.value = target;
      };
      return { mode, visible, updateMode };
    }

    useDialogWithForm

    针对表单弹窗组件封装的hooks,接收一个formRef实例,负责控制弹窗内标题及清空表单中的校验结果,减少多余的代码 ~

    import { FormInstance } from "element-plus";
    import { Ref, ref } from "vue";
    import { MODE } from "./types";
    import useDialogState from "./useDialogState";
    
    export default function useDialogFn(
      formInstance: Ref<FormInstance>
    ) {
      const { visible, mode, updateMode } = useDialogState();
    
      const closeDialog = () => {
        formInstance.value.resetFields();
        visible.value = false;
      };
      const openDialog = (target: MODE) => {
        updateMode(target);
        visible.value = true;
      };
      return { visible, mode, openDialog, closeDialog };
    }

    useDialogWithForm Demo

    Lassen Sie uns darüber sprechen, wie Sie mit Vue3+Hook Popup-Komponenten schneller und effizienter schreiben können

    <template>
      <Dialog
        :before-close="customClose"
        @confirm="confirm"
        v-model="visible"
        :title="mode == MODE.ADD ? &#39;添加数据&#39; : &#39;编辑信息&#39;"
        :confirm-text="mode == MODE.ADD ? &#39;添加&#39; : &#39;修改&#39;"
      >
        <el-form
          label-width="100px"
          :model="formData"
          ref="formDataRef"
          style="max-width: 460px"
          :rules="rules"
        >
          <el-form-item label="姓名" prop="name">
            <el-input v-model="formData.name" />
          </el-form-item>
          <el-form-item label="年龄" prop="age">
            <el-input v-model="formData.age" />
          </el-form-item>
          <el-form-item label="手机号码" prop="mobile">
            <el-input v-model="formData.mobile" />
          </el-form-item>
        </el-form>
      </Dialog>
    </template>
    <script setup>
    import { ElMessage, FormInstance } from "element-plus";
    import { Ref, ref } from "vue";
    import Dialog from "./Dialog.vue";
    import { MODE } from "./types";
    import useDialogWithForm from "./useDialogWithForm";
    
    const rules = {
      name: {
        type: "string",
        required: true,
        pattern: /^[a-z]+$/,
        trigger: "change",
        message: "只能是英文名称哦",
        transform(value: string) {
          return value.trim();
        },
      },
      age: {
        type: "string",
        required: true,
        pattern: /^[0-9]+$/,
        trigger: "change",
        message: "年龄只能是数字哦",
        transform(value: string) {
          return value.trim();
        },
      },
      mobile: {
        type: "string",
        required: true,
        pattern:
          /^(?:(?:\+|00)86)?1(?:(?:3[\d])|(?:4[5-79])|(?:5[0-35-9])|(?:6[5-7])|(?:7[0-8])|(?:8[\d])|(?:9[189]))\d{8}$/,
        trigger: "change",
        message: "请输入正确的手机号码",
        transform(value: string) {
          return value.trim();
        },
      },
    };
    
    interface FromDataType {
      name: string;
      age: string;
      mobile: string;
    }
    
    const formDataRef = ref<FormInstance | null>(null);
    
    let formData = ref<FromDataType>({
      name: "",
      age: "",
      mobile: "",
    });
    
    const { visible, closeDialog, openDialog, mode } = useDialogWithForm(
      formDataRef as Ref<FormInstance>
    );
    const confirm = () => {
      if (!formDataRef.value) return;
      formDataRef.value.validate((valid) => {
        if (valid) {
          console.log("confirm");
          ElMessage({
            message: "提交成功",
            type: "success",
          });
          closeDialog();
        }
      });
    };
    
    const customClose = () => {
      ElMessage({
        message: "取消提交",
        type: "info",
      });
      closeDialog();
    };
    defineExpose({
      closeDialog,
      openDialog,
    });
    </script>
    <style scoped></style>

    仓库地址

    useDialog

    在线demo地址

    7 (1).gif

    如果您觉得本文对您有帮助,请帮帮忙点个star

    您的反馈 是我更新的动力!

    (学习视频分享:vuejs入门教程编程基础视频

    Das obige ist der detaillierte Inhalt vonLassen Sie uns darüber sprechen, wie Sie mit Vue3+Hook Popup-Komponenten schneller und effizienter schreiben können. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!

    Stellungnahme:
    Dieser Artikel ist reproduziert unter:juejin.cn. Bei Verstößen wenden Sie sich bitte an admin@php.cn löschen