7
5

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

More than 3 years have passed since last update.

nuxt.ts(nuxt.js+Typescript)でfirebaseをインジェクションしたい。

Last updated at Posted at 2020-04-08

#前書き
 nuxt.jsとFirebaseの組み合わせでなにか作るというのは結構ポピュラーらしく、qiitaでも検索するといろいろな記事が出てきます。私も大変お世話になりました。
 nuxt.jsでFirebaseSDKを使う場合、多くはpluginディレクトリ内でfirebase.jsを用意してFirebase SDKを初期化し、各々のVueファイル内でimportするというのが定番のようです。
 一方でnuxt.jsのインジェクションという機能を使うことで、いちいちvueファイル内でimportせずにFirebaseを呼び出せるようにする方法もあります。importの方がどこで何を使っているのか明示されてわかりやすいのですが、私は横着なのでインジェクションを使っています。
 今回はそれをTypeScript導入済みのnuxt.jsでやってみようと思い、(警告が出ていますが)実際に動作させることができましたので、嬉しさのあまり記事にした次第です。

#やること
TypeScript導入済みのnuxt.jsでFirebaseSDKをインジェクションする。

##省略すること

  • nuxt.jsの基本的な情報
  • nuxt.jsにTypeScriptを導入する方法
  • Firebaseの導入方法

##諦めたこと

  • 厳密な型付け

困ったらanyでもいいって偉い人類が言ってた。
(TypeScript再入門 ― 「がんばらないTypeScript」で、JavaScriptを“柔らかい”静的型付き言語に #エンジニアHub - エンジニアHub|若手Webエンジニアのキャリアを考える!
けどやっぱり型がついている方が圧倒的に便利

  • 全部TypeScriptにする。

できるところから頑張る。

#各ファイル
基本的には公式ドキュメントに従う形で書けば問題ない。
もっと良い方法があればコメントください。

##plugins/firebase.ts

firebase.ts
import firebase from 'firebase/app'
import 'firebase/firestore'
import 'firebase/auth'
import 'firebase/storage'
import 'firebase/messaging'
import 'firebase/analytics'
import { Plugin } from '@nuxt/types' //まずこれが必要

// 親の顔より見た初期化
if (!firebase.apps.length) {
  firebase.initializeApp(firebaseConfig)
}

// FCMに対応していない場合は$messagingはnullにしてエラーを防ぐ。
const messaging: firebase.messaging.Messaging | null = firebase.messaging.isSupported()
  ? firebase.messaging()
  : null

declare module 'vue/types/vue' {
  interface Vue {
    $firebase: firebase.app.App
    $auth: firebase.auth.Auth
    $db: firebase.firestore.Firestore
    $storage: firebase.storage.Storage
    $messaging: firebase.messaging.Messaging | null
    $analytics: firebase.analytics.Analytics
  }
}

declare module '@nuxt/types' {
  interface NuxtAppOptions {
    $firebase: firebase.app.App
    $auth: firebase.auth.Auth
    $db: firebase.firestore.Firestore
    $storage: firebase.storage.Storage
    $messaging: firebase.messaging.Messaging | null
    $analytics: firebase.analytics.Analytics
  }
}

declare module 'vuex/types/index' {
  interface Store<S> {
    $firebase: firebase.app.App
    $auth: firebase.auth.Auth
    $db: firebase.firestore.Firestore
    $storage: firebase.storage.Storage
    $messaging: firebase.messaging.Messaging | null
    $analytics: firebase.analytics.Analytics
}}
// ここでインジェクション
const myPlugin: Plugin = (context, inject) => {
  inject('firebase', firebase)
  inject('auth', firebase.auth())
  inject('db', firebase.firestore())
  inject('storage', firebase.storage())
}

// 普通にexportもできる。初期化済みのfirebaseを別ファイルから呼びたいときはこちら
export default myPlugin
export { firebase, messaging }
export const auth: firebase.auth.Auth = firebase.auth()
export const db: firebase.firestore.Firestore = firebase.firestore()
export const storage: firebase.storage.Storage = firebase.storage()
export const analytics: firebase.analytics.Analytics = firebase.analytics()
export const timestamp: Function = (): firebase.firestore.FieldValue => {
  return firebase.firestore.FieldValue.serverTimestamp()

今回は認証機能をmiddlewareに持たせるため、context.appから呼び出せるようにインジェクションする。
##nuxt.config.js

nuxt.config.js
 plugins: [ 
     { src: '~/plugins/firebase.ts', mode: 'client' } |
   ], 

そのうちTypeScriptにします。

##pages/index.vue

index.vue
 <script lang="ts"> //lang="ts"が必要
 import Vue from 'vue'
 
 type LocalData = { 
   msg: String; 
 };
 export default Vue.extend({ 
   data (): LocalData { 
     return { 
       msg: 'ebi kani uni' 
     } 
   }, 
   created ():void{ 
     const user = this.$auth.currentUser //this.$authで認証機能を呼べる。 
     if (user) { 
       this.$router.push('/brazil') 
     } else { 
       this.$router.push('/exit') 
     } 
   } 
 }) 
 </script> 

ここで実際に呼びだす。

##middleware/auth.js

auth.ts
 import { Middleware } from '@nuxt/types' // これが必要

 const defaultAuth: Middleware = (context) => { 
   const auth = context.app.$auth 
   auth.onAuthStateChanged((user: any) => { 
     if (!user) { 
       context.redirect('/exit') 
     } 
   } 
   ) 
 } 
  
 export default defaultAuth 

asyncDataやmiddlewareで呼び出す場合はこのような書き方。context.appである点に注意。

#参考資料
Nuxt TypeScript

#今までお世話になった記事
[初心者向け]ゼロから始めるNuxt.js入門
【v2対応】Nuxt.jsとFirebaseを組み合わせて爆速でWebアプリケーションを構築する
Vue.js+Firebase Authenticationで認証画面を爆速実装!| 開発者ブログ | 株式会社アイソルート
nuxt.js + Firebaseでアプリを作る際に import firebase ~~ を毎回書きたくない

7
5
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
7
5

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?