2
0

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.

bitFlyerAdvent Calendar 2022

Day 10

Responderチェーンを辿る

Last updated at Posted at 2022-12-09

bitFlyer システム開発本部 フロントエンド開発部 部長の林です。
マネジメント職ですが、気持ちは今もiOSエンジニアです。
この記事は bitFlyer Advent Calendar 2022 の10日目として書かれました。

優秀なiOSアプリエンジニアは画面を平面としてではなく階層として扱う、という持論があります。
ViewやViewControllerの階層が正しく設計され、あるべき層にあるべき情報が保持されていると、複雑な状態を伴うアニメーションにも耐えられる強度を持った美しいアプリが完成します。

さて、階層構造を基本とするUIKitですが、他にもう一つ興味深いデータ構造があります。

タイトルにもなったResponderチェーンです。

UIResponderはイベントハンドリングを担当するクラスで、UIViewやUIViewController, UIWindow, UIApplication, UIApplicationDelagateなどUIKitにおける主要なクラスはUIResponderを継承しています。

UIResponderはnextというプロパティで繋がってResponderチェーンを形成します。

class UIResponder : NSObject {
    var next: UIResponder?
}

末端から始まるReponderチェーンは、最終的にAppDelagateまで辿り着きます。

UIViewがsubviewsというプロパティを持ち、UIWindowを起点に末端へ広がる階層を形成することと対照的ですね。UIViewの階層を図にするとこうなります。

さて、Responderチェーンを使って、スクロールに合わせてインタラクティブにキーボードを閉じるチャット画面を実装してみます。

最終的にはこの画面が出来上がります。

UIResponderにはキーボードの上にUIを表示するためのinputAccessoryViewというプロパティが存在します。

var inputAccessoryView: UIView?

UIViewではなく、UIResponderのプロパティというのがポイントです。

例えば以下のようにすると、テキストを入力する際にキーボードの上にtoolBarが表示されます。

textView.inputAccessoryView = toolBar

さて、チャット画面を作りたいのでした。TextViewが含まれるツールバーをinputAccessoryViewに設定すれば良さそうです。

しかし、ツールバー自身がUITextViewを含んでいるので以下のコードは動きません。

// 動作しない例
messengerInputAccessoryView.textView.inputAccessoryView = messengerInputAccessoryView

子ViewのinputAccessoryViewで親のViewを参照していますし、UITextViewがFirst responderになっていないとツールバーが出てきません。

困りました。

基本に立ち返って、一次情報に当たりましょう。

inputAccessoryViewは以下のようにコメントで説明されています。

Called and presented when object becomes first responder. Goes up the responder chain.

First responderになった時、UIKitはResponderチェーンを辿ってinputAccessoryViewを探します。つまり自身がinputAccessoryViewを返さなくても、ResponderチェーンのどこかでinputAccessoryViewが返されればこれを使うということです。

さらに、inputAccessoryViewの表示はキーボードの有無に依存しません。キーボードを表示しないUIResponderにinputAccessoryViewを設定し、first responderにすることでinputAccessoryViewだけを表示することができます。

このような理由から、ViewControllerのinputAccessoryViewでTextViewを含むツールバーを返すことで、チャットの入力ボックスが実装できます。(ViewControllerもまたUIReponderなのです)

キーボード非表示の初期状態ではViewControllerがfirst responderとなり、inputAccessoryViewとしてツールバーを返します。
TextViewにフォーカスが当たるとfirst responderが移動しますがResponderチェーンにはViewControllerが含まれており、引き続きinputAccessoryViewを返しています。

Responderチェーンをうまく使うことで、複雑に見える挙動もシンプルに実装することができます。

class ViewController: UIViewController {

    @IBOutlet var inputAccessoryTextView: MessengerInputAccessoryView!

    override var inputAccessoryView: UIView? {
        return inputAccessoryTextView
    }

    override var canBecomeFirstResponder: Bool { true }

    override func viewDidLoad() {
        super.viewDidLoad()
        becomeFirstResponder()
    }
}

また、keyboardDismissModeを設定することでaccessory viewを含むキーボードをインタラクティブに閉じることができます。

scrollView.keyboardDismissMode = .interactive

ツールバー自体も含め、全体で30行ちょっとの実装で期待通りに動くチャット画面の入力が実装できました。

フレームワークの使い方を覚えるだけでなく、フレームワークがどう動いているのかを知ることでしか辿り着けない品質というものがあると信じています。bitFlyerでは本物のiOSエンジニアを募集しています!

ぜひご応募ください。

2
0
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
2
0

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?