5
1

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

More than 1 year has passed since last update.

AndroidAdvent Calendar 2022

Day 21

開発初心者がサーバーレスにアプリを作ってみた(Kotlin)

Last updated at Posted at 2022-12-20

はじめに

Android Advent Calender2022の21日目の記事です。

普段はニフクラ mobile backendというアプリ開発のサービスを企画職として運営している立場なんですが、アドベントカレンダーを機にアプリを開発してみました。

初心者なのでコードが変だったりすると思いますが、「自分も最初はこんなもんだったな~」とか思って年末を締めくくるのもたまには良いですよね。そういうのんびりした気持ちでお読みください。

開発するもの

開発なんもわからん。コードもわからん。どうすっかなぁ~と悩んだ結果、4択のクイズアプリを開発する事にしました。コードは最後に載せておきます。
※ファイル減らすために1ファイルにたくさん書いてます。下手くそなコードですが一旦動けばよし。

今回は以下の機能が使えるアプリにしたいと思います。筆者はこれが実装出来たら拍手レベル。頑張ろう。

  • 新規会員登録/ログイン機能
  • データベースに入ってる問題と選択肢などを拾ってきて出題
  • 正誤判定

今回開発予定のアプリの全体像はこんな感じです。基本的にサーバー側の処理はニフクラ mobile backendの機能を使ってます。
image.png

開発環境

  • MacOS Monterey(バージョン 12.5.1)
  • Android Studio(dolphin)
    • 検証端末:Android emulator (Pixel2 API28)
  • mBaaSサービス
    • ニフクラ mobile backend (Kotlin SDK v1.2.0)
      • データストア機能、クエリ検索機能
      • 会員管理、認証機能(会員登録/ログイン機能のため)
  • 気合い(多分一番必要)

使った機能

  • Intent
    今回はログイン画面(ホーム)からクイズ画面への画面遷移も実装したいので、Intentを用います。
    ↓この記事が凄く参考になりました。書いてくれた人に感謝。
    【はじめてのKotlinプログラミング(7)】intent(画面遷移)

  • ViewBinding
    問題と選択肢の情報を画面に表示するために利用。以前はfindViewByIdが使われていたようですが、現在はこれがスタンダードらしいですね。モジュール単位で有効かどうか設定して、モジュールのbuild.gradleファイルに以下を記述して使用します。

app/build.gradle
android {
        ..(省略)
        viewBinding {
            enabled = true
        }
    }

また、実際に使用する時はActivityに以下を記述して使います。

各Activity.kt

class MainActivity : AppCompatActivity(){
  // lateinit で宣言しておくとonCreateメソッド内(あとで)で使える
  private lateinit var binding: ActivityMainBinding

//onCreate内
  override fun onCreate(savedInstanceState: Bundle?) {
    super.onCreate(savedInstanceState)

  //layoutファイル(xml)がactivity_mainなので、キャメル型 + Bindingで命名する
  binding = ActivityMainBinding.inflate(layoutInflater)
  setContentView(binding.root)
  }
}

アプリにボタンやテキストを実装する時に、idで呼び出して管理できるのは分かりやすいし超便利でした。

  • データストア機能
    ニフクラ mobile backendからAPI経由で呼び出す機能で、データの保存や取得、検索も簡単に出来ます。何よりサーバーレスに使えてSNSのアカウントですぐ始められるのでとてもラク。筆者くらい初心者のうちは確実に使った方が効率が良いんじゃなかろうか。

  • 新規会員登録・ログイン機能 (会員管理・認証機能)
    データストア同様にニフクラ mobile backendから呼び出して使います。今回は新規会員登録とログイン機能の部分で活用しますが、ユーザーごとに権限設定できるのも熱い。

開発の流れ

今回はバックエンド機能は開発せずにニフクラ mobile backendを使用しています。アコーディオンを開くとコードや実装手順を確認出来るようにしたので、時間がある人は開いて手順を確認してみてください。

ニフクラ mobile backend の初期設定

①ニフクラ mobile backendに無料登録
SNSアカウントを持っている人であれば以下の動画通りで1~2分でサクッと使えるようになります。
②Kotlin SDKのインストール
  • Kotlin SDK のリリースページより最新の Kotlin SDKをダウンロードする (解凍するとNCMB.jarとなる)

  • Android StudioプロジェクトのlibsフォルダにNCMB.jarを配置する
    image.png

  • app/build.gradleファイルに以下を追加する

app/build.gradle
dependencies {
    implementation 'com.squareup.okhttp3:okhttp:4.8.1'
    implementation 'com.google.code.gson:gson:2.3.1'
    api files('libs/NCMB.jar')

    //同期処理を使用する場合はこちらを追加していただく必要があります
    implementation 'org.jetbrains.kotlinx:kotlinx-coroutines-core:1.3.9'
}
  • 追加後、画面上部にSync Now (変更内容の同期)を行う※必ず実施する
③AndroidManifest.xmlの編集
  • タグの直前に以下のpermissionを追加する
AndroidManifest.xml
<uses-permission android:name="android.permission.INTERNET" />
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
④Kotlin SDKの初期化
  • ニフクラ mobile backendにアプリを作成したらAPIキーが発行されるので、.ktファイルのonCreateメソッド内に以下を記述し、"YOUR_APPLICATION_KEY", "YOUR_CLIENT_KEY"を書き換える
各Activity.kt
override fun onCreate(savedInstanceState: Bundle?) {
    super.onCreate(savedInstanceState)
    // APIキーの設定とSDK初期化
    NCMB.initialize(this, "YOUR_APPLICATION_KEY", "YOUR_CLIENT_KEY") //APIキーを設定して繋ぎこむ
   
    setContentView(R.layout.activity_main)
}
⑤必要なライブラリをインポート
  • ログイン機能やデータストア機能を実装する時はライブラリを使用するので、以下を冒頭に追記しておきます
各Activity.kt
import com.nifcloud.mbaas.core.NCMB //全機能必須
import com.nifcloud.mbaas.core.NCMBCallback //非同期処理を行う場合
import com.nifcloud.mbaas.core.NCMBObject //データストアを利用する場合
import com.nifcloud.mbaas.core.NCMBQuery //検索機能を利用する場合
import com.nifcloud.mbaas.core.NCMBException //例外処理を行う場合
import com.nifcloud.mbaas.core.NCMBUser //会員管理を利用する場合

サーバー側にクイズの元データ(.csv)を入れておく

今回はニフクラ mobile backendのデータストアにクイズの基データを格納するので、csvファイルを用意したらデータストアへのインポートを行います。今回は以下のようにcsvファイルを用意しました。
※データストアに文字列型で格納する場合は文字を""(ダブルクォーテーション)で囲む

これをメモ帳で開くとこんな感じで"""室町時代"""のようにダブルクォーテーションが3つ出現します。
このまま格納するとダブルクォーテーションが付いた状態で文字列が格納されてしまうので、メモ帳で編集します。

これを置換で以下のように1つに置換すれば綺麗になります。文字コードはUTF-8である事を確認して保存。

これをデータストアのクラス作成からインポートすると

こんな感じで格納されます。これをクイズとして呼び出せば完成ですね。

今回はExcelでcsvファイルに変換して用意しましたが、エディタなどで直接csvファイルを作ってしまえばダブルクォーテーションを修正するような作業も不要だと思います。

ホーム画面(MainActivity.kt)の実装

  • Button置いたりEditText置いたりしているだけなので、Layoutファイルに書いた内容は省略します
    • binding.XXX.textbinding.XXX.setOnClickListenerXXXはxmlファイルで指定したidです
ソースコードを表示する
MainActivity.kt
package com.example.a2022adventkotlin
import android.content.Intent
import androidx.appcompat.app.AppCompatActivity
import android.os.Bundle
import android.widget.Toast
import androidx.appcompat.app.AlertDialog
import com.example.a2022adventkotlin.databinding.ActivityMainBinding
import com.nifcloud.mbaas.core.NCMB
import com.nifcloud.mbaas.core.NCMBException
import com.nifcloud.mbaas.core.NCMBUser

class MainActivity : AppCompatActivity() {
    private lateinit var binding: ActivityMainBinding 

    private fun login(id:String,pw:String){
        // Userインスタンスの生成
        var user = NCMBUser()
        //ユーザー名・パスワードを設定
        user.userName = id //ユーザー名
        user.password = pw // パスワード
        try{
            // ユーザー名とパスワードでログイン
            user.login(id,pw)
            // ログインに成功した場合の処理
            val intent = Intent(this, QuizActivity::class.java)
            startActivity(intent)
        }
        catch(e:NCMBException){
            // ログインに失敗した場合の処理
            Toast.makeText(this, "ログイン失敗", Toast.LENGTH_LONG).show()
        }
    }
    private fun signUp(newid:String,newpw:String){
        // Userインスタンスの生成
        var user = NCMBUser()
        // ユーザー名・パスワードを設定
        user.userName = newid
        user.password = newpw
        // ユーザーの新規登録
        try {
            user.signUp(newid,newpw)
            // 新規登録に成功した場合の処理
            AlertDialog.Builder(this)
                .setTitle("会員登録完了")
                .setMessage("ホームよりログインしてください")
                .setPositiveButton("閉じる"){dialog, which ->}
                .show()
        }
        catch(e: NCMBException){
            // 新規登録に失敗した場合の処理
            AlertDialog.Builder(this)
                .setTitle("会員登録失敗")
                .setMessage("再度お試しください")
                .setPositiveButton("閉じる"){dialog, which ->}
                .show()
        }
    }

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)

        val appKey = "ニフクラ mobile backendからアプリケーションキーを入力"
        val cliKey = "ニフクラ mobile backendからクライアントキーを入力"
        NCMB.initialize(this, appKey, cliKey)

        binding = ActivityMainBinding.inflate(layoutInflater)
        setContentView(binding.root)

        binding.buttonlogin.setOnClickListener{
            //xmlファイルに指定したIDを使う(userIDとuserPWはタイトルのEditText部に紐づく)
            val userID = binding.userID.text.toString()
            val userPW = binding.userPW.text.toString()
            login(userID,userPW)
        }
        binding.buttonsignUp.setOnClickListener{
            //xmlファイルに指定したIDを使う(手を抜いたので上と同じIDを使いまわし)
            val newID = binding.userID.text.toString()
            val newPW = binding.userPW.text.toString()
            signUp(newID,newPW)
        }
    }
}

クイズ画面(QuizActivity.kt)の実装

  • こっちのLayoutファイルも上記同様に省略します
ソースコードを表示する
QuizActivity.kt
package com.example.a2022adventkotlin
import android.content.Intent
import androidx.appcompat.app.AppCompatActivity
import android.os.Bundle
import android.widget.Toast
import androidx.appcompat.app.AlertDialog
import com.example.a2022adventkotlin.databinding.ActivityQuizBinding
import com.nifcloud.mbaas.core.NCMB
import com.nifcloud.mbaas.core.NCMBObject
import com.nifcloud.mbaas.core.NCMBCallback
import com.nifcloud.mbaas.core.NCMBQuery

class QuizActivity : AppCompatActivity() {
    private var questionNum = 0
    private lateinit var binding: ActivityQuizBinding

