如何对 Vue 3“脚本设置”组合 API 上的路由更改做出反应?

Fed*_*o S 5 javascript refactoring typescript vue.js vue-router

这是我的应用程序中的路线示例:

{
  path: "/something",
  name: "SomeRoute",
  component: SomeComponent,
  meta: {showExtra: true},
},
{
  path: "/somethingElse",
  name: "SomeOtherRoute",
  component: SomeOtherComponent,
},
Run Code Online (Sandbox Code Playgroud)

然后我有以下组件,您会注意到它有两个script标签,一个带有组合 API,一个没有:

<template>
  <div>
    This will always be visible. Also here's a number: {{ number }}
  </div>

  <div v-if="showExtra">
    This will be hidden in some routes.
  </div>
</template>

<script setup lang="ts">
import type { RouteLocationNormalized } from "vue-router"
import { ref } from "vue"

const number = 5
const showExtra = ref(true)

const onRouteChange = (to: RouteLocationNormalized) => {
  showExtra.value = !!to.meta?.showExtra
}
</script>

<script lang="ts">
import { defineComponent } from "vue"


export default defineComponent({
  watch: {
    $route: {
      handler: "onRouteChange",
      flush: "pre",
      immediate: true,
      deep: true,
    },
  },
})
</script>
Run Code Online (Sandbox Code Playgroud)

这工作正常:当我输入一条路线时,meta: {showExtra: false}它会隐藏额外的 div,否则它会显示它。

然而,我想要做的是仅使用组合 API 来实现相同的目的,换句话说,<script>完全删除第二个标签。我已经尝试过这个:

<script setup lang="ts">
import type { RouteLocationNormalized } from "vue-router"
import { ref } from "vue"
import { onBeforeRouteUpdate } from "vue-router"


// ...
// same as before
// ...

onBeforeRouteUpdate(onRouteChange)
</script>
Run Code Online (Sandbox Code Playgroud)

但是当我切换路线时,这不会按预期生效。我知道watch我可以导入的函数,但我不确定如何获取有关路线的元信息以及如何安抚类型检查器。

s4k*_*k1b 10

watch您可以通过从 vue导入方法将您的观察者转换为组合 api

<script lang="ts">
import { defineComponent, vue, watch } from "vue"
import { useRoute } from "vue-router"

export default defineComponent({
  setup() {
    const route = useRoute()
    watch(route, (to) => {
      showExtra.value = !!to.meta?.showExtra
    }, {flush: 'pre', immediate: true, deep: true})
  },
})
</script>
Run Code Online (Sandbox Code Playgroud)

  • 小修正不是 `import { useRoute } from "vuex"` 而是 `import { useRoute } from 'vue-router';` (3认同)