Background task in iOS action extension

后端 未结 4 1452
梦毁少年i
梦毁少年i 2021-02-13 02:13

I\'m working on an action extension for an app which prepares some data for the user and then uses MailCore2 to send this information to a SMTP server. Preparing data is done ve

相关标签:
4条回答
  • 2021-02-13 02:23

    It seems like you can't schedule long running running tasks in background from an extension.Refer here(Some APIs Are Unavailable to App Extensions)

    Among the solutions you suggested,my suggestion would be to try using silent push notifications ,you can call an api at the preparing data stage,then sent a silent push from server and perform the background task when the silent push arrives.

    0 讨论(0)
  • 2021-02-13 02:27

    Judging from your response, your difficulties is that you need to allow background activities. You can do that by following a tutorial such as this one.

    0 讨论(0)
  • 2021-02-13 02:28

    The most useful solution is using push notifications. The solution I used in my own application:

    1. Create a simple server which can retrieve a simple request: device token and time when you'd like to wake up your device.
    2. When the application goes to background mode send the request to the server to wake up the device later. E.g. 5 minutes/20 minutes, etc. func applicationWillEnterForeground(_ application: UIApplication) { inside AppDelegate.
    3. Don't forget to allow the application to work in the background as much as possible.

    For example,

       @UIApplicationMain
        class AppDelegate: UIResponder {
        var backgroundTask: UIBackgroundTaskIdentifier = UIBackgroundTaskInvalid
        func application(_ application: UIApplication,
                             didReceiveRemoteNotification userInfo: [AnyHashable: Any],
                             fetchCompletionHandler completionHandler: @escaping (UIBackgroundFetchResult) -> Void) {
                if UIApplication.shared.applicationState != .active {
                    doFetch(completionHandler: completionHandler)
                } else {
                    // foreground here
                    completionHandler(UIBackgroundFetchResult.noData)
                }
            }
        func application(_ application: UIApplication,
                             performFetchWithCompletionHandler completionHandler: @escaping (UIBackgroundFetchResult) -> Void) {
                doFetch(completionHandler: completionHandler)
            }
        private func doFetch(completionHandler: @escaping (UIBackgroundFetchResult) -> Void) {
        sendTheRequestToWakeApp()
        backgroundTask = UIApplication.shared.beginBackgroundTask { [weak self] in
                    self?.endBackgroundTask()
                }
    /// do work here
        completionHandler(UIBackgroundFetchResult.noData)
        }
        private func endBackgroundTask() {
                print("Background task ended.")
                UIApplication.shared.endBackgroundTask(backgroundTask)
                backgroundTask = UIBackgroundTaskInvalid
            }
        private func sendTheRequestToWakeApp() {
        /// Implement request using native library or Alamofire. etc.
        }
        }
    

    on server side use simple time or loop.

    Disadvantages,

    1. Internet is required
    2. It doesn't perfectly work. When battery is low the background mode is restricted.

    Don't forget to setup the project:

    0 讨论(0)
  • 2021-02-13 02:44

    After many tests and failures, I found the following solution to execute a long performing task in the background of an extension. This works as expected, even if the extension is already finished:

    func performTask()
    {
        // Perform the task in background.
        let processinfo = ProcessInfo()
        processinfo.performExpiringActivity(withReason: "Long task") { (expired) in
            if (!expired) {
                // Run task synchronously.
                self.performLongTask()
            }
            else {
                // Cancel task.
                self.cancelLongTask()
            }
        }
    }
    

    This code uses ProcessInfo.performExpiringActivity() to execute the task in another thread. It’s important that the task in performLongTask() is executed synchronously. When the end of the block is reached, the thread will terminate and end the execution of your task.

    A similar approach is also working in the main app. It's described in detail in a small summary for background tasks in iOS.

    0 讨论(0)
提交回复
热议问题