2024年12月04日

Agoraで独自のオンラインレッスンアプリを作る

Agoraで独自のオンラインレッスンアプリを作る

※この投稿は、Agoraの日本代理店であるブイキューブが、Agoraブログを翻訳した記事です。


この記事では、自前のオンラインレッスンアプリの作成方法を紹介いたします。教師としてレッスン開催、または生徒として授業料の支払いやレッスン受講などをこのアプリにて完結させることができます。すべてのビデオレッスンはプラットフォーム上で実施されるので、認証、ユーザー管理や支払い機能の他に多様なリアルタイムの機能も含まれる包括的なアプリになります。

使用ツール一覧

このプロジェクトに以下のツールを使用します。

  • Flutter
  • Agora
  • Firebase
  • Riverpod
  • Apple Payのpayパッケージ
  • 他のFlutterパッケージ(image_picker, video_player, intl)
  • PythonとFlaskで構築されたカスタムバックエンド

Flutterアプリ向けのソースコードはこちら、バックエンドのコードはこちらをご参照ください。この記事では、アプリを構築するにあたって一般的な構造について説明します。それに加えて、Agoraの録画(Cloud Recording)および文字起こし(Real-Time Transcription)機能を紹介します。

概要

このtutor(オンラインレッスン)アプリにおいて、ユーザーのログインまたはサインアップはFirebase認証を利用します。RiverpodとFirebase Cloud Firestoreを使ってユーザー状態の管理や更新を行います。ユーザー設定で、教師または生徒のアカウントを切り替えることができます。どちらのアカウントでも支払いやレッスンに参加することができますが、教師アカウントにはレッスンを開催する権限も付与されます。

レッスンが開始されると、ユーザーはAgoraの文字起こし機能で音声をリアルタイムでテキスト化した文字を見れます。また、自分のカメラを調整することもできます。

ファイルの構成

lib
├── models
│   ├── recording
│   ├── session
│   └── user
├── pages
│   ├── home
│   ├── recordings
│   ├── class
│   ├── create
│   ├── settings
│   ├── signin
│   └── signup
├── protobuf
├── providers
│   └── user_provider
├── consts
└── main

アーキテクチャ

アーキテクチャ

build-your-own-tutoring-app-image003

ユーザーフロー

ユーザーフロー

アプリの初期画面

ユーザーがこちらでログインやサインアップを行う

アプリの初期画面

教師と生徒のホーム画面

教師と生徒のホーム画面

ナビゲーションと設定画面

ナビゲーションと設定画面

オンラインレッスンの風景

オンラインレッスンの風景

Apple Pay

Apple Pay

録画

録画

Agoraの概要

このtutor(オンラインレッスンアプリ)の主な特徴としては、教師と生徒がオンライン形式でビデオ通話レッスンの実施を可能にすることです。これに必要な機能はAgoraで実装します。Agoraはリアルタイムのコミュニケーションプラットフォームとなり、ビデオ通話、音声通話やライブ配信をアプリに組み込むことができます。また、このオンラインレッスンアプリでリアルタイムのコミュニケーションを行うに必要なすべての機能をサポートします。

Agoraの機能

ビデオ通話

オンラインレッスンを実現するにあたってビデオ通話機能はアプリの最も重要な部分と言えます。これについては、ビデオ通話画面の実装にagora_uikitパッケージを使っています。このパッケージにAgora SDKを組み込んでいてUIもすでにできているので、自分で作る手間が省けます。

このパッケージを使うにはAgoraのアカウントが必要です。こちらのAgoraコンソールページにサインアップしてください。Agoraのアカウントを登録してから、コンソール上でアプリ用のプロジェクトを作成しプロジェクトのAppID(※)を取得してください。このAppIDはAgora SDKに接続するためのキーとなります。
※AppIDは各プロジェクトに一意の識別子として自動的に割り当てられるものとなります。

iOSとAndroid端末にてAgora Video SDKを実行するための権限を追加しておいてください。

