🚒

Flutter * Firebase連携初手詰まりのTips

2020/11/23に公開

はじめに

Firestore * Flutter連携の公式ドキュメントソースがそのまま動かなかったので、エラーを修正して動いたコードをシェアしようと思う。(2020/11/23現在)

公式ドキュメントはこちら。

Firebase for Flutter | Google Codelabs

エラーの内容

ドキュメントに記載されているソースを動かすと以下のエラーが出る。

No Firebase App '[DEFAULT]' has been created - call Firebase.initializeApp()

どうやら2020年8月ごろから動かなくなっているようで、現在はinitializeApp()を呼び出す必要があるとのこと。

No Firebase App '[DEFAULT]' has been created - call Firebase.initializeApp() in Flutter and Firebase

動くサンプルソース

上記ページにあるように、pubspec.yamlを修正することが前提。

import 'package:cloud_firestore/cloud_firestore.dart';
import 'package:firebase_core/firebase_core.dart';
import 'package:flutter/material.dart';

void main() async {
  WidgetsFlutterBinding.ensureInitialized(); 
  await Firebase.initializeApp(); // ここ大事!
  runApp(MyApp());
}

class MyApp extends StatelessWidget {
  
  Widget build(BuildContext context) {
    return MaterialApp(
      title: 'Baby Names',
      home: MyHomePage(),
    );
  }
}

class MyHomePage extends StatefulWidget {
  
  _MyHomePageState createState() {
    return _MyHomePageState();
  }
}

class _MyHomePageState extends State<MyHomePage> {
  
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(title: Text('Baby Name Votes')),
      body: _buildBody(context),
    );
  }

  Widget _buildBody(BuildContext context) {
    return StreamBuilder<QuerySnapshot>(
      stream: Firestore.instance.collection('baby').snapshots(),
      builder: (context, snapshot) {
        if (!snapshot.hasData) return LinearProgressIndicator();

        return _buildList(context, snapshot.data.documents);
      },
    );
  }

  Widget _buildList(BuildContext context, List<DocumentSnapshot> snapshot) {
    return ListView(
      padding: const EdgeInsets.only(top: 20.0),
      children: snapshot.map((data) => _buildListItem(context, data)).toList(),
    );
  }

  Widget _buildListItem(BuildContext context, DocumentSnapshot data) {
    final record = Record.fromSnapshot(data);

    return Padding(
      key: ValueKey(record.name),
      padding: const EdgeInsets.symmetric(horizontal: 16.0, vertical: 8.0),
      child: Container(
        decoration: BoxDecoration(
          border: Border.all(color: Colors.grey),
          borderRadius: BorderRadius.circular(5.0),
        ),
        child: ListTile(
          title: Text(record.name),
          trailing: Text(record.votes.toString()),
          onTap: () =>
              record.reference.updateData({'votes': FieldValue.increment(1)}),
        ),
      ),
    );
  }
}

class Record {
  final String name;
  final int votes;
  final DocumentReference reference;

  Record.fromMap(Map<String, dynamic> map, {this.reference})
      : assert(map['name'] != null),
        assert(map['votes'] != null),
        name = map['name'],
        votes = map['votes'];

  Record.fromSnapshot(DocumentSnapshot snapshot)
      : this.fromMap(snapshot.data(), reference: snapshot.reference);

  
  String toString() => "Record<$name:$votes>";
}

class User {
  final String userName;
  final int userId;
  final int count;
  final DocumentReference reference;

  User.fromMap(Map<String, dynamic> map, {this.reference})
      : assert(map['user_name'] != null),
        assert(map['user_id'] != null),
        assert(map['count'] != null),
        userName = map['user_name'],
        userId = map['user_id'],
        count = map['count'];

  User.fromSnamshot(DocumentSnapshot snapshot)
      : this.fromMap(snapshot.data(), reference: snapshot.reference);

  
  String toString() => "User<$userName:$count>";
}

終わりに

Flutterは公式ドキュメントが充実しているので、単体では苦労することは少ないが、連携周りで情報が不足する場面があるなという印象があるので、コミュニティに多少なりとも貢献できたら嬉しい。

2020/11/25追記

と言いつつ早速動かなくなっていたのでソースコードを編集しました。

Discussion