NavigationDuplicated不允许导航到当前位置(“ /搜索”)[vuejs]


81

当我想进行多次搜索时,会显示NavigationDuplicated错误。我的搜索在导航栏中,而我配置搜索的方式是使用模型获取值,然后将该值作为参数传递给ContentSearched组件,然后在该组件中接收搜索的值。

我知道正确的方法是使用发射器,但是我仍然不知道如何学习使用它。要访问发射是context.emit('', someValue)

NavigationDuplicated {_name: "NavigationDuplicated", name: "NavigationDuplicated", message: "Navigating to current location ("/search") is not allowed", stack: "Error↵    at new NavigationDuplicated (webpack-int…node_modules/vue/dist/vue.runtime.esm.js:1853:26)"}

导航栏

<template>
  <nav class="navbar navbar-expand-lg navbar-dark bg-nav" v-bind:class="{'navbarOpen': show }">
    <div class="container">
      <router-link to="/" class="navbar-brand">
        <img src="../assets/logo.png" alt="Horizon Anime" id="logo">
      </router-link>

      <button class="navbar-toggler" type="button" data-toggle="collapse" data-target="#navbarSupportedContent" aria-controls="navbarSupportedContent" aria-expanded="false" aria-label="Toggle navigation" v-on:click.prevent="toggleNavbar">
        <span class="navbar-toggler-icon"></span>
      </button>

      <div class="collapse navbar-collapse" id="navbarSupportedContent" v-bind:class="{'show': show }">
        <ul class="navbar-nav mr-auto">
          <li class="nav-item">
            <router-link class="nav-link" to="/" ><i class="fas fa-compass"></i> Series</router-link>
          </li>
          <li class="nav-item">
            <router-link class="nav-link" :to="{name: 'EpisodesSection'}" ><i class="fas fa-compact-disc"></i> Episodios</router-link>
          </li>
          <li class="nav-item">
            <router-link class="nav-link" :to="{name: 'MovieSection'}" ><i class="fas fa-film"></i> Peliculas</router-link>
          </li>
        </ul>
        <div class="search-bar">
          <form class="form-inline my-2 my-lg-0">
            <input class="form-control mr-sm-2" v-model="query" type="search" placeholder="Buscar películas, series ..." aria-label="Search">
            <button class="btn btn-main my-2 my-sm-0" @click.prevent="goto()" type="submit"><i class="fas fa-search"></i></button>
          </form>
        </div>
      </div>
    </div>
  </nav>
</template>

<script>
  import {value} from 'vue-function-api';
  import {useRouter} from '@u3u/vue-hooks';

  export default {
    name: "NavBar",
    setup(context){
      const {router} = useRouter();
      const query = value("");

      let show = value(true);
      const toggleNavbar = () => show.value = !show.value;      
      
      const goto = () =>{
        let to = {name: 'ContentSearched' , params:{query: query}}
        router.push(to);
      };
        
      return{
        show,
        toggleNavbar,
        goto,
        query
      }
    }
  }
</script>

ContentSearched.vue

<template>
   <div class="container">
     <BoxLink/>
    <main class="Main">
      <div class="alert alert-primary" role="alert">
        Resultados para "{{query}}"
      </div>
      <div v-if="isLoading">
        <!-- <img class="loading" src="../assets/loading.gif" alt="loading"> -->
      </div>
      <div v-else>
        <ul class="ListEpisodios AX Rows A06 C04 D02">
          <li v-for="(content, index) in contentSearched" :key="index">
            <div v-if="content.type === 'serie'">
              <Series :series="content"/>
            </div>
            <div v-if="content.type === 'pelicula'">
              <Movies :movies="content"/>
            </div>
          </li>
        </ul>
      </div>
    </main>
  </div>
</template>


<script>
  import {onCreated} from "vue-function-api"
  import {useState , useRouter , useStore} from '@u3u/vue-hooks';
  import BoxLink from "../components/BoxLink";
  import Movies from "../components/Movies";
  import Series from "../components/Series";

  export default{
    name: 'ContentSearched',
    components:{
      BoxLink,
      Movies,
      Series
    },
    setup(context){
      const store = useStore();
      const {route} = useRouter();

      const state = {
        ...useState(['contentSearched' , 'isLoading'])
      };

      const query = route.value.params.query;

      onCreated(() =>{
        store.value.dispatch('GET_CONTENT_SEARCH' , query.value);
      });
      return{
        ...state,
        query,
      }
    }
  };