以下のコマンドを叩いてパッケージを追加します。

flutter pub add agora_uikit

本番環境のようなサービスに対して高いセキュリティが求められる場面には、トークン発行専用のサーバーを設置する必要があります。このサーバーとAgoraのUIキットとの紐付けを行います。トークン発行専用のサーバーについて詳しくはこちらをご覧ください。

上記のセットアップが完了すると、class.dartファイルを作成し以下のコードを追加してください。

import 'package:agora_uikit/agora_uikit.dart';
import 'package:flutter/material.dart';

class ClassCall extends StatefulWidget {
  const ClassCall({Key? key}) : super(key: key);
  
  @override
  State<ClassCall> createState() => _ClassCallState();
}

class _ClassCallState extends State<ClassCall> {
  final AgoraClient client = AgoraClient(
    agoraConnectionData: AgoraConnectionData(
      appId: "<--Add your App Id here-->",
      channelName: "test",
      username: "user",
      tokenUrl: "<--Add your token server url here-->",
    ),
  );

  @override
  void initState() {
    super.initState();
    initAgora();
  }
    
  void initAgora() async {
    await client.initialize();
  }
  
  @override
  Widget build(BuildContext context) {
    return Scaffold(
        appBar: AppBar(
            title: const Text('Classroom'),
        ),
        body: SafeArea(
            child: Stack(
            children: [
                AgoraVideoViewer(
                    client: client,
                    layoutType: Layout.floating,
                ),
                AgoraVideoButtons(
                    client: client,
                ),
            ],
            ),
        ),
    );
  }
}

通話が終了してユーザーがチャンネルから退室する時に、スワイプで戻ったりアプリのメニューバーに戻るボタンを設置したりするようなことせずに通話終了ボタンを押すと通話画面から離れる機能を実装します。これについてまずはAppBar()ウィジェットを削除します。

次にAgoraVideoButtonsウィジェットにonDisconnect関数を追加します。ユーザーが通話終了ボタンを押すと、この関数が呼び出されます。この関数の中で、Navigator.pop(context)をコールすることでユーザーが現在の画面から離れて元の画面に戻ることができます。

 

こうすると、アプリのメニューバーに戻るボタンを設けることなく、ユーザーは通話終了ボタンを押すだけで元の画面に戻れます。そして、端末システム側の戻るボタンも同様の制御が必要になります。iOSとAndroid端末を利用する場合は、システムの戻るボタンをタップする、またはスワイプで元の画面に戻ることができます。このような操作を制御するためにWillPopScopeウィジェットを使います。AgoraのScaffoldウィジェットをこれに組み込むと戻るボタンのデフォルトの動作をオーバーライドできます。onWillPop関数からFuture<bool>を返します。もしこの値をtrueで返すと、戻るボタンは通常通り動作します。

falseを返せば、戻るボタンは動作しません。onWillPop関数をカスタマイズして様々な条件で元の画面を表示することもできるが、今回のユースケースでは常にfalseを返すように設定します。

録画機能

AgoraのCloud Recording(クラウド録画)機能は自前のデータベースに接続して、アプリでオンラインレッスンを録画することができます。録画ファイルはお客様のデータベースに保存されます。これにAgora独自のバックエンドサービスと、AgoraのCloud Recording機能用のRESTful APIを使います。

セキュリティの観点からAgora RESTfull API利用のためのCustomer IDとCustomer Secretをアプリに保存しないので、バックエンドサービスを利用します。このサーバーはPythonとFlaskで構築されています。コードはこちらをご参照ください。この記事では、Agora APIに関わる部分だけを紹介するので、Flaskでゼロからサーバーを構築することについてもっと知りたい場合は、この動画をおすすめします。

今回はstart-recordingとstop-recordingの関数をバックエンドサーバーに実装します。

録画開始

