Search code examples
swiftmapkitmkmapviewuserdefaultsmapkitannotation

How do I update UserDefaults after deleting an MKAnnotation?


In my project, when the user presses on the screen, a pin appears on the mapView, and the pin is saved to UserDefaults. In the bottom function, when the user selects a pin that is already on the mapView, it is deleted. However, I am unsure of how to make sure that this pin stays deleted through UserDefaults... what would I use for this last line of code?

@IBAction func addPin(_ sender: UILongPressGestureRecognizer) {
    guard sender.state == .ended else { return }


    let location = sender.location(in: self.mapView)
    let locCoord = self.mapView.convert(location, toCoordinateFrom: self.mapView)

    let annotation = MKPointAnnotation()

    annotation.coordinate = locCoord
    annotation.title = titleTextField.text

    self.mapView.addAnnotation(annotation)

    //Create a dictionary from the annotation
    let newAnnotationDict = [
        "lat": locCoord.latitude,
        "lng": locCoord.longitude,
        "title": annotation.title
        ] as [String : Any]

    //Pull the stored annotations data (if local)
    var annotationsArray: [[String:Any]]!
    var annotationsData = UserDefaults.standard.data(forKey: "StoredAnnotations")

    //If the data is nil, then set the new annotation as the only element in the array
    if annotationsData == nil {
        annotationsArray = [newAnnotationDict]
    } else {
        //If it isn't nil, then convert the data into an array of dicts
        do {
            //Convert this data into an array of dicts
            annotationsArray = try JSONSerialization.jsonObject(with: annotationsData!, options: []) as! [[String:Any]]
            annotationsArray.append(newAnnotationDict)
        } catch {
            print(error.localizedDescription)
        }

    }

    do {

        //Use JSONSerialization to convert the annotationsArray into Data
        let jsonData = try JSONSerialization.data(withJSONObject: annotationsArray, options: .prettyPrinted)

        //Store this data in UserDefaults
        UserDefaults.standard.set(jsonData, forKey: "StoredAnnotations")
    } catch {
        print(error.localizedDescription)
    }

    print("This will become the annotation title: \(titleTextField.text).")
    print(annotation.coordinate.latitude, annotation.coordinate.longitude)

}


func mapView(_ mapView: MKMapView, didSelect view: MKAnnotationView) {
    var selectedAnnotation = view.annotation


    print("Selected Annotation: \((selectedAnnotation?.coordinate.latitude, selectedAnnotation?.coordinate.longitude))")

    self.mapView.removeAnnotation(selectedAnnotation!)

// What do I use for the following line?
    UserDefaults.standard.set(, forKey: "StoredAnnotations")

}

Solution

  • fastest solution would be: After deleting annotation from mapview self.mapView.removeAnnotation(selectedAnnotation!)

    Convert left over mapView.annotations to an array of newAnnotationDict by doing something like this:

    let newArray = self.mapView.annotations.map({ ["title": $0.title, "lat": $0.coordinate.latitude, "lng": $0.coordinate.longitude] as [String: Any]})
    

    then serialize it into data and override (instead of appending) in UserDefaults value like you have already done in your code.