---
title: 【Vue.js】Vue Router 之透過路由組件傳參數給元件
date: 2019-12-30
is_modified: false
disqus: cynthiahackmd
categories:
- "程式設計 › 程式語言與架構"
tags:
- "前端"
- "Vue.js"
---
{%hackmd @CynthiaChuang/Github-Page-Theme %}
<br>
一般來說,想透過路由傳遞參數給元件,多採用[**動態路由**](https://router.vuejs.org/zh/guide/essentials/dynamic-matching.html#%E5%93%8D%E5%BA%94%E8%B7%AF%E7%94%B1%E5%8F%82%E6%95%B0%E7%9A%84%E5%8F%98%E5%8C%96)的方式來實作,但這種方式一則容易使得元件高度耦合,導致只能在特定 URL 上使用;二則是你永遠別低估使用者亂輸入的能力,他們常常會輸入一些超乎設計者預期的字串 XD
因此,我希望可以找到其他方式傳參數給元件,最後終於在文件找到[**路由組件傳參**](https://router.vuejs.org/zh/guide/essentials/passing-props.html)的相關說明。不過它的說明有些簡略,還是稍微嘗試了下才找到我要的用法。
<!--more-->
## 情境說明
嚴格來說,是因為我同時實作==巢狀路由==與==[路由組件傳參](https://router.vuejs.org/zh/guide/essentials/passing-props.html)==兩件事情,才使得事情變得有些複雜。
<p class="illustration">
<img src="https://i.imgur.com/MUt7P2C.png" alt="巢狀路由(嵌套路由)示意圖">
巢狀路由示意圖 (圖片來源: <a href="https://router.vuejs.org/zh/guide/essentials/nested-routes.html">Vue Router 官網</a>)
</p>
我的情境有點像上圖,一個頁面中存在兩個元件,分別是外層的的 menu,與主體內容的 content 元件。
只是在我的應用中,兩個主體內容的 content 元件實作方式除了 Breadcrumbs 與 Title 不同外,其餘幾乎一樣。因此將他們合而為一,再使用一個標記來區分要顯示的 Breadcrumbs 與 Title,似乎是比較好的管理方法。
但因為是有限的標記選擇—不是 Profile 就是 Posts,再加上我也不希望讓使用者修改標記,畢竟若輸入不是這兩個標記時,也算==標題與內文不符==了 XDDD,因此我直接排除常用的動態路由做法。
## 路由組件傳參
因為剔除動態路由的關係,使得我碰到了點瓶頸。還好後來在文件找到[路由組件傳參](https://router.vuejs.org/zh/guide/essentials/passing-props.html),它可以通過 props 解耦,這正是我需要的。
<br>
根據文件,有三種方式可以透過 props 解耦
### Boolean mode 布林模式
布林模式的用法是跟==動態路由==一併使用的,首先建立一個用來顯示的元件,並設定所需的 props:
```htmlmixed=
<template>
<div class="hello">
<h1>{{ msg }}</h1>
<h2>I form {{from}}</h2>
</div>
</template>
<script>
export default {
name: 'HelloWorld',
props: {
from: {
type: String,
default: ""
}
},
data () {
return {
msg: 'Welcome'
}
}
}
</script>
```
<br>
接下來再 router/index.js 的檔案中,定義相關路由與一個動態路徑參數,最重要的是==為你的命名視圖添加 `props` 選項並設置為 True==。
此路徑參數將直接傳遞給該元件,需注意的是,此路徑參數需與元件中所要使用的 props 變數名稱相同,才能順利傳遞。
```javascript=
export default new Router({
routes: [
{
path: '/:from',
name: 'HelloWorld',
props: true,
component: HelloWorld
}
]
})
```
<br>
若你的路徑參數與 props 變數名稱不相同,或是路徑參數多於 props 所定義的變數,這些==無法被元件的 prop 所識別且獲取的特性==,則可以在 `vm.$attrs` 中查看。
<br>
如果你的主視圖是巢狀路由架構,則為命名視圖所添加的 `props` 需配合 `components` 一同改成物件,其 key 值則對應到在主視圖中 router-view 的標籤所定義的名字。
當主視圖 template 如下:
```htmlmixed=
<template>
<div id="app">
<router-view name="menu"/>
<router-view name="content"/>
</div>
</template>
```
<br> 則路由定義如下,在路由中定義的所有動態路徑參數,透過參數傳遞這兩個元件,若參數可被該元件的 prop 所識別且獲取,則會對應到 props 變數反之則會出現 `vm.$attrs` 中。
```javascript=
export default new Router({
routes: [
{
path: '/:from/:id',
name: 'HelloWorld',
props: { menu: true, content: true},
components: {
menu: Menu,
content: HelloWorld
}
}
]
})
```
<br> 若僅其中一個元件不需要接收 props,則可將將其布林模式設定為 False。
```javascript=
props: { menu: false, content: true},
```
### Object mode 物件模式
若是你的應用情境不需要設定動態路由,或是像我一樣僅需傳送固定參數,則建議使用==物件模式==。
物件模式,顧名思義就是使用命名視圖中添加 `props` 選項來傳遞物件。且與布林模式一樣,若所傳遞的物件可被 prop 所識別,則會對應到 props 變數反之則會出現在 `vm.$attrs` 中:
```javascript=
export default new Router({
routes: [
{
path: '/',
name: 'HelloWorld',
props: { from: "Taiwan"},
component: HelloWorld
}
]
})
```
<br>
若是巢狀路由架構,則與布林模式一樣使用物件來為不同的 router-view 元件傳遞物件。
```javascript=
export default new Router({
routes: [
{
path: '/',
name: 'HelloWorld',
props: { menu: {id:1}, content: { from: "Taiwan"}},
components: {
menu: Menu,
content: HelloWorld
}
}
]
})
```
<br>
若是其中一個元件不需要接收 props,則可將其布林模式設定為 False,或是不將其加入 key 值。
```javascript=
props: { menu:false, content: { from: "Taiwan"}}
或
props: { content: { from: "Taiwan"}}
```
### Function mode 函式模式
最後一個函式模式,顧名思義就是建立一個函式傳回 props。如此一來就可以將參數轉成你所需要的型態與名稱。
因為布林模式中 **query** 的值,是無法透過參數被傳入元件中的,因此這邊建立一個函式將 query 轉成物件傳入元件中。
```javascript=
export default new Router({
routes: [
{
path: '/from',
name: 'HelloWorld',
props: (route) => ({ from: route.query.w }),
component: HelloWorld
}
]
})
```
以這組code 為例,如過在 URL 中輸入 `/from?w=Taiwan`,將會傳入 `{from: 'Taiwan'}`,交由元件的 prop 進行識別。
<br>
也可以透過函式將參數轉成你所需要的名稱
```javascript=
export default new Router({
routes: [
{
path: '/:id',
name: 'HelloWorld',
props: (route) => ({ from: route.params.id }),
component: HelloWorld
}
```
<br>
巢狀路由架構的部分,一樣使用物件來為不同的 router-view 元件傳遞函式
```javascript=
export default new Router({
routes: [
{
path: '/from',
name: 'HelloWorld',
props: { menu: (route) => ({ id: route.query.w }),
content: (route) => ({ from: route.query.w })
},
components: {
menu: Menu,
content: HelloWorld
}
}
]
})
```
<br>
關閉其中一個元件的 props,與前兩個模式一樣則將它設定為 False,或是不將其加入 key 值。
```javascript=
props: { menu:false, content: { from: "Taiwan"}}
或
props: { content: { from: "Taiwan"}}
```
## 參考資料
1. [路由组件传参|Vue Router](https://router.vuejs.org/zh/guide/essentials/passing-props.html)
2. [API|Vue.js](https://cn.vuejs.org/v2/api/#vm-attrs)
<br><br>
> **本文作者**: 辛西亞.Cynthia
> **本文連結**: [辛西亞的技能樹](https://cynthiachuang.github.io/Vue-Router-Passing-Props-to-Route-Components) / [hackmd 版本](https://hackmd.io/@CynthiaChuang/Vue-Router-Passing-Props-to-Route-Components)
> **版權聲明**: 部落格中所有文章,均採用 [姓名標示-非商業性-相同方式分享 4.0 國際](https://creativecommons.org/licenses/by-nc-sa/4.0/deed.en) (CC BY-NC-SA 4.0) 許可協議。轉載請標明作者、連結與出處!