如何实现可重用的 vuetify 对话框

如何解决如何实现可重用的 vuetify 对话框

我想用 vuetify 创建一个可重用的 v-dialog, 我创建了 BaseModal:

     <v-dialog
      v-model="inputVal"
      :hide-overlay="overlay"
      :max-width="width"
    >
      <v-card>
        <v-toolbar flat>
          <v-btn color="black" dark icon right @click="closeModal">
            <v-icon>mdi-close</v-icon>
          </v-btn>
          <v-spacer></v-spacer>
          <v-toolbar-title id="toolbar-title">{{ title }}</v-toolbar-title>
        </v-toolbar>
        <v-card-title class="headline">
          <!--Card Title-->
        </v-card-title>
        <v-card-text>
          <slot name="content"></slot>
        </v-card-text>
      </v-card>
    </v-dialog>

InputVal 作为计算和 dataBindingName 作为道具:

  computed: {
    inputVal: {
      get() {
        return this.dataBindingName;
      },set(val) {
        this.$emit("input",val);
      }
    }
  },props: {
    dataBindingName: {
      Boolean,required: false
    },}

我在另一个组件中调用这个基本组件,如下所示:

<modal-simple
  :dataBindingName="dataBindingName"
  title="Nouveau"
  width="418"
>
  <template v-slot:content>
    <add-time-sheet-form />
  </template>

</modal-simple>

我现在的问题是如何正确实现 closeModal 以及如何在内部实现关闭模态并打开新模态的按钮

解决方法

第 1 部分。模态组件。

注意不要不要使用 v-model="dialog"。如前所述in this answer,您需要将其替换为:value 和@input。

此外,您不应该直接改变 dialog 道具,这就是为什么您需要在需要关闭模态时发出 close-dialog 事件。

<template>
  <v-dialog
    :value="dialog"
    @input="$emit('input',$event)"
    max-width="500px"
    persistent
  >
    <v-card>
      <v-card-title>
        ... dialog header ...
      </v-card-title>
      <v-card-text>
        ... dialog content ...
      </v-card-text>
      <v-card-actions>
        <v-spacer />
        <v-btn @click="close">
          Close
        </v-btn>
      </v-card-actions>
    </v-card>
  </v-dialog>
</template>

