Vue3 父组件传值给子组件+以及使用NModal组件

2024-01-11 17:52

本文主要是介绍Vue3 父组件传值给子组件+以及使用NModal组件,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

前言:我想实现表格中点击详情弹窗出一个表格展示该行详细信息。想着这个弹窗里用子组件展示。分担父组件下,怕代码过多。(使用NModal组件弹窗展示)

等我一波百度,嗯,实现方法挺多嘛,什么refs什么的,看似条条大路通罗马,我一试,一试一条死路/(ㄒoㄒ)/~~,最后也没用到refs(我还是更倾向于用这个,毕竟vue2都是这样),一直说找不到$refs,我还去了解了底层**getCurrentInstance()**逻辑,有些又说不提倡这种方式,这个方法能看到我定义的的ref方法名,但是里面是null,又或者我当时子组件不是用的<script setup lang="ts">而是用的export default defineComponent下的setup(){},所以我不信邪,我子组件又改成了export default 还是不行。最后使用的defineProps
在这里插入图片描述

父组件:重点在 <child-detail :id="childRef"/>import childDetail from './childDetail.vue',以及需要改变childRef值的方法detailsEvent()

<template><div class="content-box"><n-form label-placement="left"inlinelabel-width="120px"><n-form-item label="UserName/Email"><n-input v-model:value="usersListRef.name_email" type="text" @input="queryConditional":style="{ width: '420px' }" placeholder="Please enter username or email (At least three letters)" /></n-form-item><!-- <n-form-item label="Email"><n-input v-model:value="usersListRef.email" type="text" :style="{ width: '280px' }" placeholder="Please enter email" /></n-form-item> --></n-form><n-space vertical :size="12" ><n-data-table style="min-height:calc(100vh - 223px);":bordered="false":single-line="false":columns="columns":data="data":loading="loading":pagination="pagination"striped /></n-space></div><n-modalv-model:show="showModal"class="custom-card"preset="card":style="bodyStyle"title="Details"size="huge":bordered="false":segmented="segmented"><!-- <template #header-extra>!</template> --><child-detail :id="childRef"/><!-- <n-data-table:bordered="false":single-line="false":columns="columns":data="data":loading="loading":max-height="300":pagination="pagination"striped /> --><!-- <template #footer>尾部</template> --></n-modal>
</template><script lang="ts">
import {NInput, NSpace,NButton, NForm, NFormItem, NDataTable, NModal} from 'naive-ui'
import {ref, defineComponent, Ref, getCurrentInstance, onMounted } from 'vue'
import { apiTokenRequest } from "@/stores/modules/apiEncapsulation"
import type { PaidlicenseResult } from "../../api/classes";
import type { DataTableColumns } from 'naive-ui'
import type { DataResult } from "../../api/classes";
import { h } from 'vue'
import { NTag, NAvatar } from 'naive-ui'
import { createDiscreteApi} from "naive-ui"
import childDetail from './childDetail.vue'interface UsersListType {username : string | nullemail: string | nullname_email: string | nullpageSize: number
}interface UsersListAll {id: string
}export default defineComponent({components: {NInput,NSpace,NButton,NForm,NFormItem,NDataTable,NModal,PaidLicenseDetail// NSwitch
},setup(){const childRef = ref("")const proxy = getCurrentInstance()!.proxyconst {message} = createDiscreteApi(["message"])const { dialog } = createDiscreteApi(["dialog"])let showModal = ref(false)const loadingRef = ref(true)const imgUrl = new URL('../../assets/AccountCircleOutlined.svg', import.meta.url).hrefconst usersListRef = ref<UsersListType>({username: null,email: null,name_email: null, pageSize: 12})const usersListAllResult : Ref<UsersListAll[]> = ref([])const getPaidResultMethod =  async ()=>{await apiTokenRequest<UsersListAll[]>("url","get").then(function(res){if(res){console.log('list===', res);usersListAllResult.value = resloadingRef.value = falseconsole.log('usersListAllResult===>',  usersListAllResult.value);}})}getPaidResultMethod()const queryConditional =async () => {console.log('条件查询===>', res);}const createColumns = ({detailsLicenses,deleteItem,handleChange,statusRes}: {detailsLicenses: (rowData: UsersListAll) => void,deleteItem: (rowData: UsersListAll) => void,handleChange: (rowData: UsersListAll) => void,statusRes: (status: number) => boolean,}): DataTableColumns<UsersListAll> => {return [{title: 'Name',key: 'name'},{title: 'Avatar',key: 'headerUrl',width:"120px",render (row) {return h(NAvatar,{style: {height:'50px',width:'50px',},color: '#fff',round:true,bordered: true,src:row.headerUrl ? row.headerUrl:imgUrl},)}},{title: 'Action',key: 'actions',width:"260px",render (row) {return [h(NButton,{size: 'small',type: "info",// quaternary: true,onClick: () => detailsLicenses(row)},{ default: () => 'Licenses' }),h( // 启、禁用NButton,{size: 'small',style: {marginLeft: "10px",width: "70px"},type: 'primary',// quaternary: true,onClick: () => handleChange(row)},{ default: () =>  row.status>0 ? 'Enable':'Disable' }),h( // 删除NButton,{size: 'small',style: {marginLeft: "10px",},type:"error",// quaternary: true,onClick: () => deleteItem(row)},{ default: () => 'Delete' }),]}}]}const detailsEvent = async (rowData:UsersListAll)=>{showModal.value = truechildRef.value= rowData.id}const deleteEvent = async (rowData:UsersListAll)=>{}const handleChangeEvent = async (rowData:UsersListAll)=>{}return{childRef,usersListRef,usersListAllResult,showModal,loading: loadingRef,queryConditional,data: usersListAllResult,columns: createColumns({detailsLicenses (rowData) {message.info('send mail to ' + rowData.name)detailsEvent(rowData)},deleteItem(rowData){deleteEvent(rowData)},handleChange(rowData){handleChangeEvent(rowData)}}),pagination: {pageSize: usersListRef.value.pageSize// pageSize: 2},bodyStyle: {width: '60vw',// height: '60vh'},segmented: {content: 'soft',// footer: 'soft'} as const,}}
})
</script><style scoped>
</style>

子组件:都是重点。

<template><n-data-table:bordered="false":single-line="false":data="detailsData":loading="loading":columns="childColumns":max-height="300":pagination="pagination"striped /> 1111111{{ props?.id }}
</template><script lang="ts">
import {NInput, NSpace,NButton, NForm, NFormItem, NDataTable, NModal} from 'naive-ui'
import { defineComponent, ref, Ref, defineProps, onMounted } from 'vue'
import { apiTokenRequest } from "@/stores/modules/apiEncapsulation"
import type { Result } from "../../api/classes";
import type { DataTableColumns } from 'naive-ui'export default defineComponent({components:{NDataTable},
})
</script><script setup lang="ts">const props = defineProps({id:{type: String,}})const pagination = ref( {pageSize: 10})const loading = ref(true)let childColumns :anyconst detailsData : Ref<PaidlicenseResult[]> = ref([])const getDetail= async (id:string|undefined)=>{await apiTokenRequest<Result[]>("url","get").then(function(res){if(res){console.log('details===', res);loading.value = falsedetailsData.value = reschildColumns  = columns()console.log('=====detailsData===', detailsData.value);}})}getDetail(props.id)const columns = () => {return [{title:'id',key: 'id'}]}
</script>

我原本想await 调用getDetail() 会报错。setup function returned a promise, but no <Suspense> boundary was found in the parent component tree. A component with async setup() must be nested in a <Suspense> in order to be rendered. ,想用await是数据还没请求完,表格列名又加载了(当时还没有使用childColumns = columns()),就会同时出现这些报错type check failed for prop "columns". Expected Array, got Function Unhandled error during execution of setup function还有Unhandled error during execution of scheduler flush. This is likely a Vue internals bug. Please open an issue at。后面我想着,那就等数据请求成功后再调用columns(),用childColumns 字段赋值给组件。

这篇关于Vue3 父组件传值给子组件+以及使用NModal组件的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/595221

相关文章

Spring Boot整合Redis注解实现增删改查功能(Redis注解使用)

《SpringBoot整合Redis注解实现增删改查功能(Redis注解使用)》文章介绍了如何使用SpringBoot整合Redis注解实现增删改查功能,包括配置、实体类、Repository、Se... 目录配置Redis连接定义实体类创建Repository接口增删改查操作示例插入数据查询数据删除数据更

使用python生成固定格式序号的方法详解

《使用python生成固定格式序号的方法详解》这篇文章主要为大家详细介绍了如何使用python生成固定格式序号,文中的示例代码讲解详细,具有一定的借鉴价值,有需要的小伙伴可以参考一下... 目录生成结果验证完整生成代码扩展说明1. 保存到文本文件2. 转换为jsON格式3. 处理特殊序号格式(如带圈数字)4

Java使用Swing生成一个最大公约数计算器

《Java使用Swing生成一个最大公约数计算器》这篇文章主要为大家详细介绍了Java使用Swing生成一个最大公约数计算器的相关知识,文中的示例代码讲解详细,感兴趣的小伙伴可以了解一下... 目录第一步:利用欧几里得算法计算最大公约数欧几里得算法的证明情形 1:b=0情形 2:b>0完成相关代码第二步:加

Java中流式并行操作parallelStream的原理和使用方法

《Java中流式并行操作parallelStream的原理和使用方法》本文详细介绍了Java中的并行流(parallelStream)的原理、正确使用方法以及在实际业务中的应用案例,并指出在使用并行流... 目录Java中流式并行操作parallelStream0. 问题的产生1. 什么是parallelS

Linux join命令的使用及说明

《Linuxjoin命令的使用及说明》`join`命令用于在Linux中按字段将两个文件进行连接,类似于SQL的JOIN,它需要两个文件按用于匹配的字段排序,并且第一个文件的换行符必须是LF,`jo... 目录一. 基本语法二. 数据准备三. 指定文件的连接key四.-a输出指定文件的所有行五.-o指定输出

Linux jq命令的使用解读

《Linuxjq命令的使用解读》jq是一个强大的命令行工具,用于处理JSON数据,它可以用来查看、过滤、修改、格式化JSON数据,通过使用各种选项和过滤器,可以实现复杂的JSON处理任务... 目录一. 简介二. 选项2.1.2.2-c2.3-r2.4-R三. 字段提取3.1 普通字段3.2 数组字段四.

Linux kill正在执行的后台任务 kill进程组使用详解

《Linuxkill正在执行的后台任务kill进程组使用详解》文章介绍了两个脚本的功能和区别,以及执行这些脚本时遇到的进程管理问题,通过查看进程树、使用`kill`命令和`lsof`命令,分析了子... 目录零. 用到的命令一. 待执行的脚本二. 执行含子进程的脚本,并kill2.1 进程查看2.2 遇到的

详解SpringBoot+Ehcache使用示例

《详解SpringBoot+Ehcache使用示例》本文介绍了SpringBoot中配置Ehcache、自定义get/set方式,并实际使用缓存的过程,文中通过示例代码介绍的非常详细,对大家的学习或者... 目录摘要概念内存与磁盘持久化存储:配置灵活性:编码示例引入依赖:配置ehcache.XML文件:配置

Java 虚拟线程的创建与使用深度解析

《Java虚拟线程的创建与使用深度解析》虚拟线程是Java19中以预览特性形式引入,Java21起正式发布的轻量级线程,本文给大家介绍Java虚拟线程的创建与使用,感兴趣的朋友一起看看吧... 目录一、虚拟线程简介1.1 什么是虚拟线程?1.2 为什么需要虚拟线程?二、虚拟线程与平台线程对比代码对比示例:三

k8s按需创建PV和使用PVC详解

《k8s按需创建PV和使用PVC详解》Kubernetes中,PV和PVC用于管理持久存储,StorageClass实现动态PV分配,PVC声明存储需求并绑定PV,通过kubectl验证状态,注意回收... 目录1.按需创建 PV(使用 StorageClass)创建 StorageClass2.创建 PV