バックエンドがどのチャンネルで録画タスクを実行するかを分かるためにstart-recordingのエンドポイント(Endpoint)に対象チャンネルの情報を設定する必要があります。録画が開始されると、この録画はどこに保存されるかを把握できるようにユーザー(呼び出し元)にエンドポイント、SID、リソースID(resourceId)を送ります。

また、録画タスクを実行するための最初のステップとしては、Cloud Recording用のリソースを作ることです。これについてまずAgora Console上でAgora RESTfull API利用のためのCustomer IDとCustomer Secretを生成して、この二つの情報でCloud Recording用のリソースを作るための認証情報を生成します。

録画開始

Cloud Recording用のリソースの生成に必要な認証情報を取得するにあたって、以下のコードでCustomer IDとCustomer Secretをエンコーディングする必要があります。

def generate_credential():
    # Generate encoded token based on customer key and secret
    credentials = CUSTOMER_KEY + ":" + CUSTOMER_SECRET
    base64_credentials = base64.b64encode(credentials.encode("utf8"))
    credential = base64_credentials.decode("utf8")
    return credential

それから、この認証情報でacquireを実行してCloud Recording用のリソースを作成します。このメソッドにてresourceId(録画実行ID)が発行されます。このresourceIdで録画タスク(start Cloud Recording)を実行します。

def generate_resource(channel):
    payload = {
        "cname": channel,
        "uid": str(UID),
        "clientRequest": {}
    }
    headers = {}
    
    headers['Authorization'] = 'basic ' + credential
    headers['Content-Type'] = 'application/json'
    headers['Access-Control-Allow-Origin'] = '*'

    url = f"https://api.agora.io/v1/apps/{APP_ID}/cloud_recording/acquire"
    res = requests.post(url, headers=headers, data=json.dumps(payload))

    data = res.json()
    resourceId = data["resourceId"]
    return resourceId

これで録画を開始する準備ができました。前述したセットアップの他に、必要な設定について詳しくはこちらをご覧ください。

今回の録画設定は以下を示します。

  • 今回は、mixモード(Composite recording mode:複合モード)で録画するので、各ユーザ(UID)の映像を結合して一つのファイルにまとめられます。
  • 録画ファイルをAWS上のagoraフォルダに保存します。
  • mp4形式で録画ファイルを保存します。
def start_cloud_recording(channel):
    resource_id = generate_resource(channel)
    url = f"https://api.agora.io/v1/apps/{APP_ID}/cloud_recording/resourceid/{resource_id}/mode/mix/start"
    payload = {
        "cname": channel,
        "uid": str(UID),
        "clientRequest": {
            "token": TEMP_TOKEN,
            "recordingConfig": {
                "maxIdleTime": 3,
            },
            "storageConfig": {
                "secretKey": SECRET_KEY,
                "vendor": 1,  # 1 is for AWS
                "region": 1,
                "bucket": BUCKET_NAME,
                "accessKey": ACCESS_KEY,
                "fileNamePrefix": [
                    "agora",
                ]
            },
            "recordingFileConfig": {
                "avFileType": [
                    "hls",
                    "mp4"
                ]
            },
        },
    }
    headers = {}
    headers['Authorization'] = 'basic ' + credential
    headers['Content-Type'] = 'application/json'
    headers['Access-Control-Allow-Origin'] = '*'
    res = requests.post(url, headers=headers, data=json.dumps(payload))
    data = res.json()
    sid = data["sid"]
    return resource_id, sid

ここまでバックエンドサービスの準備が完了したので、アプリ側にAgoraのCloud Recording機能を実装するステップに入ります。まずはAgoraClientのcloudRecordingUrl引数にリンクを追加します。そして、AgoraVideoButtonsウィジェットにcloudRecordingEnabled: trueを設定します。

録画終了

録画を終了するにあたってバックエンドに他の関数を実装する必要もあるが、これはUIキットでカバーされているのでフロントエンド側で特に作業なしで問題ありません。あとは、/stop-recording/<--Channel Name-->/<--SID-->/<--Resource ID-->のフォーマットに従うエンドポイントが必要なだけです。

