溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

vue3如何使用defineAsyncComponent與component標(biāo)簽實(shí)現(xiàn)動(dòng)態(tài)渲染組件

發(fā)布時(shí)間:2023-05-20 14:59:18 來(lái)源:億速云 閱讀:131 作者:iii 欄目:編程語(yǔ)言

本篇內(nèi)容介紹了“vue3如何使用defineAsyncComponent與component標(biāo)簽實(shí)現(xiàn)動(dòng)態(tài)渲染組件”的有關(guān)知識(shí),在實(shí)際案例的操作過(guò)程中,不少人都會(huì)遇到這樣的困境,接下來(lái)就讓小編帶領(lǐng)大家學(xué)習(xí)一下如何處理這些情況吧!希望大家仔細(xì)閱讀,能夠?qū)W有所成!

一、基礎(chǔ)的動(dòng)態(tài)引入組件:

簡(jiǎn)單的動(dòng)態(tài)引入的意思是,前端知道要引入哪些組件,將多個(gè)組件引入到父組件中,但不渲染它,滿足一定條件后,才去在某個(gè)位置渲染指定的組件。

<template>
	 <custom-modal ref="custom"></custom-modal>
</template>
<script>
 import {
    reactive,
    ref,
    shallowReactive,
    onActivated,
    defineAsyncComponent,
  } from 'vue';
 const customModal = defineAsyncComponent(() => import('./modal/CustomModal.vue'));
 const custom = ref();
 </script>

以上的例子就是通過(guò)vue的defineAsyncComponent實(shí)現(xiàn)掛載組件,并賦值給customModal ,模板中可以直接使用<custom-modal>作為標(biāo)簽使用,也可以將它賦值給component中的is屬性,is屬性執(zhí)向一個(gè)變量,可通過(guò)業(yè)務(wù)邏輯動(dòng)態(tài),更改該變量的值,就可以實(shí)現(xiàn)多個(gè)組件進(jìn)行來(lái)回的渲染了

<template>
<component :is="componentKey" ref="custom"></component>
</template>
 import {
    reactive,
    ref,
    shallowReactive,
    onActivated,
    defineAsyncComponent,
  } from 'vue';
 const componentKey = ref(null);
 const components: any = shallowReactive({});
 const customModal = defineAsyncComponent(() => import('./modal/CustomModal.vue'));
 componentKey  = customModal

二、復(fù)雜的引入:不確定到底引入什么組件,組件的路徑由后端返回

將以上代碼 添加到項(xiàng)目代碼中,并不能實(shí)現(xiàn),雖然引入不報(bào)錯(cuò),但是ref一直為undefined,無(wú)法調(diào)用動(dòng)態(tài)組件內(nèi)的open函數(shù)。
不斷嘗試了很多次,得出以下結(jié)論

1.起初是在按鈕的click函數(shù)內(nèi)去掛載自定義組件并調(diào)用ref函數(shù)的,ref為undefined。
嘗試多次不能實(shí)現(xiàn)功能(這里是掛載與調(diào)用最合適的位置),
2.接著又在初始化配置數(shù)據(jù)時(shí)(查詢后端sql),axios的then函數(shù)內(nèi)掛載組件,然后點(diǎn)擊按鈕的地方調(diào)用ref內(nèi)的函數(shù),ref依舊為null。
3. 接著在最外層,調(diào)用初始化時(shí)掛載,也就是生命周期函數(shù)體內(nèi),測(cè)試還是一樣的結(jié)果。
4. 接著發(fā)現(xiàn)帶有async函數(shù)體內(nèi)掛載組件,也無(wú)法完成。
5.單獨(dú)寫(xiě)個(gè)函數(shù),不加async,函數(shù)內(nèi)掛載組件,然后再生命周期外調(diào)用該函數(shù),按鈕內(nèi)調(diào)用ref內(nèi)的方法,成功彈窗。這并不是我想要的,因?yàn)槁窂讲皇枪潭ǖ?,它要等到后端sql放回結(jié)果,才能執(zhí)行。

總結(jié):上面的多次測(cè)試,得出以下結(jié)論,都不能讓動(dòng)態(tài)組件ref對(duì)象有值
1、不能在組件的事件函數(shù)內(nèi)掛載,

vue3如何使用defineAsyncComponent與component標(biāo)簽實(shí)現(xiàn)動(dòng)態(tài)渲染組件

2、不能在axios的then函數(shù)體內(nèi)掛載

vue3如何使用defineAsyncComponent與component標(biāo)簽實(shí)現(xiàn)動(dòng)態(tài)渲染組件

3、不能在帶有async聲明的函數(shù)體內(nèi)掛載

vue3如何使用defineAsyncComponent與component標(biāo)簽實(shí)現(xiàn)動(dòng)態(tài)渲染組件

4、不能在vue的生命周期內(nèi)掛載

vue3如何使用defineAsyncComponent與component標(biāo)簽實(shí)現(xiàn)動(dòng)態(tài)渲染組件

5、只能在最外層掛載實(shí)現(xiàn),這時(shí)ref才是個(gè)對(duì)象。

