vue中實(shí)現(xiàn)子組件相互切換且數(shù)據(jù)不丟失的策略詳解
項(xiàng)目場(chǎng)景:
今天的項(xiàng)目場(chǎng)景: 項(xiàng)目為數(shù)據(jù)報(bào)表,但是一個(gè)父頁(yè)面中有很多的子頁(yè)面,而且子頁(yè)面中不是相互關(guān)聯(lián),但是數(shù)據(jù)又有聯(lián)系.
問(wèn)題描述
子頁(yè)面相互切換的時(shí)候之前填寫(xiě)好的數(shù)據(jù)會(huì)丟失,無(wú)法保存.這樣想提交所有的子頁(yè)面的數(shù)據(jù)就出現(xiàn)問(wèn)題.
原因分析:
分析原因就是子組件在切換的時(shí)候,我使用的是動(dòng)態(tài)組件,這個(gè)動(dòng)態(tài)組件的底層原理是v-if去判斷,這樣子組件就有一個(gè)消失重建的過(guò)程. 如何讓子組件保存不消失.而且還需要在切換回來(lái)的時(shí)候其他子頁(yè)面的數(shù)據(jù)改了之后,當(dāng)前的子頁(yè)面的數(shù)據(jù)也會(huì)同步?
解決方案:
vue中有一個(gè)延緩什么周期的組件,keep-alive
將子組件用keep-alive包裹之后
<keep-alive> <component :is="'tab' + active" :ref="'tab' + active"></component> </keep-alive>
整體代碼:
<template> <div class="userbox"> <comm-card> <!-- title 標(biāo)題 v-bind : v-on @ v-slot # --> <template #title> <div class="box"> <span>有子組件-案例2</span> </div> </template> <!-- content 正文--> <template #content> <div style="width: 100%"> <div class="tabtitle" style="margin-bottom: 20px"> <el-tag v-for="tag in tags" :key="tag.name" closable :type="tag.type" style="margin-left: 20px; width: 100px; text-align: center" @click="tabclicks(tag.id)" > {{ tag.name }} </el-tag> </div> <div class="tabbox"> <keep-alive> <component :is="'tab' + active" :ref="'tab' + active"></component> </keep-alive> </div> <div> <el-button type="primary" @click="save()">提交</el-button> </div> </div> </template> </comm-card> </div> </template> <script> /* 1, 不能讓子組件在切換的時(shí)候,值消失,必須使用緩存技術(shù), v-show 或者 keep-alive 2, 為了提高性能或者簡(jiǎn)化代碼,可以使用component 動(dòng)態(tài)組件加載技術(shù) 3, 使用ref技術(shù)獲取子組件的值 4, 結(jié)合tab欄點(diǎn)擊事件,實(shí)現(xiàn)獲取值的時(shí)機(jī) */ // 第一步 引入 import commCard from "../../components/commonCard"; import tab1 from "./components/sun1.vue"; import tab2 from "./components/sun2.vue"; import tab3 from "./components/sun3.vue"; // 第二步 注冊(cè)到我們的components中 export default { components: { commCard, tab1, tab2, tab3, }, data() { return { active: 1, tags: [ { id: 1, name: "菜單1", type: "" }, { id: 2, name: "菜單2", type: "success" }, { id: 3, name: "菜單3", type: "warning" }, ], values: {}, // 存不同的子組件的值 }; }, methods: { tabclicks(value) { //在點(diǎn)擊的時(shí)候開(kāi)始存入上一個(gè)子組件的值 this.values[this.active - 1] = this.$refs["tab" + this.active].obj; // 切換到已經(jīng)點(diǎn)擊的子組件上 this.active = this.tags.findIndex((ele) => { return ele.id == value; }) + 1; }, save() { //解決方案,將不同組件的值用不懂對(duì)象名稱包裹 // 獲取當(dāng)前子組件的值 this.values[this.active - 1] = this.$refs["tab" + this.active].obj; // 打印所以得子組件的值 console.log(this.values); }, }, }; </script> <style lang="less" scoped> .box { display: flex; justify-content: space-between; align-items: center; width: 100%; } </style>
完美解決問(wèn)題!
番外:component實(shí)現(xiàn)組件切換
1.代碼:
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>組件切換</title> </head> <body> <div class="body"> <input type="button" value="登錄" @click="name='login'"> <input type="button" value="注冊(cè)" @click="name='register'"> <input type="button" value="首頁(yè)" @click="name='index'"> <!-- 使用 component 元素實(shí)現(xiàn)組件之間的切換 --> <component :is="name"></component> </div> <!-- 登錄組件 --> <template id="login"> <h3>我是登錄組件</h3> </template> <!-- 注冊(cè)組件 --> <template id="register"> <h3>我是注冊(cè)組件</h3> </template> <!-- 首頁(yè)組件 --> <template id="index"> <h3>我是首頁(yè)組件</h3> </template> <script src="../lib/vue-2.4.0.js"></script> <script> /* 登錄組件 */ Vue.component("login", { template: "#login" }) /* 注冊(cè)組件 */ Vue.component("register", { template: "#register" }) /* 首頁(yè)組件 */ Vue.component('index', { template: "#index" }) let vm = new Vue({ el: ".body", data: { name:'login' } }) </script> </body> </html>
效果:
講解:
(1)定義三個(gè)不同的組件
(2)在 vue 實(shí)例的控制區(qū)域中寫(xiě)入 component 元素,設(shè)置 is 屬性(需要通過(guò) v-bind 進(jìn)行數(shù)據(jù)綁定),其值為將要顯示的組件的名稱
(3)在 data 中定義一個(gè) name 值,其初始值為 ‘login’(注意要用引號(hào)包裹起來(lái))
(4)為各個(gè)按鈕設(shè)置事件綁定,點(diǎn)擊時(shí)修改 name 為對(duì)應(yīng)的值(注意 name 的值要用引號(hào)包裹起來(lái))
以上就是vue中實(shí)現(xiàn)子組件相互切換且數(shù)據(jù)不丟失的策略詳解的詳細(xì)內(nèi)容,更多關(guān)于vue子組件相互切換的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
buildAdmin開(kāi)源項(xiàng)目引入四種圖標(biāo)方式詳解
這篇文章主要為大家介紹了buildAdmin開(kāi)源項(xiàng)目引入四種圖標(biāo)方式詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-02-02Vue實(shí)現(xiàn)固定定位圖標(biāo)滑動(dòng)隱藏效果
移動(dòng)端頁(yè)面,有時(shí)候會(huì)出現(xiàn)一些固定定位在底部圖標(biāo),比如購(gòu)物車等。這篇文章主要介紹了Vue制作固定定位圖標(biāo)滑動(dòng)隱藏效果,需要的朋友可以參考下2019-05-05在IDEA中Debug調(diào)試VUE項(xiàng)目的詳細(xì)步驟
idea竟然有一個(gè)神功能很多朋友都不是特別清楚,下面小編給大家?guī)?lái)了在IDEA中Debug調(diào)試VUE項(xiàng)目的詳細(xì)步驟,感興趣的朋友一起看看吧2021-10-10Vue3在router中使用pinia報(bào)錯(cuò)的簡(jiǎn)單解決辦法
這篇文章主要給大家介紹了關(guān)于Vue3在router中使用pinia報(bào)錯(cuò)的簡(jiǎn)單解決辦法,什么是pinia,可以理解為狀態(tài)管理工具,文中通過(guò)圖文介紹的非常詳細(xì),需要的朋友可以參考下2023-08-08Vue2實(shí)現(xiàn)txt文件在線預(yù)覽的代碼示例
txt文件在線預(yù)覽不需要下載另外的插件,主要有兩種形式,一種是上傳完成后實(shí)現(xiàn)預(yù)覽;另一種是后端提供文件下載接口,獲取文件在線地址實(shí)現(xiàn)預(yù)覽;本文給大家介紹了Vue2實(shí)現(xiàn)txt文件在線預(yù)覽的代碼示例,需要的朋友可以參考下2025-01-01