ここで重要なのは、当該録画の情報、特にmp4ファイルのリンクをエンドユーザーに返すことです。

def stop_cloud_recording(channel, resource_id, sid):
    url = f"https://api.agora.io/v1/apps/{APP_ID}/cloud_recording/resourceid/{resource_id}/sid/{sid}/mode/mix/stop"
    
    headers = {}
    headers['Authorization'] = 'basic ' + credential
    headers['Content-Type'] = 'application/json;charset=utf-8'
    headers['Access-Control-Allow-Origin'] = '*'
    payload = {
        "cname": channel,
        "uid": str(UID),
        "clientRequest": {
        }
    }
    res = requests.post(url, headers=headers, data=json.dumps(payload))
    data = res.json()
    resource_id = data['resourceId']
    sid = data['sid']
    server_response = data['serverResponse']
    mp4_link = server_response['fileList'][0]['fileName']
    m3u8_link = server_response['fileList'][1]['fileName']
    formatted_data = {'resource_id': resource_id, 'sid': sid,
                      'server_response': server_response, 'mp4_link': mp4_link, 'm3u8_link': m3u8_link}
    return formatted_data

アーカイブしたレッスンの録画ファイルを確認

ビデオ通話が終了後にいつでも録画が見れるには、まずファイルの保管場所(パス)のリンクを保存しておく必要があります。このアプリがFirebase認証とFirestoreストレージでビルドされているので、今回はFirestoreに録画のリンクを保存します。

AgoraClientのcloudRecordingCallbackの関数は録画のmp4リンクを返してくれます。

まずは、録画について我々に必要なすべての情報を保持するデータクラスを作成します。

class Recording {
    final String url;
    final String sessionId;
    final String date;
    Recording({
      required this.url,
      required this.sessionId,
      required this.date,
    });
  }

次に、このアプリはUserProviderでビルドされているので、StateNotifierに録画を保存するための関数をセットアップします。

Future<void> storeRecording(Recording recording) {
    return _firestore
        .collection("users")
        .doc(state.id)
        .collection("recordings")
        .add(
          recording.toMap(),
        );
  }

そして、cloudRecordingCallbackでこの関数を呼び出します。

ref.read(userProvider.notifier).storeRecording(
    Recording(
      url: mp4,
      sessionId: widget.sessionId,
      date:
          "${DateFormat.yMMMMd('en_US').format(DateTime.now())}\n${DateFormat("hh:mm a").format(DateTime.now())}",
    ),
  );

すべての録画をリスト化してユーザーに送ります。ユーザーは一覧から録画をクリックして内容を再生することができます。この実装方法としては、必要なすべての情報を取得するためにuser providerに他の関数を追加します。

Future<List<Recording>> getRecordings() async {
    QuerySnapshot response = await _firestore
        .collection("users")
        .doc(state.id)
        .collection("recordings")
        .get();
    List<Recording> recordings = [];
    for (DocumentSnapshot snapshot in response.docs) {
      recordings
          .add(Recording.fromMap(snapshot.data() as Map<String, dynamic>));
    }
    return recordings;
  }

次に、Drawerウィジェットからアクセスできる録画リスト(ListView)を表示します。

class RecordingList extends ConsumerWidget {
    const RecordingList({super.key});
    @override
    Widget build(BuildContext context, WidgetRef ref) {
      return Scaffold(
        appBar: AppBar(
          title: const Text('Recordings'),
        ),
        body: FutureBuilder(
            future: ref.watch(userProvider.notifier).getRecordings(),
            builder: (context, snapshot) {
              if (snapshot.hasData) {
                return ListView.builder(
                  itemCount: snapshot.data!.length,
                  itemBuilder: (context, index) {
                    return ListTile(
                      title: Text(snapshot.data![index].url),
                      subtitle: Text(snapshot.data![index].date),
                      onTap: () {
                        Navigator.push(
                          context,
                          MaterialPageRoute(
                            builder: (context) => Recording(
                              url: snapshot.data![index].url,
                            ),
                          ),
                        );
                      },
                    );
                  },
                );
              } else {
                return const Center(child: CircularProgressIndicator());
              }
            }),
      );
    }
  }