好在天無(wú)絕人之路;腦海里有個(gè)思路:
頁(yè)面初始化時(shí)將項(xiàng)目里所有的全局掛載view組件扔到一個(gè)object內(nèi),使用component組件,is:對(duì)應(yīng)object內(nèi)指定的組件對(duì)象,然后通過(guò)后端的數(shù)據(jù),這時(shí)后端就不用給組件路徑了,給個(gè)組件名,我從object中找到掛載的組件然后將對(duì)象給is。
const modules = import.meta.glob('@/views/*/**.vue'); // 獲取所有項(xiàng)目路徑
mudules為views內(nèi)所有的vue的相對(duì)路徑,然后循環(huán)它,在循環(huán)體內(nèi)實(shí)現(xiàn)掛載,將它存入一個(gè)對(duì)象內(nèi),key為相對(duì)路徑的項(xiàng)目名稱(可以截取以下)。

有了上面的思路,通過(guò)反復(fù)測(cè)試和實(shí)現(xiàn),最終功能實(shí)現(xiàn)了。

<template>
<component :is="componentKey" ref="custom"></component>
</template>
<script>
 import {
    reactive,
    ref,
    shallowReactive,
    onActivated,
    defineAsyncComponent,
  } from 'vue';
	
	//聲明componentkey,用于告訴component當(dāng)前掛載什么組件,components為一個(gè)對(duì)象,存放多個(gè)不確定的自定義組件。
  const componentKey = ref(null);
  const components: any = shallowReactive({});

  // 組件掛載
  const initTableConfig = (gridId, type) => { 
   queryTableConfig({ gridId }).then(({ data }) => {
      if (type === 'main') {
        Object.assign(mainConfig, data);
        tabsKey.value = -1;
      } else {
        tabsDetail.value.push(data);
        tabsKey.value = tabsDetail.value.length - 1;
      }
      // 涉及到自定義組件的部分,這里需要提前掛載,在用到時(shí)不至于ref為null
      XEUtils.objectEach(data.action, (action, key) => {
        if (
          action.modalCfg &&
          action.modalCfg.type === 'CustomModal' &&
          action.modalCfg.src
        ) {
          components[action.actionId] = defineAsyncComponent(
            () => import(`../../../${action.modalCfg.src}`)
          );
          //注意:這里的路徑后端只能返回相對(duì)路徑,不能使用@/xxx/xxx.vue ,不能使用src/xxx/xxx.vue,只能./xxx.vue或者../../xxx/xxx.vue。由于并不確定組件在什么位置,避免容易出錯(cuò)的原則,我在前端通過(guò)../../../的形式將路徑回退到src下,后端只需要從src下配置路徑即可,不用考慮那么多了。如后端src的值為src/xxx/xxx/xxx.vue 則在前端合成的路徑就為../../../src/xx/xxx/xxx.vue
          componentKey.value = components[action.actionId];
          // 為什么componentKey.vue在這里賦值,在后面點(diǎn)擊窗口后又賦值,這里能不能省略。
		//	答:這里省略的話,到點(diǎn)擊按鈕觸發(fā)時(shí)會(huì)報(bào)錯(cuò),第一次點(diǎn)擊會(huì)報(bào)錯(cuò),第二次點(diǎn)擊不會(huì)報(bào)錯(cuò),窗口正常彈出。可能是因?yàn)?,組件掛載時(shí)并沒(méi)有引入組件,只在使用時(shí)才引入,如果上面不提前將掛載好的組件引入進(jìn)來(lái),后面觸發(fā)事件觸發(fā)時(shí)引入在調(diào)用ref,執(zhí)行太快,costom就會(huì)報(bào)錯(cuò),所以才會(huì)點(diǎn)兩次才彈窗。
        }
      });
    });
  };
 </script>

按鈕點(diǎn)擊觸發(fā)事件,確定彈窗要彈出什么組件

		} else if (action.modalCfg.type === 'CustomModal') {
  		// 這里的actionid和組件是對(duì)應(yīng)的,所以在按鈕觸發(fā)后,通過(guò)按鈕攜帶的actionid能取到對(duì)應(yīng)的組件。
          componentKey.value = components[action.actionId];
          custom.value.init(row);
        }

經(jīng)過(guò)以上的方式:在任何地方掛載都不會(huì)報(bào)錯(cuò),完美解決。
注意:掛載與使用ref不能在同一個(gè)方法體內(nèi),如果可以的話,頁(yè)面加載時(shí),執(zhí)行掛載,需要調(diào)用ref時(shí)就不會(huì)報(bào)錯(cuò)。

“vue3如何使用defineAsyncComponent與component標(biāo)簽實(shí)現(xiàn)動(dòng)態(tài)渲染組件”的內(nèi)容就介紹到這里了,感謝大家的閱讀。如果想了解更多行業(yè)相關(guān)的知識(shí)可以關(guān)注億速云網(wǎng)站,小編將為大家輸出更多高質(zhì)量的實(shí)用文章!

向AI問(wèn)一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI