-
Notifications
You must be signed in to change notification settings - Fork 238
/
Copy pathRNCGeolocation.mm
544 lines (454 loc) · 17.5 KB
/
RNCGeolocation.mm
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
/**
* Copyright (c) Facebook, Inc. and its affiliates.
*
* This source code is licensed under the MIT license found in the
* LICENSE file in the root directory of this source tree.
*/
#import "RNCGeolocation.h"
#ifdef RCT_NEW_ARCH_ENABLED
#import "RNCGeolocationSpec.h"
#endif
#import <CoreLocation/CLError.h>
#import <CoreLocation/CLLocationManager.h>
#import <CoreLocation/CLLocationManagerDelegate.h>
#import <React/RCTAssert.h>
#import <React/RCTBridge.h>
#import <React/RCTConvert.h>
#import <React/RCTEventDispatcher.h>
#import <React/RCTLog.h>
typedef NS_ENUM(NSInteger, RNCPositionErrorCode) {
RNCPositionErrorDenied = 1,
RNCPositionErrorUnavailable,
RNCPositionErrorTimeout,
};
typedef NS_ENUM(NSInteger, RNCGeolocationAuthorizationLevel) {
RNCGeolocationAuthorizationLevelDefault,
RNCGeolocationAuthorizationLevelWhenInUse,
RNCGeolocationAuthorizationLevelAlways,
};
#define RNC_DEFAULT_LOCATION_ACCURACY kCLLocationAccuracyHundredMeters
typedef struct {
BOOL skipPermissionRequests;
RNCGeolocationAuthorizationLevel authorizationLevel;
BOOL enableBackgroundLocationUpdates;
} RNCGeolocationConfiguration;
typedef struct {
double timeout;
double maximumAge;
double accuracy;
double distanceFilter;
BOOL useSignificantChanges;
} RNCGeolocationOptions;
@implementation RCTConvert (RNCGeolocationAuthorizationLevel)
RCT_ENUM_CONVERTER(RNCGeolocationAuthorizationLevel, (@{
@"whenInUse": @(RNCGeolocationAuthorizationLevelWhenInUse),
@"always": @(RNCGeolocationAuthorizationLevelAlways)}),
RNCGeolocationAuthorizationLevelDefault, integerValue)
@end
@implementation RCTConvert (RNCGeolocationOptions)
+ (RNCGeolocationConfiguration)RNCGeolocationConfiguration:(id)json
{
NSDictionary<NSString *, id> *options = [RCTConvert NSDictionary:json];
return (RNCGeolocationConfiguration) {
.skipPermissionRequests = [RCTConvert BOOL:options[@"skipPermissionRequests"]],
.authorizationLevel = [RCTConvert RNCGeolocationAuthorizationLevel:options[@"authorizationLevel"]]
};
}
+ (RNCGeolocationOptions)RNCGeolocationOptions:(id)json
{
NSDictionary<NSString *, id> *options = [RCTConvert NSDictionary:json];
double distanceFilter = options[@"distanceFilter"] == NULL ? RNC_DEFAULT_LOCATION_ACCURACY
: [RCTConvert double:options[@"distanceFilter"]] ?: kCLDistanceFilterNone;
return (RNCGeolocationOptions){
.timeout = [RCTConvert NSTimeInterval:options[@"timeout"]] ?: 1000 * 60 * 10,
.maximumAge = [RCTConvert NSTimeInterval:options[@"maximumAge"]],
.accuracy = [RCTConvert BOOL:options[@"enableHighAccuracy"]] ? kCLLocationAccuracyBest : RNC_DEFAULT_LOCATION_ACCURACY,
.distanceFilter = distanceFilter,
.useSignificantChanges = static_cast<BOOL>([RCTConvert BOOL:options[@"useSignificantChanges"]] ?: NO),
};
}
@end
static NSDictionary<NSString *, id> *RNCPositionError(RNCPositionErrorCode code, NSString *msg /* nil for default */)
{
if (!msg) {
switch (code) {
case RNCPositionErrorDenied:
msg = @"User denied access to location services.";
break;
case RNCPositionErrorUnavailable:
msg = @"Unable to retrieve location.";
break;
case RNCPositionErrorTimeout:
msg = @"The location request timed out.";
break;
}
}
return @{
@"code": @(code),
@"message": msg,
@"PERMISSION_DENIED": @(RNCPositionErrorDenied),
@"POSITION_UNAVAILABLE": @(RNCPositionErrorUnavailable),
@"TIMEOUT": @(RNCPositionErrorTimeout)
};
}
@interface RNCGeolocationRequest : NSObject
@property (nonatomic, copy) RCTResponseSenderBlock successBlock;
@property (nonatomic, copy) RCTResponseSenderBlock errorBlock;
@property (nonatomic, assign) RNCGeolocationOptions options;
@property (nonatomic, strong) NSTimer *timeoutTimer;
@end
@implementation RNCGeolocationRequest
- (void)dealloc
{
if (_timeoutTimer.valid) {
[_timeoutTimer invalidate];
}
}
@end
@interface RNCGeolocation () <CLLocationManagerDelegate>
@end
@implementation RNCGeolocation
{
CLLocationManager *_locationManager;
NSDictionary<NSString *, id> *_lastLocationEvent;
NSMutableArray<RNCGeolocationRequest *> *_pendingRequests;
BOOL _observingLocation;
BOOL _usingSignificantChanges;
RNCGeolocationConfiguration _locationConfiguration;
RNCGeolocationOptions _observerOptions;
CLAuthorizationStatus _lastUpdatedAuthorizationStatus; // used since iOS 14.0+
NSMutableArray<NSDictionary<NSString *, RCTResponseSenderBlock>*>* _queuedAuthorizationCallbacks;
}
RCT_EXPORT_MODULE()
#pragma mark - Lifecycle
- (void)dealloc
{
[self stopMonitoring];
_locationManager.delegate = nil;
}
- (dispatch_queue_t)methodQueue
{
return dispatch_get_main_queue();
}
- (NSArray<NSString *> *)supportedEvents
{
return @[@"geolocationDidChange", @"geolocationError"];
}
#pragma mark - Private API
- (void)beginLocationUpdatesWithDesiredAccuracy:(CLLocationAccuracy)desiredAccuracy distanceFilter:(CLLocationDistance)distanceFilter useSignificantChanges:(BOOL)useSignificantChanges
{
if (!_locationConfiguration.skipPermissionRequests) {
[self requestAuthorization:nil error:nil];
} else if (_locationConfiguration.enableBackgroundLocationUpdates) {
[self enableBackgroundLocationUpdates];
}
if (!_locationManager) {
_locationManager = [CLLocationManager new];
_locationManager.delegate = self;
}
_locationManager.distanceFilter = distanceFilter;
_locationManager.desiredAccuracy = desiredAccuracy;
_usingSignificantChanges = useSignificantChanges;
if (@available(iOS 14.0, *)) {
if (
_lastUpdatedAuthorizationStatus == kCLAuthorizationStatusAuthorizedAlways ||
_lastUpdatedAuthorizationStatus == kCLAuthorizationStatusAuthorizedWhenInUse
) {
[self startMonitoring];
}
} else {
[self startMonitoring];
}
}
- (void)startMonitoring
{
_usingSignificantChanges
? [_locationManager startMonitoringSignificantLocationChanges]
: [_locationManager startUpdatingLocation];
}
- (void)stopMonitoring
{
_usingSignificantChanges
? [_locationManager stopMonitoringSignificantLocationChanges]
: [_locationManager stopUpdatingLocation];
}
#pragma mark - Timeout handler
- (void)timeout:(NSTimer *)timer
{
RNCGeolocationRequest *request = timer.userInfo;
NSString *message = [NSString stringWithFormat: @"Unable to fetch location within %.1fs.", request.options.timeout];
request.errorBlock(@[RNCPositionError(RNCPositionErrorTimeout, message)]);
[_pendingRequests removeObject:request];
// Stop updating if no pending requests
if (_pendingRequests.count == 0 && !_observingLocation) {
[self stopMonitoring];
}
}
#pragma mark - Public API
RCT_REMAP_METHOD(setConfiguration, setConfiguration:(RNCGeolocationConfiguration)config)
{
_locationConfiguration = config;
}
RCT_REMAP_METHOD(requestAuthorization, requestAuthorization:(RCTResponseSenderBlock)successBlock
error:(RCTResponseSenderBlock)errorBlock)
{
if (!_locationManager) {
_locationManager = [CLLocationManager new];
_locationManager.delegate = self;
}
if (successBlock != nil || errorBlock != nil) {
if (_queuedAuthorizationCallbacks == nil) {
_queuedAuthorizationCallbacks = [NSMutableArray new];
}
[_queuedAuthorizationCallbacks addObject:@{
@"success": successBlock,
@"error": errorBlock,
}];
}
BOOL wantsAlways = NO;
BOOL wantsWhenInUse = NO;
if (_locationConfiguration.authorizationLevel == RNCGeolocationAuthorizationLevelDefault) {
if ([[NSBundle mainBundle] objectForInfoDictionaryKey:@"NSLocationAlwaysUsageDescription"] &&
[_locationManager respondsToSelector:@selector(requestAlwaysAuthorization)]) {
wantsAlways = YES;
} else if ([[NSBundle mainBundle] objectForInfoDictionaryKey:@"NSLocationWhenInUseUsageDescription"] &&
[_locationManager respondsToSelector:@selector(requestWhenInUseAuthorization)]) {
wantsWhenInUse = YES;
}
} else if (_locationConfiguration.authorizationLevel == RNCGeolocationAuthorizationLevelAlways) {
wantsAlways = YES;
} else if (_locationConfiguration.authorizationLevel == RNCGeolocationAuthorizationLevelWhenInUse) {
wantsWhenInUse = YES;
}
// Request location access permission
if (wantsAlways) {
[_locationManager requestAlwaysAuthorization];
[self enableBackgroundLocationUpdates];
} else if (wantsWhenInUse) {
[_locationManager requestWhenInUseAuthorization];
}
}
- (void)enableBackgroundLocationUpdates
{
// iOS 9+ requires explicitly enabling background updates
NSArray *backgroundModes = [[NSBundle mainBundle] objectForInfoDictionaryKey:@"UIBackgroundModes"];
if (backgroundModes && [backgroundModes containsObject:@"location"]) {
if ([_locationManager respondsToSelector:@selector(setAllowsBackgroundLocationUpdates:)]) {
[_locationManager setAllowsBackgroundLocationUpdates:YES];
}
}
}
RCT_REMAP_METHOD(startObserving, startObserving:(RNCGeolocationOptions)options)
{
checkLocationConfig();
// Select best options
_observerOptions = options;
for (RNCGeolocationRequest *request in _pendingRequests) {
_observerOptions.accuracy = MIN(_observerOptions.accuracy, request.options.accuracy);
}
[self beginLocationUpdatesWithDesiredAccuracy:_observerOptions.accuracy
distanceFilter:_observerOptions.distanceFilter
useSignificantChanges:_observerOptions.useSignificantChanges];
_observingLocation = YES;
}
RCT_REMAP_METHOD(stopObserving, stopObserving)
{
// Stop observing
_observingLocation = NO;
// Stop updating if no pending requests
if (_pendingRequests.count == 0) {
[self stopMonitoring];
}
}
RCT_REMAP_METHOD(getCurrentPosition, getCurrentPosition:(RNCGeolocationOptions)options
position:(RCTResponseSenderBlock)successBlock
error:(RCTResponseSenderBlock)errorBlock)
{
checkLocationConfig();
if (!successBlock) {
RCTLogError(@"%@.getCurrentPosition called with nil success parameter.", [self class]);
return;
}
if (errorBlock) {
if (@available(iOS 14.0, *)) {
if (_lastUpdatedAuthorizationStatus == kCLAuthorizationStatusRestricted) {
errorBlock(@[RNCPositionError(RNCPositionErrorUnavailable, @"This application is not authorized to use location services")]);
return;
}
if (_lastUpdatedAuthorizationStatus == kCLAuthorizationStatusDenied) {
errorBlock(@[RNCPositionError(RNCPositionErrorDenied, nil)]);
return;
}
} else {
if (![CLLocationManager locationServicesEnabled]) {
errorBlock(@[RNCPositionError(RNCPositionErrorUnavailable, @"Location services disabled.")]);
return;
}
if ([CLLocationManager authorizationStatus] == kCLAuthorizationStatusDenied) {
errorBlock(@[RNCPositionError(RNCPositionErrorDenied, nil)]);
return;
}
}
}
// Check if previous recorded location exists and is good enough
if (_lastLocationEvent &&
[NSDate date].timeIntervalSince1970 - [RCTConvert NSTimeInterval:_lastLocationEvent[@"timestamp"]] < options.maximumAge &&
[_lastLocationEvent[@"coords"][@"accuracy"] doubleValue] <= options.accuracy) {
// Call success block with most recent known location
successBlock(@[_lastLocationEvent]);
return;
}
// Create request
RNCGeolocationRequest *request = [RNCGeolocationRequest new];
request.successBlock = successBlock;
request.errorBlock = errorBlock ?: ^(NSArray *args){};
request.options = options;
request.timeoutTimer = [NSTimer scheduledTimerWithTimeInterval:options.timeout
target:self
selector:@selector(timeout:)
userInfo:request
repeats:NO];
if (!_pendingRequests) {
_pendingRequests = [NSMutableArray new];
}
[_pendingRequests addObject:request];
// Configure location manager and begin updating location
CLLocationAccuracy accuracy = options.accuracy;
if (_locationManager) {
accuracy = MIN(_locationManager.desiredAccuracy, accuracy);
}
[self beginLocationUpdatesWithDesiredAccuracy:accuracy
distanceFilter:options.distanceFilter
useSignificantChanges:options.useSignificantChanges];
}
#pragma mark - CLLocationManagerDelegate
- (void)locationManager:(CLLocationManager *)manager
didUpdateLocations:(NSArray<CLLocation *> *)locations
{
// Create event
CLLocation *location = locations.lastObject;
BOOL mocked = NO;
if (@available(iOS 15.0, *)) {
mocked = location.sourceInformation.isSimulatedBySoftware;
}
_lastLocationEvent = @{
@"mocked": @(mocked),
@"coords": @{
@"latitude": @(location.coordinate.latitude),
@"longitude": @(location.coordinate.longitude),
@"altitude": @(location.altitude),
@"accuracy": @(location.horizontalAccuracy),
@"altitudeAccuracy": @(location.verticalAccuracy),
@"heading": @(location.course),
@"speed": @(location.speed),
},
@"timestamp": @([location.timestamp timeIntervalSince1970] * 1000) // in ms
};
// Send event
if (_observingLocation) {
[self sendEventWithName:@"geolocationDidChange" body:_lastLocationEvent];
}
// Fire all queued callbacks
for (RNCGeolocationRequest *request in _pendingRequests) {
request.successBlock(@[_lastLocationEvent]);
[request.timeoutTimer invalidate];
}
[_pendingRequests removeAllObjects];
// Stop updating if not observing
if (!_observingLocation) {
[self stopMonitoring];
}
}
- (void)locationManagerDidChangeAuthorization:(CLLocationManager *)manager
{
CLAuthorizationStatus currentStatus;
if (@available(iOS 14.0, *)) {
currentStatus = manager.authorizationStatus;
} else {
currentStatus = [CLLocationManager authorizationStatus];
}
if (
currentStatus == kCLAuthorizationStatusAuthorizedAlways ||
currentStatus == kCLAuthorizationStatusAuthorizedWhenInUse
) {
if (_queuedAuthorizationCallbacks != nil && _queuedAuthorizationCallbacks.count > 0){
for (NSDictionary<NSString *, RCTResponseSenderBlock>* callbacks in _queuedAuthorizationCallbacks) {
[callbacks objectForKey:@"success"](@[]);
}
_queuedAuthorizationCallbacks = nil;
}
[self startMonitoring];
} else {
NSDictionary<NSString *, id> *jsError = currentStatus == kCLAuthorizationStatusRestricted
? RNCPositionError(RNCPositionErrorUnavailable, @"This application is not authorized to use location services")
: currentStatus == kCLAuthorizationStatusDenied
? RNCPositionError(RNCPositionErrorDenied, nil)
: nil;
if (jsError != nil) {
if (_observingLocation) {
[self sendEventWithName:@"geolocationError" body:jsError];
}
if (_queuedAuthorizationCallbacks != nil && _queuedAuthorizationCallbacks.count > 0){
for (NSDictionary<NSString *, RCTResponseSenderBlock>* callbacks in _queuedAuthorizationCallbacks) {
[callbacks objectForKey:@"error"](@[jsError]);
}
_queuedAuthorizationCallbacks = nil;
}
// Fire all queued error callbacks
for (RNCGeolocationRequest *request in _pendingRequests) {
request.errorBlock(@[jsError]);
[request.timeoutTimer invalidate];
}
[_pendingRequests removeAllObjects];
}
// Stop updating if user has explicitly denied authorization for this application, or
// location services are disabled in Settings, or any other reason.
[self stopMonitoring];
}
_lastUpdatedAuthorizationStatus = currentStatus;
}
- (void)locationManager:(CLLocationManager *)manager didFailWithError:(NSError *)error
{
// Check error type
NSDictionary<NSString *, id> *jsError = nil;
switch (error.code) {
case kCLErrorDenied:
jsError = RNCPositionError(RNCPositionErrorDenied, nil);
break;
case kCLErrorNetwork:
jsError = RNCPositionError(RNCPositionErrorUnavailable, @"Unable to retrieve location due to a network failure");
break;
case kCLErrorLocationUnknown:
default:
jsError = RNCPositionError(RNCPositionErrorUnavailable, nil);
break;
}
// Send event
if (_observingLocation) {
[self sendEventWithName:@"geolocationError" body:jsError];
}
// Fire all queued error callbacks
for (RNCGeolocationRequest *request in _pendingRequests) {
request.errorBlock(@[jsError]);
[request.timeoutTimer invalidate];
}
[_pendingRequests removeAllObjects];
}
#ifdef RCT_NEW_ARCH_ENABLED
- (std::shared_ptr<facebook::react::TurboModule>)getTurboModule:
(const facebook::react::ObjCTurboModule::InitParams &)params
{
return std::make_shared<facebook::react::NativeRNCGeolocationSpecJSI>(params);
}
#endif
static void checkLocationConfig()
{
#if RCT_DEV
if (!([[NSBundle mainBundle] objectForInfoDictionaryKey:@"NSLocationWhenInUseUsageDescription"] ||
[[NSBundle mainBundle] objectForInfoDictionaryKey:@"NSLocationAlwaysUsageDescription"] ||
[[NSBundle mainBundle] objectForInfoDictionaryKey:@"NSLocationAlwaysAndWhenInUseUsageDescription"])) {
RCTLogError(@"Either NSLocationWhenInUseUsageDescription or NSLocationAlwaysUsageDescription or NSLocationAlwaysAndWhenInUseUsageDescription key must be present in Info.plist to use geolocation.");
}
#endif
}
@end