それから、video_playerパッケージを使って、ビデオを新しい画面で表示します。

class Recording extends StatefulWidget {
    final String url;
    const Recording({super.key, required this.url});
    @override
    State<Recording> createState() => _RecordingState();
  }
  
  class _RecordingState extends State<Recording> {
    late VideoPlayerController _controller;
    late Future<void> _initializeVideoPlayerFuture;
    @override
    void initState() {
      super.initState();
      _controller = VideoPlayerController.network(
        "https://agora-server.s3.us-east-2.amazonaws.com/${widget.url}",
      );
      _initializeVideoPlayerFuture = _controller.initialize();
      _controller.setLooping(true);
    }
    @override
    void dispose() {
      _controller.dispose();
      super.dispose();
    }
    @override
    Widget build(BuildContext context) {
      return Scaffold(
        appBar: AppBar(),
        body: FutureBuilder(
          future: _initializeVideoPlayerFuture,
          builder: (context, snapshot) {
            if (snapshot.connectionState == ConnectionState.done) {
              return AspectRatio(
                aspectRatio: _controller.value.aspectRatio,
                child: VideoPlayer(_controller),
              );
            } else {
              return const Center(
                child: CircularProgressIndicator(),
              );
            }
          },
        ),
        floatingActionButton: FloatingActionButton(
          onPressed: () {
            setState(() {
              if (_controller.value.isPlaying) {
                _controller.pause();
              } else {
                _controller.play();
              }
            });
          },
          child: Icon(
            _controller.value.isPlaying ? Icons.pause : Icons.play_arrow,
          ),
        ),
      );
    }
  }
  

文字起こし機能

リアルタイムの文字起こし機能の利用方法はCloud Recording機能に似ています。この機能でもバックエンドサービスと組み合わせてAgora RESTful APIを使います。

セキュリティの観点からAgora RESTfull API利用のためのCustomer IDとCustomer Secretをアプリに保存しないので、バックエンドサービスを利用します。このサーバーはPythonとFlaskで構築されています。コードはこちらをご参照ください。この記事では、Agora APIに関わる部分だけを紹介するので、Flaskでゼロからサーバーを構築することについてもっと知りたい場合は、この動画をおすすめします。

今回はstart-transcribingとstop-transcribingの関数をバックエンドサーバーに実装します。

文字起こし機能を起動

バックエンドがどのチャンネルで文字起こしタスクを実行するかを分かるためにstart-transcribingのエンドポイントに対象チャンネルの情報を設定する必要があります。文字起こしのタスクが開始されると、音声をテキスト化したデータはどこに保存されるかを把握できるようにユーザー(呼び出し元)にエンドポイント、タスクID(task ID)、トークン(builder token)を送ります。

また、文字起こしタスクを実行するための最初のステップとしては、文字起こし用のリソースを作ることです。これに必要な手順は前述したようにまずAgora Console上でAgora RESTfull API利用のためのCustomer IDとCustomer Secretを生成して、この二つの情報で文字起こし用のリソースを作るための認証情報を生成します。

build-your-own-tutoring-app-image011

def rtt_generate_resource(channel):
    payload = {
        "instanceId": channel,
    }
    headers = {}
    headers['Authorization'] = 'basic ' + credential
    headers['Content-Type'] = 'application/json'
    headers['Access-Control-Allow-Origin'] = '*'
    url = f"https://api.agora.io/v1/projects/{APP_ID}/rtsc/speech-to-text/builderTokens"
    res = requests.post(url, headers=headers, data=json.dumps(payload))
    data = res.json()
    tokenName = data["tokenName"]
    return tokenName

