SwiftUI における開発で Firebase の Realtime Database を連携する方法を1から解説していきます。
Firebase プロジェクトの作成
Firebase console にアクセスして、プロジェクトを新規作成します。
data:image/s3,"s3://crabby-images/288cc/288ccc03a284b395e703ca4e6c7dc8ab8e8cbe7d" alt=""
プロジェクト名を入力します。
data:image/s3,"s3://crabby-images/dc070/dc0707ff319df15aed9ced3a27f6827769d69f80" alt=""
Google アナリティクス設定は今回は使わないのでどちらでも良いですが、とりあえず ON にしておきます。
data:image/s3,"s3://crabby-images/0500a/0500af9855d924d74597c084a5b048a2356c0825" alt=""
Google アナリティクスアカウントを聞かれるので、新規で作成します。
アナリティクスの地域は日本にしておきましょう。
data:image/s3,"s3://crabby-images/81035/810356958d344812b34029c18d9966293fc76378" alt=""
Google とのデータの共有は好みでチェックを付け、利用規約に同意すると Firebase のプロジェクトを作成できます。
data:image/s3,"s3://crabby-images/1363a/1363a0d1baa2ed36ebd095bef43861b308929bce" alt=""
しばらく待つと、プロジェクトの作成が完了します。
data:image/s3,"s3://crabby-images/5a0ca/5a0cacef9ea9eef10713f1b3a38aa56170a001e4" alt=""
Realtime Database の作成
左のメニューから「Realtime Database」を選択して、「データベースを作成」をクリックします。
data:image/s3,"s3://crabby-images/86e8b/86e8b059fbc62c0fe62a89ba1b1099f65d41c4f1" alt=""
Realtime Database location はデフォルトの 「米国 (us-central1)」のままで OK です。
data:image/s3,"s3://crabby-images/3a96d/3a96d2054f99934943b648c73861b592d857ea46" alt=""
セキュリティルールはいったんテストモードにしておきましょう。
ここでは詳しく開設しませんが、この rules にていろいろと read / write の権限設定をすることができます。
テストモードのデフォルトだと「作成から30日間はプロジェクト内であれば誰でも読み書き可能」となっています。
もし、テスト期間を伸ばしたければ該当の数値を増やすか、そもそも
".read": true,
としてしまえば OK です。
(開発完了時に適切なセキュリティルールにすることをお忘れなく)
data:image/s3,"s3://crabby-images/65a16/65a160d9238b5401341e35890ff75c5e8bf19890" alt=""
これで Realtime Database が作成されました。
data:image/s3,"s3://crabby-images/48654/48654089b1dd9b5f9a36f6ae3f8f02865e6de8c5" alt=""
Firebase プロジェクトに iOS アプリを追加
プロジェクトの概要から iOS のアイコンをクリックします。
data:image/s3,"s3://crabby-images/c65f8/c65f899743819e5a90a7d646befbcfa3bbacbb07" alt=""
iOS バンドル ID を設定します。
この値は Xcode にて設定する Bundle Identifier と一致させてください。
data:image/s3,"s3://crabby-images/d4d57/d4d575797a3b24ca9ae460d6078f83f114c03712" alt=""
GoogleService-Info.plist をダウンロードし、所定の場所に置きます。
data:image/s3,"s3://crabby-images/0ce1a/0ce1aa3380fc29cae8fb0cb270d6014715854115" alt=""
Firebase SDK の追加ですが、ここに書いてある手順ではなく、Swift Package Manager を使って追加してみます。
手順は下記を参考にしています。
Swift Package Manager を使用して Firebase をインストールする
data:image/s3,"s3://crabby-images/3135e/3135e4a7b86d303241dc21fee21ac2071923f8e4" alt=""
Xcode で
File > Swift Package > Add Package Dependency...
と選択します。
data:image/s3,"s3://crabby-images/04ef0/04ef072a8ec7324bbae0a7e2a4bae74e666a3807" alt=""
検索窓に https://github.com/firebase/firebase-ios-sdk.git
と入力します。
data:image/s3,"s3://crabby-images/4863c/4863cecf7816cb579cb239b1704cba0299390bcb" alt=""
バージョンはそのままで OK です。
data:image/s3,"s3://crabby-images/a8ec4/a8ec45ae9d91b6210c753ab13d72b02e56da1e07" alt=""
しばらく待ちます。
data:image/s3,"s3://crabby-images/9725a/9725a6330b1eceb7278acdca491010733e366f4e" alt=""
利用するパッケージを選択します。
ここでは「FirebaseAnalytics」と「FirebaseDatabase」にチェックを入れます。
data:image/s3,"s3://crabby-images/50058/50058eb0f53271ac61f8b0e5d79f4b4a1e515578" alt=""
Firebase SDK の追加が終わったら、初期化コードを書きます。
data:image/s3,"s3://crabby-images/88767/88767d096029d3436a21915985261b241f4438d8" alt=""
ただ、Xcode 12 からプロジェクトの作成時の Life Cycle で 「SwiftUI App」を選ぶと AppDelegate が作成されないので注意が必要です。
※「UIKit App Delegate」を選択した場合は従来どおり AppDelegate が生成されているはずなので大丈夫です
その際は「<プロジェクト名>App.swift」というファイルが生成されるので、そちらを下記のように修正します。
import SwiftUI
import Firebase // 追加
@main
struct realtimedb_sampleApp: App {
@UIApplicationDelegateAdaptor(AppDelegate.self) var appDelegate // 追加
var body: some Scene {
WindowGroup {
ContentView()
}
}
}
// 下記追加
class AppDelegate: UIResponder, UIApplicationDelegate {
func application(_ application: UIApplication, didFinishLaunchingWithOptions launchOptions: [UIApplication.LaunchOptionsKey : Any]? = nil) -> Bool {
FirebaseApp.configure()
return true
}
}
下記サイトを参考にさせていただきました。ありがとうございます。
[Xcode 12] アプリの起動について変更になった部分まとめ
これで完了です。
data:image/s3,"s3://crabby-images/33705/3370571207c597348918d0a5bf36902eba38d1bc" alt=""
アプリをビルドして起動後、暫く経つと Firebase console のダッシュボードにデータが表示されるようになります。
Realtime Database のデータを取得して表示
Realtime Databse にデータを挿入します。
key は「message」、value は 「こんにちは!」とします。
data:image/s3,"s3://crabby-images/1272d/1272d2b04c97352a35051f4cddec12cf9e21c498" alt=""
ContentView.swift を下記のように修正します。
import SwiftUI
import FirebaseDatabase
struct ContentView: View {
@State var message = ""
var body: some View {
VStack {
Text(message)
.padding()
}.onAppear {
let ref = Database.database().reference()
ref.child("message").getData { (error, snapshot) in
if let error = error {
print("Error getting data \(error)")
}
else if snapshot.exists() {
guard let message = snapshot.value as? String else {
return
}
self.message = message
}
else {
print("No data available")
}
}
}
}
}
struct ContentView_Previews: PreviewProvider {
static var previews: some View {
ContentView()
}
}
アプリをビルドして実行すると、画面に「こんにちは!」と表示されます。
data:image/s3,"s3://crabby-images/7e865/7e865e08ac6e080614dfcc5333869d33c0c51a7a" alt=""
Realtime Database のデータを更新
ContentView.swift を下記のように修正します。
ラベルとテキストフィールドに変更し、ボタンを追加しました。
import SwiftUI
import FirebaseDatabase
struct ContentView: View {
@State var message = ""
var body: some View {
VStack {
TextField("", text: $message)
.textFieldStyle(RoundedBorderTextFieldStyle())
Button(action: {
let ref = Database.database().reference()
ref.child("message").setValue(message)
}) {
Text("保存")
}
}.onAppear {
let ref = Database.database().reference()
ref.child("message").getData { (error, snapshot) in
if let error = error {
print("Error getting data \(error)")
}
else if snapshot.exists() {
guard let message = snapshot.value as? String else {
return
}
self.message = message
}
else {
print("No data available")
}
}
}
}
}
struct ContentView_Previews: PreviewProvider {
static var previews: some View {
ContentView()
}
}
アプリを起動して、テキストフィールドの値を書き換えて「保存」ボタンを押すと値が更新されます。
そのままでは値が本当に更新されたかわからないので、一度アプリを終了させて再度起動してみてください。
data:image/s3,"s3://crabby-images/41f46/41f46257fe4a18fa537ba258d776737484a5f449" alt=""
値が変わりましたね。
Firebase console で確認すると、実際に値が更新されていることがわかります。
data:image/s3,"s3://crabby-images/88725/88725381564d77a9be627a0edde20e04a12bb4ce" alt=""
まとめ
SwiftUI で Firebase Realtime Database を利用する手順を解説してきました。
非常に少ないコードかつある程度までなら無料で外部データベースとの連携ができるので、サクッとデータを保存、共有したい場合などには便利だと思います。
なお今回のコードは Realtime Database の説明に集中するために view ファイルに直接データの読み込み処理などを書いていますが、実際は model クラスなどを使って書くのが良いです。