</script>

Answers:


96

当我router-link指向同一条路线时,这发生在我身上。例如/products/1

用户可以单击产品,但是如果已经单击产品(并且已经加载了组件视图),并且用户尝试再次单击它,则控制台中将显示错误/警告。

您可以在github问题上了解更多信息。

Posva是vue-router的主要贡献者之一,他建议:

router.push('您的路径').catch(err => {})

但是,如果您不想让某个catch块什么也不做,为了解决该问题,您可以将路由器导航与当前路径进行比较,并且仅在它们不同时进行导航:

const path = `/products/${id}`
if (this.$route.path !== path) this.$router.push(path)

注意:$route是vue-router为每个组件提供的对象。有关更多信息,请参见路由对象


33

我认为,如果我们不打算进一步将其Router.push用作异步调用,则可以在根标签上解决此问题的最佳解决方案。

import Router from 'vue-router';

const originalPush = Router.prototype.push;
Router.prototype.push = function push(location) {
  return originalPush.call(this, location).catch(err => err)
};

Vue.use(Router);

哈雷路亚!虽然即使使用router-link时也出现此错误!在Github上,posva声称仅当使用router.push()时才需要此解决方法,而对于router-link则不需要。知道为什么为什么在使用router-link时仍然会收到此错误(并在使用您的代码时摆脱错误)吗?
Tech Nomad

@TechNomad,您可以使用$router.beforeEach'钩子,并next(false)以此终止导航
Saksham

@TechNomad,会在$router.beforeEach调用之前生成NavigationDuplicated 。
Mark Dornian

26

如果您不愿意捕获所有类型的错误,那么我认为此实现更为体贴:

this.$router.push("path").catch(error => {
  if (error.name != "NavigationDuplicated") {
    throw error;
  }
});

16

截至2020年,全局配置:

我只想使NavigationDuplicated错误保持沉默,空着的陷阱可能很危险。所以我这样做了:

const router = new VueRouter({/* ... */})

function patchRouterMethod (router, methodName)
{
    router['old' + methodName] = router[methodName]
    router[methodName] = async function (location)
    {
        return router['old' + methodName](location).catch((error) =>
        {
            if (error.name === 'NavigationDuplicated')
            {
                return this.currentRoute
            }
            throw error
        })
    }
}

patchRouterMethod(router, 'push')
patchRouterMethod(router, 'replace')

初始化vue-router时,请插入一次。



12

如果您在代码中使用router.push,并且不关心导航失败,则应使用catch来捕获它:

router.push('/location').catch(err => {})


8

我在搜索时遇到了同样的问题。我的解决方案是添加timestampthis.$route.query搜索页面的参数。

this.$router.push({
    path: "/search",
    query: {
      q: this.searchQuery,
      t: new Date().getTime(),
    }
  });

希望对您有帮助。


谢谢,这是一个很好的答案。其他答案使NavigationDuplicated错误消失,但这实际上可以导航到相同的路线。这样就可以导航到相同的路线,但具有不同的参数。就我而言,我需要进行单元测试。
JuroOravec

4

您在这里混合了多个概念 router-links到程序化导航,以将参数查询到状态存储。这使得帮助您和告诉您这里的“正确”解决方案有点困难。

尽管如此,我认为最好的方法是:
1)将路线定义为

{
  path: "/search/:searchString",
  component: MySearchComponent,
  props: true
}

2)使用回应式<router-link>而不是您的router.push

<input type="text" v-model="searchString">
<router-link :to="'/search/'+searchString" tag="button">search</router-link>

3)searchString通过props: ['searchString']和访问搜索组件中的this.searchString

props: ['searchString'],
...
computed: {
  msg() {
    return `Searching for, ${this.searchString}!`;
  }
}

完整示例:https :
//codesandbox.io/s/vue-routing-example-9zc6g注意,我只是将第一个codeandbox叉了一个router我可以找到的代码,并进行了相应调整。


-1

这是一个简单而有效的解决方案:

if(from.fullPath === to.fullPath){
    return
}

@ hero592请通读行为准则 ,不要发表这样的评论。我们所有人都在这里帮助他人,而不是沉迷于这种不可接受的做法。
Manish
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.