iOS アプリ開発 SkeletonView ロード待ちを感じさせない工夫


SkeletonView

「フットサル」というフットサルを一緒に蹴るメンバーを募集したり簡単に参加できるアプリをリリースしました。

Swift + Firebase(Cloud Firestore) の構成です。

フットサル 募集
フットサル
開発元: Players1st inc.
無料

メンバーを募集するイベントを誰でも作れるのですが、イベント一覧を Firebase から読み込むのに若干時間がかかり、空白のページで待ちが発生するような操作感になっていました。

ユーザーに待ちを感じさせないように SkeletonView を導入することにしました。

https://github.com/Juanpe/SkeletonView

Facebookアプリにも使われていて、データが入る部分がグレーの背景色でまず描画されあとから読み込まれたデータが表示されると言う見え方をするので、ユーザーが真っ白な画面を見て待つ事がなくなり、感覚的に待たされている感じが減るというものです。

使い方

Github の SkeletonView のページにありますが、対象の view に対して showAnimatedSkeleton() でスケルトン表示、hideSkeleton() で非表示にあります。

サンプル

私のアプリの場合、イベント一覧の UITableViewController のページで使っています。

import UIKit
import CoreData
import Firebase
import SkeletonView

class EventsTableViewController: UITableViewController {

  (snip)

  func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
      // セルを取得する
      let cell = tableView.dequeueReusableCell(withIdentifier: "repeatCell", for: indexPath) as! CustomEventCell

      // SkeletonView show
      cell.dateLabel.showAnimatedSkeleton()
      cell.timeLabel.showAnimatedSkeleton()
      cell.courtLabel.showAnimatedSkeleton()
      cell.supportLabel.showAnimatedSkeleton()
      cell.numberLabel.showAnimatedSkeleton()
      cell.afterSupportLabel.showAnimatedSkeleton()

      let event = self.events[indexPath.row]
      let documentID = documentIDs[indexPath.row]
      // 申込人数 event.document.entries 数
      db.collection("entries").whereField("eventID", isEqualTo: documentID).getDocuments() { (querySnapshot, err) in
          if let err = err {
              print("Error getting documents: \(err)")
          } else {
              let d = DateFormatter()
              d.dateFormat = "M/d(E)"
              d.locale = Locale(identifier: "ja_JP")
              cell.dateLabel.text = d.string(from: event.start.dateValue())
              let t = DateFormatter()
              t.dateStyle = .none
              t.timeStyle = .short
              t.locale = Locale(identifier: "ja_JP")

              cell.timeLabel.text = "\(t.string(from: event.start.dateValue())) - \(t.string(from: event.end.dateValue()))"
              cell.courtLabel.text = event.courtName

              self.remainingCount = event.recruitment - querySnapshot!.documents.count
              if event.recruitment > querySnapshot!.documents.count {
                  cell.supportLabel.text = "あと"
                  cell.supportLabel.textColor = UIColor.black
                  cell.supportLabel.backgroundColor = UIColor.white
                  cell.numberLabel.text = String(self.remainingCount)
                  cell.afterSupportLabel.text = "名"
              } else {
                  cell.supportLabel.text = "開催決定"
                  cell.supportLabel.textColor = UIColor.black
                  cell.supportLabel.backgroundColor = UIColor.gray
                  cell.numberLabel.text = String(querySnapshot!.documents.count)
                  cell.afterSupportLabel.text = "名 申込"
              }

              // SkeletonView hide
              cell.dateLabel.hideSkeleton()
              cell.timeLabel.hideSkeleton()
              cell.courtLabel.hideSkeleton()
              cell.supportLabel.hideSkeleton()
              cell.numberLabel.hideSkeleton()
              cell.afterSupportLabel.hideSkeleton()
          }
      }
      return cell
  }

  (snip)

end

各セルを表示する時に showAnimatedSkeleton() して、Firebase からデータを受け取って view にデータを設定した後に hideSkeleton() しています。

このように簡単に使うことができます。

sample

ローディング中のアニメーションなどのカスタマイズできます。

https://github.com/Juanpe/SkeletonView#-custom-animations

かんたんにユーザー体験のコントロールができるツールですね。 お役に立てれば幸いです😊


自社サービスのためだけに開発をしているので、iOSアプリ開発の受託はやっていませんが、新規事業・サービス立ち上げ・システム開発のコンサルティング・アドバイザーを行っています。 今までの実績は こちら をご覧ください。

お仕事の相談いつでもウェルカムです。ご興味ある方は こちらからでもSNSのDMからでもお気軽にご相談ください。

Related Posts

長沼公園 - お気に入りの場所

Flutter in_app_purchase で定期購入を実現する方法 2023年版

ペップのビルドアップ UEFA Champions League Final

テクニカルアドバイザー仕事が終わったのでまとめ

Amazon で Kindle とペーパーバックを作るやり方

[思考]老いへの許容

「ひとりスタートアップ」が本の形になりました

2022年を振り返る

ハードウエアのWeb化

本を書きました「ひとりスタートアップ」