<script>
export default {
  props: {
    editedId: Number,dialog: Boolean,},methods: {
    close() {
      this.$emit("close-dialog");
    },};
</script>

第 2 部分。父组件。

假设您想在添加或编辑实体时使用您的组件。

您需要实现两个方法:addItemeditItem 来打开您的对话框。您还需要传递额外的实体道具(在我的示例中为 editedId)、带有 sync modifierdialog 道具和 close-dialog 事件处理程序这是从模态组件发出的。

<template>
  <div>
    <v-toolbar flat color="white">
      <v-spacer />
      <edit-modal
        :edited-id="editedId"
        :dialog.sync="dialog"
        @close-dialog="
          editedId = null;
          dialog = false;
        "
      />
      <v-btn color="primary" dark class="mb-2" @click="addItem">
        Add entity
      </v-btn>
    </v-toolbar>
    <v-data-table :items="items" :headers="headers">
      <template v-slot:item="props">
        <tr>
          <td>{{ props.item.name }}</td>
          <td class="justify-center text-center">
            <v-icon small class="mr-2" @click="editItem(props.item)">
              edit
            </v-icon>
          </td>
        </tr>
      </template>
    </v-data-table>
  </div>
</template>

<script>
import Dialog from "./ReusableDialog";

export default {
  components: {
    "edit-modal": Dialog,data() {
    return {
      items: [
        ... some items ...
      ],headers: [
        ... some headers ...
      ],editedId: null,dialog: false,};
  },methods: {
    addItem() {
      this.dialog = true;
    },editItem(item) {
      this.editedId = item.id;
      this.dialog = true;
    },};
</script>

第 3 部分。修改模态以自动重新打开。

在这个例子中,模态组件会自动重新打开,直到 editedId 不会被设置。

在模态组件中:

...
close() {
  this.$emit("close-dialog");
  if (this.editedId === null) {
    this.$emit("open-dialog");
  }
},...

在父组件中:

...
<edit-modal
  ... some props ...
  @open-dialog="
    editedId = 999;
    dialog = true;
  "
/>
...

有一个 codesandbox with working example

,

这是我使用 Composition API 插件 + provide/inject 的实现:

https://gist.github.com/wobsoriano/1e63708d7771c34f835c0f6e3c5e731a

您只需要在提供的要点中使用 DialogProvider 组件包装模板组件,如下所示:

<template>
  <v-app>
    <v-main>
      <v-container>
        <DialogProvider>
          <Nuxt />
        </DialogProvider>
      </v-container>
    </v-main>
  </v-app>
</template>

<script lang="ts">
import { defineComponent } from '@nuxtjs/composition-api';
import DialogProvider from '~/components/DialogProvider.vue';
export default defineComponent({
  components: {
    DialogProvider,}
});
</script>

然后您可以使用 useDialog 可组合在任何地方显示确认对话框:

<script lang="ts">
import { defineComponent } from '@nuxtjs/composition-api';
import { useDialog } from '~/composables';
export default defineComponent({
  setup() {
    const createConfirmDialog = useDialog();
    const handleDelete = async () => {
      try {
        const shouldProceed = await createConfirmDialog(
          'Confirm','Delete this post?',{ width: 300 }
        );
        if (shouldProceed) {
          // delete post
        }
      } catch (_e) {}
    };
    return {
      handleDelete
    };
  },});
</script>

它也对 TypeScript 友好。

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


使用本地python环境可以成功执行 import pandas as pd import matplotlib.pyplot as plt # 设置字体 plt.rcParams[&#39;font.sans-serif&#39;] = [&#39;SimHei&#39;] # 能正确显示负号 p
错误1:Request method ‘DELETE‘ not supported 错误还原:controller层有一个接口,访问该接口时报错:Request method ‘DELETE‘ not supported 错误原因:没有接收到前端传入的参数,修改为如下 参考 错误2:cannot r
错误1:启动docker镜像时报错:Error response from daemon: driver failed programming external connectivity on endpoint quirky_allen 解决方法:重启docker -&gt; systemctl r
错误1:private field ‘xxx‘ is never assigned 按Altʾnter快捷键,选择第2项 参考:https://blog.csdn.net/shi_hong_fei_hei/article/details/88814070 错误2:启动时报错,不能找到主启动类 #
报错如下,通过源不能下载,最后警告pip需升级版本 Requirement already satisfied: pip in c:\users\ychen\appdata\local\programs\python\python310\lib\site-packages (22.0.4) Coll
错误1:maven打包报错 错误还原:使用maven打包项目时报错如下 [ERROR] Failed to execute goal org.apache.maven.plugins:maven-resources-plugin:3.2.0:resources (default-resources)
错误1:服务调用时报错 服务消费者模块assess通过openFeign调用服务提供者模块hires 如下为服务提供者模块hires的控制层接口 @RestController @RequestMapping(&quot;/hires&quot;) public class FeignControl
错误1:运行项目后报如下错误 解决方案 报错2:Failed to execute goal org.apache.maven.plugins:maven-compiler-plugin:3.8.1:compile (default-compile) on project sb 解决方案:在pom.
参考 错误原因 过滤器或拦截器在生效时,redisTemplate还没有注入 解决方案:在注入容器时就生效 @Component //项目运行时就注入Spring容器 public class RedisBean { @Resource private RedisTemplate&lt;String
使用vite构建项目报错 C:\Users\ychen\work&gt;npm init @vitejs/app @vitejs/create-app is deprecated, use npm init vite instead C:\Users\ychen\AppData\Local\npm-
参考1 参考2 解决方案 # 点击安装源 协议选择 http:// 路径填写 mirrors.aliyun.com/centos/8.3.2011/BaseOS/x86_64/os URL类型 软件库URL 其他路径 # 版本 7 mirrors.aliyun.com/centos/7/os/x86
报错1 [root@slave1 data_mocker]# kafka-console-consumer.sh --bootstrap-server slave1:9092 --topic topic_db [2023-12-19 18:31:12,770] WARN [Consumer clie
错误1 # 重写数据 hive (edu)&gt; insert overwrite table dwd_trade_cart_add_inc &gt; select data.id, &gt; data.user_id, &gt; data.course_id, &gt; date_format(
错误1 hive (edu)&gt; insert into huanhuan values(1,&#39;haoge&#39;); Query ID = root_20240110071417_fe1517ad-3607-41f4-bdcf-d00b98ac443e Total jobs = 1
报错1:执行到如下就不执行了,没有显示Successfully registered new MBean. [root@slave1 bin]# /usr/local/software/flume-1.9.0/bin/flume-ng agent -n a1 -c /usr/local/softwa
虚拟及没有启动任何服务器查看jps会显示jps,如果没有显示任何东西 [root@slave2 ~]# jps 9647 Jps 解决方案 # 进入/tmp查看 [root@slave1 dfs]# cd /tmp [root@slave1 tmp]# ll 总用量 48 drwxr-xr-x. 2
报错1 hive&gt; show databases; OK Failed with exception java.io.IOException:java.lang.RuntimeException: Error in configuring object Time taken: 0.474 se
报错1 [root@localhost ~]# vim -bash: vim: 未找到命令 安装vim yum -y install vim* # 查看是否安装成功 [root@hadoop01 hadoop]# rpm -qa |grep vim vim-X11-7.4.629-8.el7_9.x
修改hadoop配置 vi /usr/local/software/hadoop-2.9.2/etc/hadoop/yarn-site.xml # 添加如下 &lt;configuration&gt; &lt;property&gt; &lt;name&gt;yarn.nodemanager.res