質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

ただいまの
回答率

89.06%

UIAlertController上のUITextFieldの未入力を防ぐ方法

解決済

回答 1

投稿

  • 評価
  • クリップ 1
  • VIEW 3,551

TsksHsgw

score 21

前提・実現したいこと

質問をご覧いただきまして、ありがとうございます。

現在、UIAlertController上のUITextFieldでinputしてもらうという機能を実装したいと考えております。

そして、このTextFieldが未入力になってしまうのを防ぎたいと考えているのですが、
未入力を防ぐ方法がわからなくて困っております。

これまでに、下記のようなサイトを拝見し、
方法としては、
「テキストフィールドに入力が確認できるまでAlertAction.enabledをfalseにしておく」
という方法が良いのかなと思ったのですが、うまく実装できません。
Check on UIAlertController TextField for enabling the button - StackOverflow
Enable UIAlertAction of UIAlertController only after input - StackOverflow
Objective-c UITextFieldの未入力チェックの罠。

サイトなどはObjective-Cについて書かれたものも参照したのですが、
Swiftの考え方でご回答いただければ幸いです。

よろしくお願い致します。

下記にソースコードを記載致します。

ソースコード

ViewController.swift

@IBAction func addMemoButtonTapped(sender: AnyObject) {
       self.navigationController?.setNavigationBarHidden(true, animated: true)

        let alertController = UIAlertController(title: "新しくアイテムを追加", message: "項目を入力して下さい。", preferredStyle: .Alert)

        let cancelAction = UIAlertAction(title: "Cancel", style: .Default){
            action in print("Cancelが押された!")
            self.navigationController?.setNavigationBarHidden(false, animated: true)
        }

        let otherAction = UIAlertAction(title: "Done", style: .Default){
            action in print("Doneが押された!")

            self.string1 = self.textField1!.text!
            self.string2 = self.textField2!.text!
            self.navigationController?.setNavigationBarHidden(false, animated: true)
        }

        alertController.addAction(cancelAction)
        alertController.addAction(otherAction)

        //textFieldの追加
        alertController.addTextFieldWithConfigurationHandler({(textField: UITextField!) -> Void in

            self.textField1 = textField
            self.textField1!.tag = 1
            self.textField1!.placeholder = "タイトルを記入"

        })


        alertController.addTextFieldWithConfigurationHandler({(textField: UITextField!) -> Void in

            self.textField2 = textField
            self.textField2!.tag = 2
            self.textField2!.placeholder = "期限を設定"
            self.textField2!.inputView = self.datePickerView
            self.textField2!.addTarget(self, action: #selector(ToDoVC.textFieldEditing(_:)), forControlEvents: UIControlEvents.EditingDidBegin)

        })


        presentViewController(alertController, animated: true, completion: nil) 
    }
  • 気になる質問をクリップする

    クリップした質問は、後からいつでもマイページで確認できます。

    またクリップした質問に回答があった際、通知やメールを受け取ることができます。

    クリップを取り消します

  • 良い質問の評価を上げる

    以下のような質問は評価を上げましょう

    • 質問内容が明確
    • 自分も答えを知りたい
    • 質問者以外のユーザにも役立つ

    評価が高い質問は、TOPページの「注目」タブのフィードに表示されやすくなります。

    質問の評価を上げたことを取り消します

  • 評価を下げられる数の上限に達しました

    評価を下げることができません

    • 1日5回まで評価を下げられます
    • 1日に1ユーザに対して2回まで評価を下げられます

    質問の評価を下げる

    teratailでは下記のような質問を「具体的に困っていることがない質問」、「サイトポリシーに違反する質問」と定義し、推奨していません。

    • プログラミングに関係のない質問
    • やってほしいことだけを記載した丸投げの質問
    • 問題・課題が含まれていない質問
    • 意図的に内容が抹消された質問
    • 過去に投稿した質問と同じ内容の質問
    • 広告と受け取られるような投稿

    評価が下がると、TOPページの「アクティブ」「注目」タブのフィードに表示されにくくなります。

    質問の評価を下げたことを取り消します

    この機能は開放されていません

    評価を下げる条件を満たしてません

    評価を下げる理由を選択してください

    詳細な説明はこちら

    上記に当てはまらず、質問内容が明確になっていない質問には「情報の追加・修正依頼」機能からコメントをしてください。

    質問の評価を下げる機能の利用条件

    この機能を利用するためには、以下の事項を行う必要があります。

回答 1

checkベストアンサー

0

以下に簡単な例を作成しました、UITextFieldを2つ使用して、両方に入力がない場合OKボタンが押せなくなっています。
試してみてください。

import UIKit

class ViewController: UIViewController, UITextFieldDelegate {

    var okAction: UIAlertAction!
    var textFieldText1 = ""
    var textFieldText2 = ""

    override func viewDidLoad() {
        super.viewDidLoad()

    }

    func textField(textField: UITextField, shouldChangeCharactersInRange range: NSRange, replacementString string: String) -> Bool {
        let delayTime = dispatch_time(DISPATCH_TIME_NOW, Int64(100 * Double(NSEC_PER_MSEC)))
        dispatch_after(delayTime, dispatch_get_main_queue()) {
            self.searchBySearchBarText(textField)
        }
        return true
    }


    func searchBySearchBarText(textF: UITextField) {
        switch textF.tag {
        case 1: textFieldText1 = textF.text!
        case 2: textFieldText2 = textF.text!
        default: break
        }
        okAction.enabled =  textFieldText1.characters.count > 0 && textFieldText2.characters.count > 0 ? true : false

    }


    @IBAction func showAlert() {

        let alert = UIAlertController(title:"action",
                                      message: "alertView",
                                      preferredStyle: UIAlertControllerStyle.Alert)

        let cancelAction = UIAlertAction(title: "Cancel",
                                         style: UIAlertActionStyle.Cancel,
                                         handler:{
                                            (action:UIAlertAction!) -> Void in
                                            print("Cancel")
        })
        okAction = UIAlertAction(title: "OK",
                                          style: UIAlertActionStyle.Default,
                                          handler:{
                                            (action:UIAlertAction!) -> Void in
                                            print("OK")
        })
        
        okAction.enabled = false
        alert.addAction(cancelAction)
        alert.addAction(okAction)
        
        
        alert.addTextFieldWithConfigurationHandler({(text:UITextField!) -> Void in
            text.delegate = self
            text.tag = 1
            text.placeholder = "first textField"
        })
        
        alert.addTextFieldWithConfigurationHandler({(text:UITextField!) -> Void in
            text.delegate = self
            text.tag = 2
            text.placeholder = "second textField"
        })
        presentViewController(alert, animated: true, completion: nil)
    }
}

     

 回答追記  

     

import UIKit

class ViewController: UIViewController, UITextFieldDelegate {
    
    var okAction: UIAlertAction!
    var textFieldText1 = ""
    var textFieldText2 = ""
    var pikerTextField: UITextField!
    let datePicker = UIDatePicker()

    
    override func viewDidLoad() {
        super.viewDidLoad()
        
        datePicker.datePickerMode = .Date
        datePicker.addTarget(self, action: #selector(self.textFieldEditing(_:)), forControlEvents: UIControlEvents.ValueChanged)
        
    }
    
    func textField(textField: UITextField, shouldChangeCharactersInRange range: NSRange, replacementString string: String) -> Bool {
        let delayTime = dispatch_time(DISPATCH_TIME_NOW, Int64(100 * Double(NSEC_PER_MSEC)))
        dispatch_after(delayTime, dispatch_get_main_queue()) {
            self.searchBySearchBarText(textField)
        }
        return true
    }
    
    
    func searchBySearchBarText(textF: UITextField) {
        switch textF.tag {
        case 1: textFieldText1 = textF.text!
        default: break
        }
        okAction.enabled =  textFieldText1.characters.count > 0 && textFieldText2.characters.count > 0 ? true : false
    }
    
    
    @IBAction func showAlert() {
        
        let alert = UIAlertController(title:"action",
                                      message: "alertView",
                                      preferredStyle: UIAlertControllerStyle.Alert)
        
        let cancelAction = UIAlertAction(title: "Cancel",
                                         style: UIAlertActionStyle.Cancel,
                                         handler:{
                                            (action:UIAlertAction!) -> Void in
                                            print("Cancel")
        })
        okAction = UIAlertAction(title: "OK",
                                          style: UIAlertActionStyle.Default,
                                          handler:{
                                            (action:UIAlertAction!) -> Void in
                                            print("OK")
        })
        
        okAction.enabled = false
        alert.addAction(cancelAction)
        alert.addAction(okAction)
        
        
        alert.addTextFieldWithConfigurationHandler({(text:UITextField!) -> Void in
            text.delegate = self
            text.tag = 1
            text.placeholder = "first textField"
        })
        
        alert.addTextFieldWithConfigurationHandler({(text:UITextField!) -> Void in
            text.delegate = self
            text.tag = 2
            text.placeholder = "second textField"
            self.pikerTextField = text
            text.inputView = self.datePicker
        })
        presentViewController(alert, animated: true, completion: nil)
    }
    
    
    func textFieldEditing(datePicker: UIDatePicker) {
        let formatter = NSDateFormatter()
        formatter.dateFormat = "yyyy/MM/dd HH:mm"
        let dateStr = formatter.stringFromDate(datePicker.date)
        pikerTextField.text = dateStr
        textFieldText2 = dateStr
        searchBySearchBarText(UITextField())
    }
}

投稿

編集

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

  • 2016/08/23 13:23 編集

    _Kentarou様

    ご回答いただき、誠にありがとうございます。
    ご教授いただいた内容でしっかり動作することは確認できたのですが、

    2つ目のinputViewにDatePickerを使用しているせいか、
    DatePickerを使うとokAction.enable = false が入力後も true になりませんでした。

    お手数をおかけして、大変申し訳ないのですが、
    もし可能であれば、この原因も併せてご教授いただければ幸いです。

    よろしくお願い致します。

    キャンセル

  • 2016/08/23 13:37

    そこを見逃していました、すみません。
    後で試して追記いたします。

    キャンセル

  • 2016/08/23 20:38

    回答にinputViewにdatePickerを使用した例を追記しました。
    確認してみてください。

    キャンセル

  • 2016/08/23 22:25

    _Kentarou様

    重ね重ね申し訳ございません。

    ご回答いただいたコードでしっかり動作を確認することができました。
    ご丁寧にありがとうございました。

    キャンセル

15分調べてもわからないことは、teratailで質問しよう!

  • ただいまの回答率 89.06%
  • 質問をまとめることで、思考を整理して素早く解決
  • テンプレート機能で、簡単に質問をまとめられる

関連した質問

同じタグがついた質問を見る