ファイルシステムルーティング
Nuxt は pages ディレクトリにある Vue ファイルの木構造に基づいて、vue-router の設定を自動的に生成します。pages ディレクトリに .vue ファイルを作成すると、特別な設定をせずとも基本的なルーティングが動作するようになります。
動的なルーティングやネストされたルートを生成したり、ルーターのプロパティをより細かく設定したりする必要がある場合があるでしょう。この章では、ルーターを最大限に活用するために知るべきことすべてを説明します。
<template>
<NuxtLink to="/">Home page</NuxtLink>
</template>
ルーティングの基礎
下記のような木構造のとき:
pages/
--| user/
-----| index.vue
-----| one.vue
--| index.vue
自動的に以下が生成されます:
router: {
routes: [
{
name: 'index',
path: '/',
component: 'pages/index.vue'
},
{
name: 'user',
path: '/user',
component: 'pages/user/index.vue'
},
{
name: 'user-one',
path: '/user/one',
component: 'pages/user/one.vue'
}
]
}
動的なルーティング
ユーザーやブログ記事の一覧を API から取得する場合など、ルートの名前を決定することができない場合があります。私たちはこれを動的なルーティングと呼びます。動的なルーティングを生成するには、.vue
ファイル名またはディレクトリ名の前にアンダースコア(_
)を付ける必要があります。ファイルやディレクトリには好きな名前をつけることができますが、アンダースコアのプレフィックスを必ず付ける必要があります。
下記のような木構造のとき:
pages/
--| _slug/
-----| comments.vue
-----| index.vue
--| users/
-----| _id.vue
--| index.vue
自動的に以下が生成されます:
router: {
routes: [
{
name: 'index',
path: '/',
component: 'pages/index.vue'
},
{
name: 'users-id',
path: '/users/:id?',
component: 'pages/users/_id.vue'
},
{
name: 'slug',
path: '/:slug',
component: 'pages/_slug/index.vue'
},
{
name: 'slug-comments',
path: '/:slug/comments',
component: 'pages/_slug/comments.vue'
}
]
}
users-id
と名付けられたルートに :id?
というパスがありますが、これはこの :id
が必須ではないことを表します。もし必須にしたい場合は、代わりに users/_id
ディレクトリ内に index.vue
ファイルを作成してください。ルートパラメーターへのローカルアクセス
ローカルページやコンポーネント内の現在のルートパラメータは、this.$route.params.{parameterName}
を参照することでアクセスすることができます。例えば、動的な ユーザーページ(users/_id.vue
)があり、ユーザーまたはプロセス情報を読み込むために id
パラメーターにアクセスしたい場合、次のような変数にアクセスできます: this.$route.params.id
ネストされたルート
Nuxt は vue-router の子ルートを使ってルートをネストさせることができます。ネストされたルートの親コンポーネントを定義するには、子ビューを含むディレクトリと同じ名前の Vue ファイルを作成する必要があります。
.vue
ファイル)内に NuxtChild コンポーネント を含めることを忘れないでください。下記のような木構造のとき:
pages/
--| users/
-----| _id.vue
-----| index.vue
--| users.vue
自動的に以下が生成されます:
router: {
routes: [
{
path: '/users',
component: 'pages/users.vue',
children: [
{
path: '',
component: 'pages/users/index.vue',
name: 'users'
},
{
path: ':id',
component: 'pages/users/_id.vue',
name: 'users-id'
}
]
}
]
}
動的でネストされたルート
あまり頻繁に使う場面はありませんが、Nuxt は動的な親ルーティングの中に動的な子ルーティングを持つことが可能です。
下記のような木構造のとき:
pages/
--| _category/
-----| _subCategory/
--------| _id.vue
--------| index.vue
-----| _subCategory.vue
-----| index.vue
--| _category.vue
--| index.vue
自動的に以下が生成されます:
router: {
routes: [
{
path: '/',
component: 'pages/index.vue',
name: 'index'
},
{
path: '/:category',
component: 'pages/_category.vue',
children: [
{
path: '',
component: 'pages/_category/index.vue',
name: 'category'
},
{
path: ':subCategory',
component: 'pages/_category/_subCategory.vue',
children: [
{
path: '',
component: 'pages/_category/_subCategory/index.vue',
name: 'category-subCategory'
},
{
path: ':id',
component: 'pages/_category/_subCategory/_id.vue',
name: 'category-subCategory-id'
}
]
}
]
}
]
}
未知の動的でネストされたルート
もし URL 構造の深さが不明な場合は、ネストされたパスに動的にマッチさせる _.vue
ファイルを使うことができます。これはより詳細なルートにマッチしなかったリクエストをハンドリングします。
下記のような木構造のとき:
pages/
--| people/
-----| _id.vue
-----| index.vue
--| _.vue
--| index.vue
次のようにリクエストをハンドリングします:
/ -> index.vue
/people -> people/index.vue
/people/123 -> people/_id.vue
/about -> _.vue
/about/careers -> _.vue
/about/careers/chicago -> _.vue
_.vue
ページのロジックに依存します。ルーターの拡張
Nuxt のルーティングを拡張する方法はいくつかあります:
- router-extras-module を使ってページのルートパラメータをカスタマイズする
-
@nuxtjs/router コンポーネントを使い、独自の
router.js
で Nuxt router を上書きする -
nuxt.config.js
内で router.extendRoutes プロパティを使う
router プロパティ
router プロパティを使って Nuxt のルーター (vue-router) をカスタマイズできます。
export default {
router: {
// Nuxt router をカスタマイズする
}
}
Base:
アプリケーションのベース URL です。例えばシングルページアプリケーション全体を /app/
配下で配信したい場合、base に /app/
を設定します。
extendRoutes
Nuxt によって生成されるルーティングを拡張したい場合があるでしょう。それは extendRoutes
オプションで実現できます。
独自のルートを追加する例:
export default {
router: {
extendRoutes(routes, resolve) {
routes.push({
name: 'custom',
path: '*',
component: resolve(__dirname, 'pages/404.vue')
})
}
}
}
ルートをソートしたい場合、@nuxt/utils
の sortRoutes(routes)
関数を使うことができます:
import { sortRoutes } from '@nuxt/utils'
export default {
router: {
extendRoutes(routes, resolve) {
// ここにルートを追加します ...
// それらをソートします
sortRoutes(routes)
}
}
}
export default {
router: {
extendRoutes(routes, resolve) {
routes.push({
path: '/users/:id',
components: {
default: resolve(__dirname, 'pages/users'), // または routes[index].component
modal: resolve(__dirname, 'components/modal.vue')
},
chunkNames: {
modal: 'components/modal'
}
})
}
}
}
fallback
history.pushState をサポートしていないブラウザにおいて、モードが history に設定されている場合に、ルーターを hash モードにフォールバックするかどうかを制御します。
mode
ルーティングのモードを設定します。サーバーサイドレンダリングのため、この設定を変更することは非推奨です。
parseQuery / stringifyQuery
カスタムクエリ構文解析関数 / 文字列化関数を提供します。
routeNameSplitter
Nuxt が使うルート名の区切り文字を変更したい場合があるでしょう。それは設定ファイル内の routeNameSplitter
オプションを使用して実現できます。pages/posts/_id.vue
というページファイルがあるとします。Nuxt はプログラムに従ってルート名を生成します。この場合は posts-id
です。routeNameSplitter
の設定を /
に変更することによって posts/id
へ変更されます。
export default {
router: {
routeNameSplitter: '/'
}
}
scrollBehavior
scrollBehavior
オプションを使って、ページ間遷移のスクロール位置について独自の振る舞いを定義することができます。このメソッドはページがレンダリングされるたびに毎回呼び出されます。
v2.9.0 以降で利用可能:
ファイルを利用してルーターの scrollBehavior を上書きすることができます。このファイルは app フォルダに配置する必要があります。
~/app/router.scrollBehavior.js
.
すべてのルートに置いて強制的にトップまでスクロールさせる例:
export default function (to, from, savedPosition) {
return { x: 0, y: 0 }
}
trailingSlash
v2.10 以降で利用可能:
このオプションを true に設定した場合、すべてのルートの末尾にスラッシュが追加されます。false に設定した場合、それらは取り除かれます。
export default {
router: {
trailingSlash: true
}
}
router.trailingSlash
に undefined
(デフォルトの値)以外の値を設定した場合、どちらかのルートは機能しなくなります。したがって、301 リダイレクトは適切に行われ、内部リンクは正しく適応される必要があります。trailingSlash
を true
に設定する場合、example.com/abc/
のみが機能し、example.com/abc
は機能しません。false に設定する場合はその逆になります。data:image/s3,"s3://crabby-images/c079f/c079f408bc783e75be10a58380dcd8f75fc6fe30" alt="Sébastien Chopin"
data:image/s3,"s3://crabby-images/a36f1/a36f155e74808db2757a1b6265083d6a6d4fc94c" alt="Nazaré da Piedade"
data:image/s3,"s3://crabby-images/96e50/96e5023b65c2d33f36fe7c3eac0a0c96b238658e" alt="Nobu"
data:image/s3,"s3://crabby-images/f29a0/f29a0d620d87519a52cfd0ae8e69128a162a7b67" alt="川音리오"
data:image/s3,"s3://crabby-images/2677d/2677de88513b8010c060962c34aeb67dac770a18" alt="Maciek Palmowski"
data:image/s3,"s3://crabby-images/404a5/404a58aea4ef3c54629ba26fc32a51d6e2a18a5f" alt="Nestor Vera"
data:image/s3,"s3://crabby-images/98a40/98a40243e9a287aa704ef3dc22e5fc28e59b7574" alt="Daniel Roe"
data:image/s3,"s3://crabby-images/b7385/b7385c01953ba5d76898a41fb5b24feb98bc4bb2" alt="Yue Yang"
data:image/s3,"s3://crabby-images/2a3d1/2a3d1e419f61998b33f62e14cf9b5a4783aa7794" alt="Jeronimas"
data:image/s3,"s3://crabby-images/b336a/b336a791a4de2d8cb75b955cb90eabf2879b83a5" alt="Clément Ollivier"
data:image/s3,"s3://crabby-images/37dca/37dcaf7d38f5846b7c5c9c391f5918a51ccd934c" alt="Alexander Lichter"
data:image/s3,"s3://crabby-images/a27d2/a27d2e6048cbb2955844e53dd5ec8a75e08ea2cd" alt="N3-rd"
data:image/s3,"s3://crabby-images/d6a94/d6a9457c88e2400fa917f2d9ae084e5b1a1cb517" alt="Adrien Zaganelli"
data:image/s3,"s3://crabby-images/e2529/e2529558cb857cbf0010c38a88b255deea628bdf" alt="Mag"
data:image/s3,"s3://crabby-images/5b539/5b5399cdda0f1d2b5b03e847623b619a64ac1d0f" alt="Stefan Huber"
data:image/s3,"s3://crabby-images/2b5ad/2b5ad589e7cc4f5c0babe7670eefddc43968146e" alt="Olga Bulat"
data:image/s3,"s3://crabby-images/c30e9/c30e9a86a4692d5c123c90220ddca7a458e17e35" alt="Paiva"
data:image/s3,"s3://crabby-images/5d4bf/5d4bff0bf543f1ad8ce5610ae03bb5c2901dc1df" alt="Florian Reuschel"
data:image/s3,"s3://crabby-images/b8800/b8800632d3225a51be4bed995e635716c72d6761" alt="Savas Vedova"
data:image/s3,"s3://crabby-images/3eb50/3eb5088ee9bebfb29c66604180d5f1786dcc288e" alt="HIJACK"
data:image/s3,"s3://crabby-images/ba3ff/ba3ffe962ceab2637a46936c4b3832823bb0f5e8" alt="Vinícius Alves"
data:image/s3,"s3://crabby-images/75fcf/75fcfe777c70ebb7577bde2a0c2e2f82512da323" alt="Kareem Dabbeet"
data:image/s3,"s3://crabby-images/da55d/da55d7a86e3c71ef5371bfcd5fbf9faaed6ddc51" alt="Valentín Costa"
data:image/s3,"s3://crabby-images/9d225/9d22581d38ab2acd1069230435358270c09a5128" alt="Ryan Skinner"
data:image/s3,"s3://crabby-images/c9d3b/c9d3b3b0b99544192e54a41ae709bd992c32794f" alt="Alex Hirzel"
data:image/s3,"s3://crabby-images/a66f5/a66f51077e5283065acdcf5249a88a6dcda457b7" alt="Ajeet Chaulagain"
data:image/s3,"s3://crabby-images/cc775/cc77528ad32d6df1c92dcd7819df9c709e7aae17" alt="René Eschke"
data:image/s3,"s3://crabby-images/4f930/4f9307ea6d868688a41bce88daa5e734dd7b4f30" alt="Nico Devs"
data:image/s3,"s3://crabby-images/642a4/642a45e9f53437a0dc0786dad3822b47542ab41e" alt="Muhammad"
data:image/s3,"s3://crabby-images/578b7/578b7a4826d9e9c79b10c0a440bf6023cac8634d" alt="Naoki Hamada"
data:image/s3,"s3://crabby-images/18000/18000ed7b913e1a5ef0c301cec202908424f2b36" alt="Tom"
data:image/s3,"s3://crabby-images/f2eaa/f2eaafbc890bdba93ddfc694d9ceb7a59bf21a4c" alt="Yann Aufray"
data:image/s3,"s3://crabby-images/28706/2870667dfa80efaff50559c179c068fbf01ef205" alt="Anthony Chu"
data:image/s3,"s3://crabby-images/18462/18462ffca9a6899eec5a54b7de6a45887d0729c8" alt="Nuzhat Minhaz"
data:image/s3,"s3://crabby-images/9f43b/9f43ba42692b50b513a0f9fee1d9c100d0c9f177" alt="Lucas Portet"
data:image/s3,"s3://crabby-images/87bb8/87bb8cdadf010bc5ead4801489bb088532a79443" alt="Richard Schloss"
data:image/s3,"s3://crabby-images/6bf20/6bf2065989faa4ad57a63c65a21c0d4918a0dd73" alt="Bobby"
data:image/s3,"s3://crabby-images/c5e5e/c5e5e61af957ba4bebf4242162f8db94d3da7ca1" alt="bpy"
data:image/s3,"s3://crabby-images/bbc3f/bbc3fdf0e5e37795f6a3a8267834e6b3b9cdb8ce" alt="Antony Konstantinidis"
data:image/s3,"s3://crabby-images/3ba7a/3ba7a27e05f9691d23fa8d0b28e16355254be5ec" alt="Hibariya"
data:image/s3,"s3://crabby-images/e12ba/e12ba2b49b19dd4dee0cab2dee520f022fc39436" alt="Jose Seabra"
data:image/s3,"s3://crabby-images/ab2b3/ab2b3791f0b93c3d266a07a7331e8c639c45b840" alt="Eze"
data:image/s3,"s3://crabby-images/fe12d/fe12d1c0b546b012067bd695ea18752e750c162f" alt="Florian Lefebvre"
data:image/s3,"s3://crabby-images/3f4be/3f4bee239f7004d6c28c9006fd5b9bb5ad3cae9e" alt="Lucas Recoaro"
data:image/s3,"s3://crabby-images/736af/736af6fee3723bd40ca1a4cea3f57ece7f182b15" alt="Julien SEIXAS"