この認証情報でacquireを実行して文字起こし用のリソースを作成します。このメソッドにてtokenName(トークン名)が発行されます。このtokenNameで文字起こし機能を起動します。

def rtt_generate_resource(channel):
    payload = {
        "instanceId": channel,
    }
    headers = {}
    headers['Authorization'] = 'basic ' + credential
    headers['Content-Type'] = 'application/json'
    headers['Access-Control-Allow-Origin'] = '*'
    url = f"https://api.agora.io/v1/projects/{APP_ID}/rtsc/speech-to-text/builderTokens"
    res = requests.post(url, headers=headers, data=json.dumps(payload))
    data = res.json()
    tokenName = data["tokenName"]
    return tokenName

これで文字起こし機能を起動する準備ができました。上記の他に必要な設定について詳しくはこちらをご覧ください。

文字起こしについての設定は以下を示します。

  • 今回の利用言語は英語とスペイン語です。
  • 文字に起こしたテキストデータをrttフォルダに保存します。
def start_transcription(channel):
    tokenName = rtt_generate_resource(channel)
    url = f"https://api.agora.io/v1/projects/{APP_ID}/rtsc/speech-to-text/tasks?builderToken={tokenName}"
    payload = {
        "audio": {
            "subscribeSource": "AGORARTC",
            "agoraRtcConfig": {
                "channelName": channel,
                "uid": "101",
                "token": "{{channelToken}}"
                "channelType": "LIVE_TYPE",
                "subscribeConfig": {
                    "subscribeMode": "CHANNEL_MODE"
                },
                "maxIdleTime": 60
            }
        },
        "config": {
            "features": [
                "RECOGNIZE"
            ],
            "recognizeConfig": {
                "language": "en-US,es-ES",
                "model": "Model",
                "output": {
                    "destinations": [
                        "AgoraRTCDataStream",
                        "Storage"
                    ],
                    "agoraRTCDataStream": {
                        "channelName": channel,
                        "uid": "101",
                        "token": "{{channelToken}}"
                    },
                    "cloudStorage": [
                        {
                            "format": "HLS",
                            "storageConfig": {
                                "accessKey": ACCESS_KEY,
                                "secretKey": SECRET_KEY,
                                "bucket": BUCKET_NAME,
                                "vendor": 1,
                                "region": 1,
                                "fileNamePrefix": [
                                    "rtt"
                                ]
                            }
                        }
                    ]
                }
            }
        }
    }
    headers = {}
    headers['Authorization'] = 'basic ' + credential
    headers['Content-Type'] = 'application/json'
    res = requests.post(url, headers=headers, data=json.dumps(payload))
    data = res.json()
    taskID = data["taskId"]
    return taskID, tokenName

以下のコール内容をAgoraのStatefulWidgetのinitStateに追加することで、プロセスを起動します。

final response = await http.post(
    Uri.parse(
        'https://agora-server-hr4b.onrender.com/start-transcribing/main'),
  );
  taskId = jsonDecode(response.body)['taskId'];
  builderToken = jsonDecode(response.body)['builderToken'];

Agoraのイベントに基づいてアクションをカスタマイズ

エンドポイントをコールするだけではアプリに何も表示されないので、こちらでプロジェクトのprotobufをセットアップする必要があります。

以下はAgoraリアルタイムの文字起こしのテンプレートになります。

syntax = "proto3";

    package agora.audio2text;
    option java_package = "io.agora.rtc.audio2text";
    option java_outer_classname = "Audio2TextProtobuffer";
    
    message Text {
      int32 vendor = 1;
      int32 version = 2;
      int32 seqnum = 3;
      int32 uid = 4;
      int32 flag = 5;
      int64 time = 6;
      int32 lang = 7;
      int32 starttime = 8;
      int32 offtime = 9;
      repeated Word words = 10;
    }
    message Word {
      string text = 1;
      int32 start_ms = 2;
      int32 duration_ms = 3;
      bool is_final = 4;
      double confidence = 5;
    }

