搜尋

首頁  >  問答  >  主體

模擬Vue RouterView/RouterLink在Vitest(組合 API)中

根據標題,這有可能嗎?

我正在嘗試測試Vue應用程式的一個簡單元件,其中包含一個標題和一個按鈕,該按鈕將使用者重定向到下一個頁面。我想測試一下,當按鈕被點擊時,是否會向路由器發送事件/訊息,並檢查目標路由是否正確。

應用程式的架構如下:

App.Vue - 入口元件

<script setup lang="ts">
import { RouterView } from "vue-router";
import Wrapper from "@/components/Wrapper.vue";
import Container from "@/components/Container.vue";
import HomeView from "@/views/HomeView.vue";
</script>

<template>
  <Wrapper>
    <Container>
      <RouterView/>
    </Container>
  </Wrapper>
</template>

<style>

  body{
    @apply bg-slate-50 text-slate-800 dark:bg-slate-800 dark:text-slate-50;
  }

</style>

router/index.ts - Vue Router設定檔

import {createRouter, createWebHistory} from "vue-router";
import HomeView from "@/views/HomeView.vue";

export enum RouteNames {
    HOME = "home",
    GAME = "game",
}

const routes = [
    {
      path: "/",
      name: RouteNames.HOME,
      component: HomeView,
      alias: "/home"
    },
    {
      path: "/game",
      name: RouteNames.GAME,
      component: () => import("@/views/GameView.vue"),
    },
];

const router = createRouter({
  history: createWebHistory(import.meta.env.BASE_URL),
  routes
});

export default router;

HomeView.Vue - App元件中RouterView的入口視圖

<template>
  <Header title="My App"/>
  <ButtonRouterLink :to="RouteNames.GAME" text="Start" class="btn-game"/>
</template>

<script setup>
  import Header from "@/components/Header.vue";
  import ButtonRouterLink from "@/components/ButtonRouterLink.vue";
  import {RouteNames} from "@/router/";
</script>

ButtonRouterLink.vue

<template>
<RouterLink v-bind:to="{name: to}" class="btn">
  {{ text }}
</RouterLink>
</template>

<script setup>
import { RouterLink } from "vue-router";

const props = defineProps({
  to: String,
  text: String
})
</script>

考慮到它使用CompositionAPI和TypeScript,有沒有辦法在Vitest測試中模擬路由器?

這是一個測試檔案結構的範例(HomeView.spec.ts):

import {shallowMount} from "@vue/test-utils";
import { describe, test, vi, expect } from "vitest";
import { RouteNames } from "@/router";
import HomeView from "../../views/HomeView.vue";

describe("Home", () => {
    test ('navigates to game route', async () => {
        // Check if the button navigates to the game route

        const wrapper = shallowMount(HomeView);

        await wrapper.find('.btn-game').trigger('click');

        expect(MOCK_ROUTER.push).toHaveBeenCalledWith({ name: RouteNames.GAME });
    });
});

我嘗試了多種方法:vi.mock('vue-router'),vi.spyOn(useRoute,'push'),vue-router-mock庫,但是我無法讓測試運行起來,似乎路由器就是不可用的。

更新(15/04/23)

根據@tao 的建議,我意識到在HomeView中測試點擊事件並不是一個很好的測試(測試的是庫而不是我的應用程式),所以我添加了一個測試ButtonRouterLink,看看它是否正確渲染了Vue RouterLink,使用to參數:

import {mount, shallowMount} from "@vue/test-utils";
import { describe, test, vi, expect } from "vitest";
import { RouteNames } from "@/router";
import ButtonRouterLink from "../ButtonRouterLink.vue";

vi.mock('vue-router');

describe("ButtonRouterLink", () => {
    test (`correctly transforms 'to' param into a router-link prop`, async () => {
       
        const wrapper = mount(ButtonRouterLink, {
            props: {
                to: RouteNames.GAME
            }
        });

        expect(wrapper.html()).toMatchSnapshot();
    });
});

它渲染了一個空的HTML字串""

#
exports[`ButtonRouterLink > correctly transforms 'to' param into a router-link prop 1`] = `""`;

伴隨著一個不太有用的Vue警告(未指定預期類型):

