🚒
Flutter * Firebase連携初手詰まりのTips
はじめに
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