Protocol Bufferのセットアップが完了すると、onStreamMessageのコールバックからテキスト化したデータを取得し、protobufを通してこのテキストをリストに追加することができます。

agoraEventHandlers: AgoraRtcEventHandlers(
    onStreamMessage:
        (connection, remoteUid, streamId, data, length, sentTs) {
      protoText.Text sttText = protoText.Text.fromBuffer(data);
      if (sttText.words.isNotEmpty) {
        sttText.words.last.isFinal
            ? updateConversation(sttText.words.last.text)
            : null;
      }
    },
    onStreamMessageError:
        (connection, remoteUid, streamId, code, missed, cached) {
      print("error $code");
    },
  )

以下のviewをアプリの画面に追加すれば、文字がリアルタイムで表示されます。

Padding(
    padding: const EdgeInsets.only(bottom: 100.0, top: 200),
    child: ListView.builder(
      controller: _controller,
      itemBuilder: (context, index) {
        return ListTile(
          title: Text(
            conversation[index],
            style: const TextStyle(
              color: Colors.white,
            ),
          ),
        );
      },
      itemCount: conversation.length,
    ),
  ),

文字起こし機能を終了

通話終了後に最後のステップとしては、文字起こしタスクを終了することです。こちらで必要なのは、/stop-transcribing/<--Channel Name-->/<--Task ID-->/<--Builder Token-->のフォーマットに従うエンドポイントです。

def stop_transcription(task_id, builder_token):
    url = f"https://api.agora.io/v1/projects/{APP_ID}/rtsc/speech-to-text/tasks/{task_id}?builderToken={builder_token}"
    headers = {}
    headers['Authorization'] = 'basic ' + credential
    headers['Content-Type'] = 'application/json'
    payload = {}
    res = requests.delete(url, headers=headers, data=payload)
    data = res.json()
    return data

通話終了ボタンが押されるタイミングでFlutterアプリからこのエンドポイントをコールします。これにより、当該セッションのリアルタイムの文字起こし機能は終了になります。

http.get(Uri.parse(
    'https://agora-server-hr4b.onrender.com/stop-transcribing/$taskId/$builderToken'));

Agora

他のコア機能

Apple Payを統合

アプリでApple Payを利用するにあたってApple Payの利用業者の申請かつマーチャントID(merchant ID)を自前のアプリと紐付ける必要があります。これについての詳細は、こちらのApple Payガイドをご参照ください。

必要なセットアップが完了すると、Apple PayボタンのonPaymentResult引数でuser providerが定義しているjoinSession関数を実行します。この関数はレッスンを購入したユーザー(生徒)の受講リストにセッションを追加すると同時に、このユーザーのIDを当該レッスンの受講生リストに追加します。

Future<void> joinSession(String sessionId, bool isLecture) async {
    await _firestore.collection("users").doc(state.id).update({
      'upcomingSessions': FieldValue.arrayUnion([sessionId])
    });
    if (!isLecture) {
      await _firestore.collection("sessions").doc(sessionId).update({
        'students': FieldValue.arrayUnion([state.id])
      });
    }
    state = state.copyWith(
        user: state.user.copyWith(
            upcomingSessions: [...state.user.upcomingSessions, sessionId]));
  }

無料でAgoraをご体験いただけます。

こちらからサインアップし、リアルタイムの音声とビデオ通話を実装してみましょう!
利用分数が多くなければ料金は発生しません。

Agoraの料金シミュレーションやお問合せ、導入についてのサポートなど、
お気軽にお問い合わせ下さい。日本語にて対応いたします。

料金シミュレーション・ご相談
ブイキューブ

執筆者ブイキューブ

Agoraの日本総代理店として、配信/通話SDKの提供だけでなく、導入支援から行い幅広いコミュニケーションサービスに携わっている。

関連記事

先頭へ戻る