[Vue warn]: Invalid prop: type check failed for prop "to". Expected , got Object  
  at <RouterLink to= { name: 'game' } class="btn btn-blue" > 
  at <ButtonRouterLink to="game" ref="VTU_COMPONENT" > 
  at <VTUROOT>
[Vue warn]: Invalid prop: type check failed for prop "ariaCurrentValue". Expected , got String with value "page". 
  at <RouterLink to= { name: 'game' } class="btn btn-blue" > 
  at <ButtonRouterLink to="game" ref="VTU_COMPONENT" > 
  at <VTUROOT>
[Vue warn]: Component is missing template or render function. 
  at <RouterLink to= { name: 'game' } class="btn btn-blue" > 
  at <ButtonRouterLink to="game" ref="VTU_COMPONENT" > 
  at <VTUROOT>

P粉478188786P粉478188786283 天前526

全部回覆(1)我來回復

  • P粉759451255

    P粉7594512552024-03-28 00:46:02

    這是可能的

    你的範例和官方範例之間有一個重要的區別:在被測試的元件中,你將其放置在了一個中間元件中,而不是使用RouterLink(或使用router.push的按鈕)。

    這會如何改變單元測試?

    第一個問題是你使用了shallowMount,它會用空容器取代子元件。在你的情況下,這意味著ButtonRouterLink不再包含RouterLink,並且不會對點擊事件做任何操作,它只是接收點擊事件。

    解決這個問題的選項有:

    • a) 使用mount而不是shallowMount,將其變成整合測試而不是單元測試
    • b) 將這個功能的測試移到ButtonRouterLink的測試中。一旦測試了ButtonRouterLink的任何:to是否正確轉換為{ name: to }並傳遞給RouterLink,你只需要測試這些:to是否正確傳遞給使用它們的任何元件中的<ButtonRouterLink />

    將這個功能移到自己的元件中所創建的第二個問題稍微微妙一些。通常,我期望以下ButtonRouterLink的測試可以工作:

    import { RouterLinkStub } from '@vue/test-utils'
    
    const wrapper = shallowMount(YourComp, {
      stubs: {
        RouterLink: RouterLinkStub
      }
    })
    
    expect(wrapper.findComponent(RouterLinkStub).props('to')).toEqual({
      name: RouterNames.GAME
    })
    

    令我驚訝的是,這個測試失敗了,它聲稱在RouterLinkStub中接收到的:to屬性值不是{ name: RouterNames.GAME },而是'game',這是我傳給ButtonRouterLink的值。就像我們的元件從未將value轉換為{ name: 值 }一樣。

    相當奇怪。
    事實證明,問題是<RouterLink />恰好是我們組件的根元素。這表示元件(<BottomRouterLink>)在DOM中被<RouterLinkStub>取代,但:to的值是從前者而不是後者讀取的。簡而言之,如果模板是這樣的,測試就會成功:

      <span>
        <RouterLink ... />
      </span>
    

    為了解決這個問題,我們需要導入實際的RouterLink元件(來自vue-router),並找到,而不是找到RouterLinkStub。為什麼? @vue/test-utils足夠聰明,可以找到存根元件而不是實際元件,但是這樣,.findComponent()只會匹配替換後的元素,而不是在它仍然是ButtonRouterLink(未處理)時。此時,:to的值已經被解析為RouterLink實際接收的值。

    完整的測試如下:

    import { shallowMount } from '@vue/test-utils'
    import { describe, it, expect } from 'vitest'
    import ButtonRouterLink from '../src/ButtonRouterLink.vue'
    import { RouterLinkStub } from '@vue/test-utils'
    import { RouterLink } from 'vue-router'
    
    const TEST_STRING = 'this is a test string'
    
    describe('ButtonRouterLink', () => {
      it(`should transform 'to' into '{ name: to }'`, () => {
        const wrapper = shallowMount(ButtonRouterLink, {
          props: {
            to: TEST_STRING
          },
          stubs: {
            RouterLink: RouterLinkStub
          }
        })
    
        expect(wrapper.findComponent(RouterLink).props('to')).toEqual({
          name: TEST_STRING
        })
      })
    })

    有點棘手。

    回覆
    0
  • 取消回覆