    private fun count(): Int{
        val query = NCMBQuery.forObject("QUIZ1")
        val size = query.count()
        return size
    }
    private fun Question(x:Int) {
        //QUIZ1クラスにある問題をデータストアから取得する
        val query = NCMBQuery.forObject("QUIZ1")
        query.findInBackground(NCMBCallback { e, objects ->
            if (e != null) {
                //error処理(省略)
            } else {
                if (objects is List<*>) {
                    var arr: Any? = objects[x]
                    if (arr is NCMBObject) {
                        //選択肢情報を格納しておく(正誤判定でも使う)
                        val a1 = arr.getString("a1")
                        val a2 = arr.getString("a2")
                        val a3 = arr.getString("a3")
                        val a4 = arr.getString("a4")
                        //出題
                        binding.TextView2.text = arr.getString("question")
                        binding.button.text  = a1
                        binding.button2.text = a2
                        binding.button3.text = a3
                        binding.button4.text = a4
                        //選択肢を選んだ時の処理
                        val answer = arr.getString("answer")
                        binding.button.setOnClickListener {
                            if (answer == a1) {
                                AlertDialog.Builder(this)
                                    .setTitle("正解!")
                                    .setMessage( answer + "です!")
                                    .setPositiveButton("閉じる"){dialog, which ->}
                                    .show()
                            } else {
                                AlertDialog.Builder(this)
                                    .setTitle("不正解!")
                                    .setMessage("正解は" + answer +"です")
                                    .setPositiveButton("閉じる"){dialog, which ->}
                                    .show()
                            }
                        }
                        binding.button2.setOnClickListener {
                            if (answer == a2) {
                                AlertDialog.Builder(this)
                                    .setTitle("正解!")
                                    .setMessage( answer + "です!")
                                    .setPositiveButton("閉じる"){dialog, which ->}
                                    .show()
                            } else {
                                AlertDialog.Builder(this)
                                    .setTitle("不正解!")
                                    .setMessage("正解は" + answer +"です")
                                    .setPositiveButton("閉じる"){dialog, which ->}
                                    .show()
                            }
                        }
                        binding.button3.setOnClickListener {
                            if (answer == a3) {
                                AlertDialog.Builder(this)
                                    .setTitle("正解!")
                                    .setMessage( answer + "です!")
                                    .setPositiveButton("閉じる"){dialog, which ->}
                                    .show()
                            } else {
                                AlertDialog.Builder(this)
                                    .setTitle("不正解!")
                                    .setMessage("正解は" + answer +"です")
                                    .setPositiveButton("閉じる"){dialog, which ->}
                                    .show()
                            }
                        }
                        binding.button4.setOnClickListener {
                            if (answer == a4) {
                                AlertDialog.Builder(this)
                                    .setTitle("正解!")
                                    .setMessage( answer + "です!")
                                    .setPositiveButton("閉じる"){dialog, which ->}
                                    .show()
                            } else {
                                AlertDialog.Builder(this)
                                    .setTitle("不正解!")
                                    .setMessage("正解は" + answer +"です")
                                    .setPositiveButton("閉じる"){dialog, which ->}
                                    .show()
                            }
                        }
                    }
                }
            }
        })
    }
    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        val appKey = "ニフクラ mobile backendからアプリケーションキーを入力"
        val cliKey = "ニフクラ mobile backendからクライアントキーを入力"
        NCMB.initialize(this, appKey, cliKey)

        binding = ActivityQuizBinding.inflate(layoutInflater)
        setContentView(binding.root)
        //出題を切り替えるボタン
        binding.nextButton.setOnClickListener {
            binding.nextButton.text= "次の問題へ"
            //出題
            if (questionNum >= count()) {
                //問題を最初に戻す
                questionNum = 0
                Question(questionNum)
            } else {
                //次の問題へ
                Question(questionNum)
                questionNum++
            }
        }
        binding.homeButton.setOnClickListener{
            val intent = Intent(this, MainActivity::class.java)
            startActivity(intent)
        }
    }
}

完成画面


無事実装したい機能の動作が確認出来ました。タイトルの鳥はニフクラ mobile backend のイメージキャラクターのタカノくんです。
色が入るだけで画面が綺麗に見えるので、デザイン面も勉強しながら綺麗なUIも作っていきたいですね~。

おわりに

いかがでしたでしょうか。簡単なアプリにはなりましたが、Androidアプリ開発者への一歩目を踏み出せたような気がして楽しくなってきました。来年のアドカレではつよつよな記事を出せるように頑張ります。ニフクラ mobile backendもSNSアカウントで始められるので、まだ使ったことない人は是非。使ってくれたら私が喜びます。

筆者はこのアプリをベースに新しい技術を勉強しようと思っています。(年末に餅を食べながら頑張ります。)
この記事が誰かのモチベに繋がればいいですね。餅だけに。(ごめん)

参考

5
1
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
5
1

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?