自分の備忘録用記事です。
※2021/03/03 こちらのページに従い、firebase_coreを使うパターンも記載しました。
下記の記事を参考にしてます。
Flutter projectの作成
まずはFlutter projectの作成。
今回はAndroid Studioメインで作成します。
(Flutterのプラグインは導入済みの想定。まだの人はこちらから)
Start a new Flutter projectをクリック。
data:image/s3,"s3://crabby-images/3c1fe/3c1fea0df917f3bfc3d481c7381e4cc16d2ca3c3" alt="スクリーンショット 2020-04-15 17.35.45.png"
Flutter Applicationを選択し、Nextをクリック。
data:image/s3,"s3://crabby-images/ba94d/ba94d883e2a16726a52929a75f22c1b43e5d25fd" alt="スクリーンショット 2020-04-16 12.20.28.png"
Packeage Nameを入力し、Finishをクリック。。
※Packeage Nameは任意で良いです。
data:image/s3,"s3://crabby-images/f5dbd/f5dbd715a7bdd94978c4d926bc7070830db1f2a8" alt="スクリーンショット 2020-04-15 17.38.28.png"
プロジェクトが立ち上がったら、pubspec.yamlのdependenciesに下記の文言を追加。
(Android StudioならShiftキーを2回押すとファイル検索できるので、pubspec.yamlと入力してください)
dependencies:
flutter:
sdk: flutter
// この行を追加
cloud_firestore: ^0.13.0+1
// firebase_coreを使う場合、こちらの行に変更する
// firebase_core: "^1.0.0"
// cloud_firestore: "^1.0.0"
pubspec.yamlを編集したら、flutter packages getを実施するか、
Android Studioであれば画面右上のPub getをクリック。
data:image/s3,"s3://crabby-images/df1da/df1dae42020fba91d2e25ccc1b9cad91134a0017" alt="スクリーンショット 2020-04-16 13.02.52.png"
packagesをインストールし終えたら、Android emulatorでアプリを実行し
デフォルトのFlutter Appが立ち上がることを確認します。
data:image/s3,"s3://crabby-images/abfa8/abfa8d836848448c6409770b67263c89ca536432" alt="スクリーンショット 2020-04-16 13.23.54.png"
※Cannot fit requested classes in a single dex fileエラーが出た場合、minSdkVersionのバージョンを16->21に変更し、再度アプリを実行してください。
// minSdkVersion 16
minSdkVersion 21
サンプルコードの実装
main.dartの既存コードを削除し、下記のコードを貼り付けます。
実行するとdummySnapshotの値がListViewで表示されるはずです。
コードの詳細は元ページを参照してください。
firebase_core利用なし
import 'package:cloud_firestore/cloud_firestore.dart';
import 'package:flutter/material.dart';
void main() => runApp(MyApp());
final dummySnapshot = [
{"name": "Filip", "votes": 15},
{"name": "Abraham", "votes": 14},
{"name": "Richard", "votes": 11},
{"name": "Ike", "votes": 10},
{"name": "Justin", "votes": 1},
];
class MyApp extends StatelessWidget {
@override
Widget build(BuildContext context) {
return MaterialApp(
title: 'Baby Names',
home: MyHomePage(),
);
}
}
class MyHomePage extends StatefulWidget {
@override
_MyHomePageState createState() {
return _MyHomePageState();
}
}
class _MyHomePageState extends State<MyHomePage> {
@override
Widget build(BuildContext context) {
return Scaffold(
appBar: AppBar(title: Text('Baby Name Votes')),
body: _buildBody(context),
);
}
Widget _buildBody(BuildContext context) {
return _buildList(context, dummySnapshot);
}
Widget _buildList(BuildContext context, List<Map> snapshot) {
return ListView(
padding: const EdgeInsets.only(top: 20.0),
children: snapshot.map((data) => _buildListItem(context, data)).toList(),
);
}
Widget _buildListItem(BuildContext context, Map data) {
final record = Record.fromMap(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: () => print(record),
),
),
);
}
}
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);
@override
String toString() => "Record<$name:$votes>";
}
firebase_core利用あり
import 'package:cloud_firestore/cloud_firestore.dart';
import 'package:firebase_core/firebase_core.dart';
import 'package:flutter/material.dart';
void main() => runApp(MyApp());
final dummySnapshot = [
{"name": "Filip", "votes": 15},
{"name": "Abraham", "votes": 14},
{"name": "Richard", "votes": 11},
{"name": "Ike", "votes": 10},
{"name": "Justin", "votes": 1},
];
class MyApp extends StatelessWidget {
@override
Widget build(BuildContext context) {
return MaterialApp(
title: 'Baby Names',
home: MyHomePage(),
);
}
}
class MyHomePage extends StatefulWidget {
@override
_MyHomePageState createState() {
return _MyHomePageState();
}
}
class _MyHomePageState extends State<MyHomePage> {
@override
Widget build(BuildContext context) {
return Scaffold(
appBar: AppBar(title: Text('Baby Name Votes')),
body: _buildBody(context),
);
}
Widget _buildBody(BuildContext context) {
return _buildList(context, dummySnapshot);
}
Widget _buildList(BuildContext context, List<Map> snapshot) {
return ListView(
padding: const EdgeInsets.only(top: 20.0),
children: snapshot.map((data) => _buildListItem(context, data)).toList(),
);
}
Widget _buildListItem(BuildContext context, Map data) {
final record = Record.fromMap(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: () => print(record),
),
),
);
}
}
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);
@override
String toString() => "Record<$name:$votes>";
}
data:image/s3,"s3://crabby-images/c8388/c8388b5bed3f97e8e7fd68807aeee9d85fcae22f" alt="スクリーンショット 2020-04-16 14.13.07.png"
Firebaseにプロジェクト作成
プロジェクトを作成をクリック。
(アカウント作成は省略しますので必要に応じて行ってください。)
data:image/s3,"s3://crabby-images/e78f0/e78f0ad51be5128e6954fab6fad005b7f83acb42" alt="スクリーンショット 2020-04-15 19.07.09.png"
プロジェクト名を入力。
※任意でOK。
data:image/s3,"s3://crabby-images/8d7fb/8d7fbbfbf7aa1a693e079d9ff1a8b1c0c3b38cdc" alt="スクリーンショット 2020-04-15 19.07.53.png"
下記の通り作成できればOKです。
data:image/s3,"s3://crabby-images/78fc1/78fc10da3995ce4db4053f7f0cff686415b2c5ab" alt="スクリーンショット 2020-04-15 19.09.45.png"
Androidアプリの設定
まずはAndroidの設定を追加します。
プロジェクトページ真ん中あたりのドロイド君をクリック。
data:image/s3,"s3://crabby-images/d8e28/d8e28a401ef358b50ef5b944142527dd81745ed9" alt="スクリーンショット 2020-04-15 19.10.15.png"
パッケージ名を入力。
※パッケージ名はFlutter projectを作成した時のものです。
ex)com.example.babynames
忘れた場合、android/app/src/main/AndroidManifest.xmlの一番上の方に記載されています。
data:image/s3,"s3://crabby-images/bc253/bc2530861ff8852a5738d7406b6da84bc2934877" alt="スクリーンショット 2020-04-15 19.18.44.png"
※ Google sign-inを使いたい場合、
デバッグ用の署名証明書 SHA-1も入力しておく必要があるみたいです。
ターミナルで下記のコマンドを実行すると
SHA1: DA:39:A3:EE:5E:6B:4B:0D:32:55:BF:EF:95:60:18:90:AF:D8:07:09 のように取得できるので、必要に応じて入力してください。
入力するとパスワードを聞かれますが、初期パスワードはandroidです。
keytool -list -v \
-alias androiddebugkey -keystore ~/.android/debug.keystore
アプリを登録したら、次はgoogle-services.jsonをダウンロードします。
data:image/s3,"s3://crabby-images/a8fa4/a8fa4deb325330139491f80005c46cfdf81b4e09" alt="スクリーンショット 2020-04-15 19.19.04.png"
ダウンロードしたgoogle-services.jsonをプロジェクトのandroid/app配下に配置。
(Android Studioを開き、android/appの部分にドラッグ&ドロップでOK)
google-services.jsonを配置したら、
android/app/build.gradleとandroid/build.gradleをそれぞれ編集。
dependencies {
// dependencies内に下記の一行を追加
implementation 'com.google.firebase:firebase-analytics:17.2.2'
}
...
// ファイルの一番下に下記の一行を追加
apply plugin: 'com.google.gms.google-services'
buildscript {
...
dependencies {
...
// 下記の文言を追加
classpath 'com.google.gms:google-services:4.3.3'
}
}
Androidエミュレータを起動し、下記の画面が表示されることを確認する。
data:image/s3,"s3://crabby-images/3ad08/3ad0842c9821435460b45196bfef1c19b9bf0805" alt="スクリーンショット 2020-04-16 11.07.20.png"
Firebaseも下記のような表示になるはずです。
data:image/s3,"s3://crabby-images/72363/72363b54335698360a9352441635d00341f6b2c2" alt="スクリーンショット 2020-04-16 10.43.30.png"
iOSアプリの設定
次にiOSの設定を追加します。
プロジェクトページ真ん中あたりのiOSをクリック。
data:image/s3,"s3://crabby-images/d8e28/d8e28a401ef358b50ef5b944142527dd81745ed9" alt="スクリーンショット 2020-04-15 19.10.15.png"
bundle idを入力。
※bundle idはAndroidのパッケージ名と同じです。
ex)com.example.babynames
data:image/s3,"s3://crabby-images/d21fc/d21fca309cf7af99cfef8aff29e0853d558efc9a" alt="スクリーンショット 2020-04-15 19.26.03.png"
アプリを登録したら、次はGoogleService-Info.plistをダウンロードします。
data:image/s3,"s3://crabby-images/90657/9065771b412eea6e71afeff75a56df0c2549b7b7" alt="スクリーンショット 2020-04-15 19.26.14.png"
ダウンロードしたら、ターミナルで下記のコマンドを入力し、Xcodeでプロジェクトを開きます。
open {プロジェクトファイルパス}/ios/Runner.xcworkspace
開いたら、まずはbundle idが正しく入力されているか確認します。
(bundle identifierが正しいことを確認してください)
data:image/s3,"s3://crabby-images/151c5/151c524c83484c512bf4ec9adabcdc87155773fd" alt="スクリーンショット 2020-04-16 10.00.32.png"
次にダウンロードしたGoogleService-Info.plistをRunner/Runner配下に配置。
(Runner/Runner直下にドラッグ&ドロップでOK)
data:image/s3,"s3://crabby-images/c3efa/c3efafe087f49eb0b7e0c233ce9a75c986cb17ee" alt="スクリーンショット 2020-04-16 10.05.51.png"
配置後は下記のようになります。
data:image/s3,"s3://crabby-images/9c59f/9c59fb2511d1e99bccfcc80f3fc44d2490cd6f37" alt="スクリーンショット 2020-04-16 10.06.19.png"
iOSエミュレータを起動し、Androidと同じ画面が表示されていることを確認する。
data:image/s3,"s3://crabby-images/df36e/df36e32fddff1791ca45becb5123de940bb40233" alt="スクリーンショット 2020-04-16 16.03.47.png"
data:image/s3,"s3://crabby-images/911f1/911f10c63776558df8976e151980b23df6cc5b7e" alt="スクリーンショット 2020-04-16 10.48.27.png"
Databaseの作成
投票数をアプリで共有するためにDatabaseを作成します。
左ペインからDatabaseを選択。
data:image/s3,"s3://crabby-images/0da0a/0da0a1a17394ab761029bdbf28b7ef41b322db38" alt="スクリーンショット 2020-04-16 10.53.22.png"
データベースの作成をクリック。
data:image/s3,"s3://crabby-images/ca423/ca423fdb28bdf489dbf7dad29f7ae44b21d2cc16" alt="スクリーンショット 2020-04-16 10.53.42.png"
テストモードで開始を選択し、次へをクリック。
data:image/s3,"s3://crabby-images/9e39c/9e39c8c431ededc73183c96c6e4162a8be0e9a76" alt="スクリーンショット 2020-04-16 10.54.11.png"
Cloud Firestoreのロケーションを選択。
※任意でOK
data:image/s3,"s3://crabby-images/743c2/743c229f4292940c95b49ee265abc629d3bc66eb" alt="スクリーンショット 2020-04-16 10.54.31.png"
コレクションを開始をクリックし、コレクションIDを入力。
※任意でOK(コレクションIDは後で使うので覚えておいてください)
data:image/s3,"s3://crabby-images/c4517/c451731bc84e34778053e0881cda82ec79467049" alt="スクリーンショット 2020-04-16 10.56.28.png"
リスト作成用のデータを登録していく。
詳細はキャプチャ参照
data:image/s3,"s3://crabby-images/1c546/1c546d1d374f365cdc4c987efe989b3116c6f500" alt="スクリーンショット 2020-04-16 16.39.20.png"
適当な数を追加し、下記のような形にします。
data:image/s3,"s3://crabby-images/bc14f/bc14fc16341ab155f1ce1cdaa9e506ec2dda3b9a" alt="スクリーンショット 2020-04-16 11.02.01.png"
投票アプリの実装
Android Studioを開き、main.dartの下記の部分を編集します。
firebase_core利用なし
// 削除
// final dummySnapshot = [
// {"name": "Filip", "votes": 15},
// {"name": "Abraham", "votes": 14},
// {"name": "Richard", "votes": 11},
// {"name": "Ike", "votes": 10},
// {"name": "Justin", "votes": 1},
// ];
Widget _buildBody(BuildContext context) {
// return _buildList(context, dummySnapshot); 削除
return StreamBuilder<QuerySnapshot>(
stream: Firestore.instance.collection('baby').snapshots(), // babyは各々のコレクションIDに変更してください
builder: (context, snapshot) {
if (!snapshot.hasData) return LinearProgressIndicator();
return _buildList(context, snapshot.data.documents);
},
);
}
// Widget _buildList(BuildContext context, List<Map> snapshot) { 変更
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, Map data) { 変更
Widget _buildListItem(BuildContext context, DocumentSnapshot data) {
// final record = Record.fromMap(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: () => print(record), 変更
onTap: () => record.reference.updateData({'votes': FieldValue.increment(1)}),
),
),
);
}
firebase_core利用あり
// 削除
// final dummySnapshot = [
// {"name": "Filip", "votes": 15},
// {"name": "Abraham", "votes": 14},
// {"name": "Richard", "votes": 11},
// {"name": "Ike", "votes": 10},
// {"name": "Justin", "votes": 1},
// ];
Widget _buildBody(BuildContext context) {
return StreamBuilder<QuerySnapshot>(
stream: FirebaseFirestore.instance.collection('baby').snapshots(), // babyは各々のコレクションIDに変更してください
builder: (context, snapshot) {
if (!snapshot.hasData) return LinearProgressIndicator();
return _buildList(context, snapshot.data.docs);
},
);
}
// Widget _buildList(BuildContext context, List<Map> snapshot) { 変更
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, Map data) { 変更
Widget _buildListItem(BuildContext context, DocumentSnapshot data) {
// final record = Record.fromMap(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: () => print(record), 変更
onTap: () => record.reference.update({'votes': FieldValue.increment(1)}),
),
),
);
}
}
以上で完了です。
アプリを起動し、Databaseとアプリのデータが連動していることを確認してみてください。