2
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 3 years have passed since last update.

キーボード表示でViewをずらすか判定するViewController作ってみた

Last updated at Posted at 2019-11-20

はじめに

キーボードの表示非表示処理って簡単なのですが、テキストボックスとかぶったり、viewをずらしたらテキストフィールドが上に行き過ぎて見えなくなったりして面倒だなぁと思ってました。
何とかならないかと試行錯誤したので記事書いてみました。

考え方

  • できるだけ簡単に
  • テキストフィールドが増えても何もしなくていい
  • テキストフィールドが隠れる時だけずらしたい

以上を踏まえて考えたところ、継承したUIViewControllerを作るのが手っ取り早いかな?と結論に至りました。

実装

まずはおきまりのキーボードが表示された時と隠れた時、バックグラウンドから復帰した時の通知を拾います。

class ForInputViewController: UIViewController {

    override func viewWillAppear(_ animated: Bool) {
        super.viewWillAppear(animated)
        NotificationCenter.default.addObserver(self, selector: #selector(self.keyboardWillShowNotification(notification:)), name: UIResponder.keyboardWillShowNotification, object: nil)
        NotificationCenter.default.addObserver(self, selector: #selector(self.keyboardWillHideNotification(notification:)), name: UIResponder.keyboardWillHideNotification, object: nil)
        NotificationCenter.default.addObserver(self, selector: #selector(self.viewWillEnterForeground(notification:)), name: UIApplication.willEnterForegroundNotification, object: nil)
    }

    override func viewWillDisappear(_ animated: Bool) {
        super.viewWillDisappear(animated)
        NotificationCenter.default.removeObserver(self, name: UIResponder.keyboardWillShowNotification, object: nil)
        NotificationCenter.default.removeObserver(self, name: UIResponder.keyboardWillHideNotification, object: nil)
        NotificationCenter.default.removeObserver(self, name: UIApplication.willEnterForegroundNotification, object: nil)
    }

    @objc override func keyboardWillShowNotification(notification: NSNotification) {
        guard let userInfo = notification.userInfo,
            let keyboard = userInfo[UIResponder.keyboardFrameEndUserInfoKey] as? NSValue else {
                return
        }
        let keyboardScreenEndFrame = keyboard.cgRectValue
        let myBoundSize: CGSize = UIScreen.main.bounds.size
        // TODO:ここでテキストフィールドが隠れるかどうか判定したい
    }

    @objc override func keyboardWillHideNotification(notification: NSNotification) {

    }

    @objc override func viewWillEnterForeground(notification: Notification) {

    }

}

問題はキーボードが表示された時のテキストフィールドの情報がないので、キーボードに隠れるかどうかはわかりません。
そこで、
textFieldShouldBeginEditingイベントで対象のテキストフィールドのインスタンスを取得して変数に保持しておきます。

    private var txtActiveField = UITextField()
    func textFieldShouldBeginEditing(_ textField: UITextField) -> Bool {
        txtActiveField = textField
        return true
    }

こうすることで、キーボードが表示された時のテキストボックスの座標・高さが取得できるようになります。
そして、キーボードが表示された時に実行される関数(ここではkeyboardWillShowNotification)の中でキーボードでテキストフィールドが隠れるかどうかを判定してあげます。
ポイントはconvertで親のviewからみた座標に変換してあげることですね。

@objc override func keyboardWillShowNotification(notification: NSNotification) {
        guard let userInfo = notification.userInfo,
            let keyboard = userInfo[UIResponder.keyboardFrameEndUserInfoKey] as? NSValue else {
                return
        }
        let keyboardScreenEndFrame = keyboard.cgRectValue
        let myBoundSize: CGSize = UIScreen.main.bounds.size

        // textFieldの座標を全体座標に変換
        let textframeParent = txtActiveField.convert(txtActiveField.frame, to: self.view)
        let txtLimit = textframeParent.origin.y + textframeParent.height + 8.0
        let kbdLimit = myBoundSize.height - keyboardScreenEndFrame.size.height

        log.debug("テキストフィールドの下辺:(\(txtLimit))")
        log.debug("キーボードの上辺:(\(kbdLimit))")

        // テキストフィールドがキーボードに隠れる時のみスライドさせる
        if txtLimit >= kbdLimit {
            UIView.animate(withDuration: 0, animations: { () in
                let transform = CGAffineTransform(translationX: 0, y: -(keyboardScreenEndFrame.size.height))
                self.view.transform = transform
            })
        }
    }

そしてキーボードが隠れる時やバックグラウンドからの復帰処理も入れます。

    @objc override func keyboardWillHideNotification(notification: NSNotification) {
        UIView.animate(withDuration: 0, animations: { () in
            self.view.transform = CGAffineTransform.identity
        })
    }

    @objc override func viewWillEnterForeground(notification: Notification) {
        UIView.animate(withDuration: 0, animations: { () in
            self.view.transform = CGAffineTransform.identity
        })
    }

    func textFieldShouldReturn(_ textField: UITextField) -> Bool {
        textField.resignFirstResponder()
        return true
    }

後はこのできたViewControllerを継承すればいいのですが、対象のUITextFieldのdelegateを設定してあげる必要があります。普通なら使うcontroller側で以下のようにすると思いますが、

    @IBOutlet private weak var textField: UITextField! {
        didSet {
            textField.delegate = self
        }
    }

これだとテキストフィールドが増えると面倒です。

そこで、自分が保持しているtextField全てにdelegateを自動で設定するようにviewDidLoadに組み込みます。

    override func viewDidLoad() {
        super.viewDidLoad()
        // 全てのテキストフィールドのdelegate設定
        let textFileds = getAllTextFields(fromView: self.view)
        for textFiled in textFileds {
            textFiled.delegate = self
        }
    }

    private func getAllTextFields(fromView view: UIView) -> [UITextField] {
        return view.subviews.compactMap { (view) -> [UITextField]? in
            return (view is UITextField) ? [(view as! UITextField)] : getAllTextFields(fromView: view)
        }.flatMap({$0})
    }

これでTextFieldがいくら増えても自動で隠れるかどうか判定してキーボードを表示してくれます。

完成ソース

class ForInputViewController: UIViewController, UITextFieldDelegate {

    private var txtActiveField = UITextField()

    // MARK: - Lifecycle

    override func viewDidLoad() {
        super.viewDidLoad()
        // 全てのテキストフィールドのdelegate設定
        let textFileds = getAllTextFields(fromView: self.view)
        for textFiled in textFileds {
            textFiled.delegate = self
        }
    }

    override func viewWillAppear(_ animated: Bool) {
        super.viewWillAppear(animated)
        NotificationCenter.default.addObserver(self, selector: #selector(self.keyboardWillShowNotification(notification:)), name: UIResponder.keyboardWillShowNotification, object: nil)
        NotificationCenter.default.addObserver(self, selector: #selector(self.keyboardWillHideNotification(notification:)), name: UIResponder.keyboardWillHideNotification, object: nil)
        NotificationCenter.default.addObserver(self, selector: #selector(self.viewWillEnterForeground(notification:)), name: UIApplication.willEnterForegroundNotification, object: nil)
    }

    override func viewWillDisappear(_ animated: Bool) {
        super.viewWillDisappear(animated)
        NotificationCenter.default.removeObserver(self, name: UIResponder.keyboardWillShowNotification, object: nil)
        NotificationCenter.default.removeObserver(self, name: UIResponder.keyboardWillHideNotification, object: nil)
        NotificationCenter.default.removeObserver(self, name: UIApplication.willEnterForegroundNotification, object: nil)
    }

    override func touchesBegan(_ touches: Set<UITouch>, with event: UIEvent?) {
        self.view.endEditing(true)
    }

    @objc func keyboardWillShowNotification(notification: NSNotification) {
        guard let userInfo = notification.userInfo,
            let keyboard = userInfo[UIResponder.keyboardFrameEndUserInfoKey] as? NSValue else {
                return
        }
        let keyboardScreenEndFrame = keyboard.cgRectValue
        let myBoundSize: CGSize = UIScreen.main.bounds.size

        // textFieldの座標を全体座標に変換
        let textframeParent = txtActiveField.convert(txtActiveField.frame, to: self.view)
        let txtLimit = textframeParent.origin.y + textframeParent.height + 8.0
        let kbdLimit = myBoundSize.height - keyboardScreenEndFrame.size.height

        log.debug("テキストフィールドの下辺:(\(txtLimit))")
        log.debug("キーボードの上辺:(\(kbdLimit))")

        // テキストフィールドがキーボードに隠れる時のみスライドさせる
        if txtLimit >= kbdLimit {
            UIView.animate(withDuration: 0, animations: { () in
                let transform = CGAffineTransform(translationX: 0, y: -(keyboardScreenEndFrame.size.height))
                self.view.transform = transform
            })
        }
    }

    @objc func keyboardWillHideNotification(notification: NSNotification) {
        UIView.animate(withDuration: 0, animations: { () in
            self.view.transform = CGAffineTransform.identity
        })
    }

    @objc func viewWillEnterForeground(notification: Notification) {
        UIView.animate(withDuration: 0, animations: { () in
            self.view.transform = CGAffineTransform.identity
        })
    }

    override func touchesBegan(_ touches: Set<UITouch>, with event: UIEvent?) {
        self.view.endEditing(true)
    }

    // MARK: - public

    func textFieldShouldBeginEditing(_ textField: UITextField) -> Bool {
        txtActiveField = textField
        return true
    }

    func textFieldShouldReturn(_ textField: UITextField) -> Bool {
        textField.resignFirstResponder()
        return true
    }

    // MARK: - private

    private func getAllTextFields(fromView view: UIView) -> [UITextField] {
        return view.subviews.compactMap { (view) -> [UITextField]? in
            return (view is UITextField) ? [(view as! UITextField)] : getAllTextFields(fromView: view)
        }.flatMap({$0})
    }

}

使い方は継承するだけなのでいい感じに手抜きできると思いますwww
効率よい仕事につながりますね!

参考

大変勉強になりました。ありがとうございましたm(_ _)m
- [Swift] UITextFieldがキーボードに隠れないようにするやり方

更新(11/25)

上記のようにviewDidLoadでdelegateをセットするとテーブルビューなどコードでセットするテキストフィールドには反応しません。そこでdelegateのセットのタイミングをviewDidAppearに変更することで対応ができました。

    private var initialized: Bool = false

    override func viewDidLoad() {
        super.viewDidLoad()
        initialized = false
    }

    override func viewDidAppear(_ animated: Bool) {
        super.viewDidAppear(animated)
        guard !initialized else { return }

        // 全てのテキストフィールドのdelegate設定
        setDelegate()
    }

    private func setDelegate() {
        let textFileds = getAllTextFields(fromView: self.view)
        for textFiled in textFileds {
            textFiled.delegate = self
        }
        initialized = true
    }

また、UIScrollViewを使用している場合はtouchesBeganイベントを伝搬させてあげる必要があります。

extension UIScrollView {
    override open func touchesBegan(_ touches: Set<UITouch>, with event: UIEvent?) {
        self.next?.touchesBegan(touches, with: event)
    }
}

UITextViewも同じ処理で実現できます!

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