Основные данные не сохраняются - PullRequest
0 голосов
/ 15 марта 2019

Я пытаюсь создать приложение, которое требует, чтобы пользователь успешно ввел пин-код, прежде чем он будет разрешен для остальной части приложения.Я немного покопался и нашел пример существующего базового приложения coredata, которое работает здесь .

. Я зашел в xcdatamodel, удалил их атрибуты и заменил на «pin», который является строкой.Вот снимок экрана xcdatamodel. xcdatamodel screenshot

Затем я изменил ViewController так, чтобы пользовательский интерфейс createData открывал alertController, который предлагает пользователю дважды ввести новый вывод, проверяет, что онито же самое, и если они есть, он создает запись coredata с этим пин.

Вот соответствующий код ViewController:

import UIKit
import CoreData

class ViewController: UIViewController {
var firstPinNumber:String = ""
var secondPinNumber:String = ""

override func viewDidLoad() {
    super.viewDidLoad()
    // Do any additional setup after loading the view, typically from a nib.

}

@IBAction func createData(_ sender: Any) {
    let enterPinAlertController = UIAlertController(title: "Enter New PIN", message: "", preferredStyle: .alert)
    enterPinAlertController.addTextField{ (textField1:UITextField)->Void in
        textField1.placeholder = "Enter PIN"
        textField1.isSecureTextEntry = true
    }
    enterPinAlertController.addTextField{   (textField2:UITextField)->Void in
        textField2.placeholder = "Re-Enter PIN"
        textField2.isSecureTextEntry = true
    }
    let okAction = UIAlertAction(title: "OK", style: .cancel)   {(action) in
        if let textFields = enterPinAlertController.textFields  {
            let theTextFields = textFields as [UITextField]
            self.firstPinNumber = theTextFields[0].text!
            self.secondPinNumber = theTextFields[1].text!
            if self.firstPinNumber != self.secondPinNumber   {
                print ("PINs dont match!")
                let pinsDontMatchAlertController = UIAlertController(title: "PINs don't match!", message: "Try again", preferredStyle: .alert)
                let okAction = UIAlertAction(title: "OK", style: .cancel)   {(action) in
                }
                pinsDontMatchAlertController.addAction(okAction)
                self.present(pinsDontMatchAlertController, animated: true, completion: nil)
            }
        }
    }

    enterPinAlertController.addAction(okAction)
    self.present(enterPinAlertController, animated: true, completion: nil)
    createPIN(pinNum: secondPinNumber)


}
func createPIN(pinNum: String){

    //As we know that container is set up in the AppDelegates so we need to refer that container.
    guard let appDelegate = UIApplication.shared.delegate as? AppDelegate else { return }

    //We need to create a context from this container
    let managedContext = appDelegate.persistentContainer.viewContext

    //Now let’s create an entity and new user records.
    let userEntity = NSEntityDescription.entity(forEntityName: "User", in: managedContext)!

    let user = NSManagedObject(entity: userEntity, insertInto: managedContext)
    user.setValue(pinNum, forKeyPath: "pin")
    print(user.value(forKey: "pin") as Any)


    //Now we have set the pin. The next step is to save it inside the Core Data

    do {
        try managedContext.save()

    } catch let error as NSError {
        print("Could not save. \(error), \(error.userInfo)")
    }
}

@IBAction func retrieveData(_ sender: Any) {
    let storedPin = retrievePIN()
    print(storedPin)
}

func retrievePIN()->String {
    var storedPin:String = ""

    //As we know that container is set up in the AppDelegates so we need to refer that container.
    guard let appDelegate = UIApplication.shared.delegate as? AppDelegate else { return "" }

    //We need to create a context from this container
    let managedContext = appDelegate.persistentContainer.viewContext

    //Prepare the request of type NSFetchRequest  for the entity
    let fetchRequest = NSFetchRequest<NSFetchRequestResult>(entityName: "User")

    fetchRequest.fetchLimit = 1
//        fetchRequest.predicate = NSPredicate(format: "username = %@", "Ankur")
//        fetchRequest.sortDescriptors = [NSSortDescriptor.init(key: "email", ascending: false)]
//
    do {
        let result = try managedContext.fetch(fetchRequest)
        for data in result as! [NSManagedObject] {
            if data.value(forKey: "pin") != nil {
                storedPin = data.value(forKey: "pin") as! String
                print(storedPin)
            } else  {
                print ("Found nil")
            }
        }

    } catch {

        print("Failed")
    }
    return storedPin
}

Используя точки останова, я убедился, что он входит в createPin(), но, похоже, она вводится в эту функцию ДО того, как она представляет enterPinAlertController для ввода нового вывода, даже если createPin () вызывается ПОСЛЕ представления enterPinAlertController.

Также, если я использую UT кнопки retrieveData, она печатаетout "Found nil"

Итак, если то, что я считаю правильным, это создание записи coredata с пустой строкой или вообще ничего?Как я могу исправить это так, чтобы он создавал запись coredata со строкой, которую пользователь вводит в качестве нового контакта, а также извлекает ее позже?

1 Ответ

1 голос
/ 15 марта 2019

Ваш вызов createPin должен быть внутри обработчиком действия для okAction. Как у вас есть, secondPinNumber будет вызываться до того, как будет показано предупреждение, поэтому оно будет пустым или nil, в зависимости от того, как вы его инициализируете.

IBAction func createData(_ sender: Any) {
    let enterPinAlertController = UIAlertController(title: "Enter New PIN", message: "", preferredStyle: .alert)
    enterPinAlertController.addTextField{ (textField1:UITextField)->Void in
        textField1.placeholder = "Enter PIN"
        textField1.isSecureTextEntry = true
    }
    enterPinAlertController.addTextField{   (textField2:UITextField)->Void in
        textField2.placeholder = "Re-Enter PIN"
        textField2.isSecureTextEntry = true
    }
    let okAction = UIAlertAction(title: "OK", style: .cancel)   {(action) in
        if let textFields = enterPinAlertController.textFields,
           let firstPinNumber = textFields[0].text,
           let secondPinNumber = textFields[1].text,
           firstPinNumber == secondPinNumber {
               createPIN(pinNum: secondPinNumber)
            } else {
               print ("PINs dont match!")
               let pinsDontMatchAlertController = UIAlertController(title: "PINs don't match!", message: "Try again", preferredStyle: .alert)
               let okAction = UIAlertAction(title: "OK", style: .cancel)  
               pinsDontMatchAlertController.addAction(okAction)
               self.present(pinsDontMatchAlertController, animated: true, completion: nil)
            } 
        }
    }

    enterPinAlertController.addAction(okAction)
    self.present(enterPinAlertController, animated: true, completion: nil)

}
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...