Xcode / iOS simulator: Trigger significant location change manually

IphoneIosCore LocationCllocationmanager

Iphone Problem Overview


Is there a way to manually tell the device to dispatch a significant location change notification that will wake up any applications registered for this notification? This is for testing only, and I realize this private API call would get rejected upon submission to the app store.

Iphone Solutions


Solution 1 - Iphone

So two answers:

A) Hop on a train :-)

B) Use the simulator. In iOS 5.x simulator, there is a debug menu that has a location submenu. Choose freeway drive. This will start the simulator on an imaginary journey down the scenic 280 in Northern California. It gives you everything but the view: your app will get Significant Location Change updates, and will also be launched in the background if it has been suspended.

To verify that you are actually moving, launch safari in the simulator, and go to maps.google.com and click the little track my location button. You should be moving.

Awesome! Now how to debug the lifecycle problem of being launched by the system? Easy! Have xCode wait for your app to be launched to start debugging. From the Scheme menu, choose edit scheme. In the Run Scheme, and the Info tab, for the "Launch" setting choose : "Wait for My.app to launch".

Run your app once in the simulator, so that it starts monitoring for location updates, then force quit it, so that it is suspended. Add a break point in your application did finish launching function, and wait. As soon as your simulator has gone far enough, your application will be woken up, your breakpoint hit, and you are in the money.

But really, the train ride is more fun.

Solution 2 - Iphone

Well, I've found that I can do this by toggling on and off Airplane mode and/or WiFi. Perhaps start the app with the device in airplane mode, then close the app and turn airplane mode off. That will turn the GPS on and force a location update to be dispatched.

Solution 3 - Iphone

I also wanted to test the relaunch of my terminated app which uses significant change monitoring. I wrote a piece of code so that it would display a local notification when it gets launched by a location key in the launch options dictionary.

I ran my app in the simulator. Then killed it from the multitasking bar. Then I set the location of the iOS simulator to a custom location. I quit the simulator and started it again. My app received the significant location update and showed the local notification.

Solution 4 - Iphone

I was struggling with the same issue, how to test 'startMonitoringSignificantLocationChanges' and check if my app is receiving location updates when suspended.

I couldn't manage to catch the execution on a breakpoint but I managed to see the results of my implementation working by sending the new location data to the server.

The whole flow:

  • Implemented with 'startMonitoringSignificantLocationChanges' and an API call to my server to update the location latitude and longitude
  • Set the location updates background mode capability to true
  • Run the app so the location manager is initiated and the app is listening for location changes
  • Force closed the app
  • Set the debug->location on simulator to freeway drive
  • Opened maps to see if the location is changing
  • Waited on the server to see for location updates and was getting new results every about 3 minutes

However, I'm still not sure if this is fine enough on a real device.

I'm working on Xcode Version 6.0.1 (6A317); tested on Simulator iPhone 5s (8.0).

Solution 5 - Iphone

Adding to MagicSeth's answer, if you need to test this on a real device instead of a simulator, you can trigger a background launch with UIApplication.LaunchOptionsKey.location key by disabling and re-enabling Location Services in General > Privacy screen in the Settings app.

Solution 6 - Iphone

Depending on your scenario I would suggest two solutions:

  1. Use a Timer or LocalNotification that periodically calls stopMonitoringSignificantLocationChanges followed by startMonitoringSignificantLocationChanges which should trigger a new location to be sent to your code (might be the same Location as before).

  2. Build your own GPS Simulator that you start in debug builds and that will call the same delegate methods like CLLocationManager would do.

Solution 7 - Iphone

One thing I noticed in iOS 7 and Xcode 5.1.1 - If you are expecting SLC events to fire up your app into background mode, it may or may not hit the breakpoints you set. For me, sometimes the NSLog message are not even showing.

If that's the case for you, you can view NSLog outputs from the System Log. You can open the System Log from iOS Simulator's Debug menu.

Solution 8 - Iphone

In iOS 4, you can register for significant location changes. From the Apple docs: With this service, location updates are generated only when the user’s location changes significantly; thus, it is ideal for social applications or applications that provide the user with noncritical, location-relevant information. If the application is suspended when an update occurs, the system wakes it up in the background to handle the update. If the application starts this service and is then terminated, the system relaunches the application automatically when a new location becomes available. This service is available in iOS 4 and later, only on devices that contain a cellular radio.

See the Apple docs here and here.

Here is some example code to register for signification location updates:

- (void)startSignificantChangeUpdates {

// Create the location manager if it doesn't exist
if (nil == locationManager)
    locationManager = [[CLLocationManager alloc] init];

locationManager.delegate = self;
[locationManager startMonitoringSignificantLocationChanges];

}

The docs say: if you leave this service running and your application is subsequently suspended or terminated, the service automatically wakes up your application when new location data arrives. At wake-up time, your application is put into the background and given a small amount of time to process the location data. Because your application is in the background, it should do minimal work and avoid any tasks (such as querying the network) that might prevent it from returning before the allocated time expires. If it does not, your application may be terminated.

Solution 9 - Iphone

Freeway drive

enter image description here

// MARK: - MKMapViewDelegate

  func mapView(_ mapView: MKMapView, didUpdate userLocation: MKUserLocation) {
    guard let location = userLocation.location else { return }

    print(location.speed)
  }

Solution 10 - Iphone

Well, this is not possible as the application scope is limited to its own space and such kind of notification cannot be generated with Apple Documented APIs list. Of course.. if any undocumented API is used, application will get rejection from apple due to the use of undocumented/private API.

Attributions

All content for this solution is sourced from the original question on Stackoverflow.

The content on this page is licensed under the Attribution-ShareAlike 4.0 International (CC BY-SA 4.0) license.

Content TypeOriginal AuthorOriginal Content on Stackoverflow
QuestionNickView Question on Stackoverflow
Solution 1 - IphoneMagicSethView Answer on Stackoverflow
Solution 2 - IphoneChrisView Answer on Stackoverflow
Solution 3 - IphoneaslisabanciView Answer on Stackoverflow
Solution 4 - IphoneBardh LohajView Answer on Stackoverflow
Solution 5 - IphoneEugene TulushevView Answer on Stackoverflow
Solution 6 - IphonesliverView Answer on Stackoverflow
Solution 7 - Iphonekesun421View Answer on Stackoverflow
Solution 8 - IphoneJane SalesView Answer on Stackoverflow
Solution 9 - Iphoneonmyway133View Answer on Stackoverflow
Solution 10 - IphoneSaurabh PassoliaView Answer on Stackoverflow