Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

UIApplication.delegate must be used from main thread only [duplicate]

I have the following code in my app delegate as a shortcut for working with CoreData in my other viewControllers:

let ad = UIApplication.shared.delegate as! AppDelegate
let context = ad.persistentContainer.viewContext

However, I now get the error message:

"UI API called from background thread" and "UIApplication.delegate must be used from main thread only".

I am working with CoreData while my app is in the background, but this is the first time I've seen this error message. Does anybody know what's going on here?

Update: I tried to move this inside the appDelegate class itself, and using the following code -

let dispatch = DispatchQueue.main.async {
    let ad = UIApplication.shared.delegate as! AppDelegate
    let context = ad.persistentContainer.viewContext
}

Now, I can no longer access the ad and context variables outside the AppDelegate. Is there something I'm missing?

like image 640
LFHS Avatar asked Nov 09 '17 13:11

LFHS


1 Answers

With ref to this (-[UIApplication delegate] must be called from main thread only) in Swift (for your query resolution)

    DispatchQueue.main.async(execute: {

      // Handle further UI related operations here....
      //let ad = UIApplication.shared.delegate as! AppDelegate
      //let context = ad.persistentContainer.viewContext   

    })

With edit: (Where is the correct place to declare ad and context? Should I declare these in my viewControllers in the main dispatch)
Place of variables (ad and context) declaration defines scope for it. You need to decide what would be scope of these variable. You can declare them Project or application level (Globally), class level or particular this function level. If you want to use these variable in other ViewControllers then declare it globally or class level with public/open/internal access control.

   var ad: AppDelegate!    //or var ad: AppDelegate?
   var context: NSManagedObjectContext!    //or var context: NSManagedObjectContext?


   DispatchQueue.main.async(execute: {

      // Handle further UI related operations here....
      ad = UIApplication.shared.delegate as! AppDelegate
      context = ad.persistentContainer.viewContext   

      //or 

      //self.ad = UIApplication.shared.delegate as! AppDelegate
      //self.context = ad.persistentContainer.viewContext   

    })
like image 130
Krunal Avatar answered Sep 29 '22 21:09

Krunal