简体   繁体   English

有没有办法强制Swift中的TextField中的文本是非可选的?

[英]Is there a way to force the text in a TextField in Swift to be non-optional?

I am making an app that takes inputs from textFields in an alertController, then stored into CoreData and eventually be displayed in a tableView. 我正在制作一个应用程序,它从一个alertController中的textFields获取输入,然后存储到CoreData中,最终显示在tableView中。

The plan for now is to create a string to create a CSV by combining the textField.text together, something like 现在的计划是通过将textField.text组合在一起来创建一个用于创建CSV的字符串

let string CSV = textField.text + " ," + textField2.text + " ," + ...

However, when I tried to do this, they say that I cannot force unwrap an optional value. 但是,当我尝试这样做时,他们说我无法强制打开可选值。

I tried to put a "!" 我试着把“!” in textField.text, etc but they are all still seen as optional values. 在textField.text等中,但它们仍被视为可选值。 The textFields, when employed in the app must always be filled, but I could not find a way to provide an error that forces the user of the app to fill in a value. textFields在应用程序中使用时必须始终填充,但我找不到提供强制应用程序用户填写值的错误的方法。

Here is my ViewController: 这是我的ViewController:

import UIKit
import CoreData

class ViewController: UITableViewController {

    var alarmItems: [NSManagedObject] = []
    let cellId = "cellId"

    override func viewDidLoad() {
        super.viewDidLoad()
        view.backgroundColor = .white
        navigationController?.navigationBar.barTintColor = UIColor(red: 21/255, green: 101/255, blue: 192/255, alpha: 1)
        navigationController?.navigationBar.tintColor = .white
        navigationItem.rightBarButtonItem = UIBarButtonItem(title: "Add Alarm", style: .plain, target: self, action: #selector(addAlarmItem))
        tableView.register(UITableViewCell.self, forCellReuseIdentifier: cellId)
    }

    override func viewWillAppear(_ animated: Bool) {
        super.viewWillAppear(animated)
        guard let appDelegate = UIApplication.shared.delegate as? AppDelegate else { return }
        let managedContext = appDelegate.persistentContainer.viewContext
        let fetchRequest = NSFetchRequest<NSManagedObject>(entityName: "AlarmItems")
        do {
            alarmItems = try managedContext.fetch(fetchRequest)
        } catch let err as NSError {
            print("Failed to fetch items", err)
        }
    }

    //now to figure out how to add multiple text fields

    @objc func addAlarmItem(_ sender: AnyObject) {
        print("this works")
        let alertController = UIAlertController(title: "Add New Item", message: "Please fill in the blanks", preferredStyle: .alert)
        let saveAction = UIAlertAction(title: "Save", style: .default) { [unowned self] action in
            guard let textField = alertController.textFields!.first, let itemToAdd = textField.text else { return }

            //guard let textField2 = alertController.textFields?.first, let itemToAdd2 = textField2.text else { return } //tried this to get the input of a second textField

            /*

            one thing I tried:
            let textField1 = alertController.textFields[0]
            let textField1String = textField1.text!

            let textField2 = alertController.textFields![1]
            let textField2String = textField2.text
            let itemToAdd = textField1String + textField2String
            */



            //var textField1String = textField1.text
            //let textField2 = alertController.textFields![1]
            //var textField2String = textField2.text


            self.save(itemToAdd)
            //self.save(itemToAdd2)
            self.tableView.reloadData()
        }
        let cancelAction = UIAlertAction(title: "Cancel", style: .destructive, handler: nil)
        //alertController.addTextField(configurationHandler: nil)
        alertController.addTextField { (textField) in
            textField.placeholder = "First"
        }
        alertController.addTextField { (textField) in
            textField.placeholder = "Second"
        }

        //let combinedString = UITextField[0] + " ," + UITextField[1]
        alertController.addAction(saveAction)
        alertController.addAction(cancelAction)
        present(alertController, animated: true, completion: nil)
    }

    func save(_ itemName: String) {
        guard let appDelegate = UIApplication.shared.delegate as? AppDelegate else { return }
        let managedContext = appDelegate.persistentContainer.viewContext
        let entity = NSEntityDescription.entity(forEntityName: "AlarmItems", in: managedContext)!
        let item = NSManagedObject(entity: entity, insertInto: managedContext)
        item.setValue(itemName, forKey: "alarmAttributes")

        do {
            try managedContext.save()
            alarmItems.append(item)

        } catch let err as NSError {
            print("Failed to save an item", err)
        }
    }

    override func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int{
        return alarmItems.count
    }

    override func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
        let cell = tableView.dequeueReusableCell(withIdentifier: cellId, for: indexPath)
        let alarmItem = alarmItems[indexPath.row]
        cell.textLabel?.text = alarmItem.value(forKeyPath: "alarmAttributes") as? String
        return cell
    }
}

You can use compactMap to easily convert an array of optionals into an array of non-optionals. 您可以使用compactMap轻松地将选项数组转换为非选项数组。

let myStrings: [String] = alert.textfields!.compactMap { $0.text }
let myText = myStrings.joined(separator: ", ")

To learn more about compactMap, head over to https://developer.apple.com/documentation/swift/sequence/2950916-compactmap 要了解有关compactMap的更多信息,请访问https://developer.apple.com/documentation/swift/sequence/2950916-compactmap

One of the ways you can do the error validation is that when save is pressed, do empty checks on the textfields values in a separate function and show an error alert box. 您可以执行错误验证的方法之一是按下保存时,在单独的函数中对textfields值进行空检查并显示错误警告框。

I don't think you can show some kind of error inside your initial textfield alert controller. 我不认为您可以在初始文本字段警报控制器中显示某种错误。 You have to validated the text separately and then show a separate alert error alert box. 您必须单独验证文本,然后显示单独的警报错误警告框。

You can use String extension to unwrap text this way: 您可以使用String扩展以这种方式展开文本:

extension Optional where Wrapped == String{
func safe() -> String{
    if self == nil{
        return ""
    }else{
        return self.unsafelyUnwrapped
    }
}

Hope this helps. 希望这可以帮助。

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM