blob: de9133bf08faabcaf7f85dbdad5b08a134a23faf [file] [log] [blame]
/*
* Copyright (C) 2008, 2009, 2010, 2012, 2014 Apple Inc. All Rights Reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR
* CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
* EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
* PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
* OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
#if PLATFORM(IOS_FAMILY)
#import "WebGeolocationCoreLocationProvider.h"
#import <CoreLocation/CLLocation.h>
#import <CoreLocation/CLLocationManagerDelegate.h>
#import <CoreLocation/CoreLocation.h>
#import <WebCore/GeolocationPositionData.h>
#import <WebKitLogging.h>
#import <objc/objc-runtime.h>
#import <wtf/RefPtr.h>
#import <wtf/RetainPtr.h>
#import <wtf/SoftLinking.h>
SOFT_LINK_FRAMEWORK(CoreLocation)
SOFT_LINK_CLASS(CoreLocation, CLLocationManager)
SOFT_LINK_CLASS(CoreLocation, CLLocation)
SOFT_LINK_CONSTANT(CoreLocation, kCLLocationAccuracyBest, double)
SOFT_LINK_CONSTANT(CoreLocation, kCLLocationAccuracyHundredMeters, double)
#define kCLLocationAccuracyBest getkCLLocationAccuracyBest()
#define kCLLocationAccuracyHundredMeters getkCLLocationAccuracyHundredMeters()
using namespace WebCore;
@interface WebGeolocationCoreLocationProvider () <CLLocationManagerDelegate>
@end
@implementation WebGeolocationCoreLocationProvider
{
id<WebGeolocationCoreLocationUpdateListener> _positionListener;
RetainPtr<CLLocationManager> _locationManager;
BOOL _isWaitingForAuthorization;
CLAuthorizationStatus _lastAuthorizationStatus;
}
- (void)createLocationManager
{
ASSERT(!_locationManager);
_locationManager = adoptNS([allocCLLocationManagerInstance() init]);
_lastAuthorizationStatus = [getCLLocationManagerClass() authorizationStatus];
[ _locationManager setDelegate:self];
}
- (id)initWithListener:(id<WebGeolocationCoreLocationUpdateListener>)listener
{
self = [super init];
if (self) {
_positionListener = listener;
[self createLocationManager];
}
return self;
}
- (void)dealloc
{
[_locationManager setDelegate:nil];
[super dealloc];
}
- (void)requestGeolocationAuthorization
{
#if PLATFORM(MACCATALYST)
[_positionListener geolocationAuthorizationDenied];
return;
#else
if (![getCLLocationManagerClass() locationServicesEnabled]) {
[_positionListener geolocationAuthorizationDenied];
return;
}
switch ([getCLLocationManagerClass() authorizationStatus]) {
case kCLAuthorizationStatusNotDetermined: {
if (!_isWaitingForAuthorization) {
_isWaitingForAuthorization = YES;
[_locationManager requestWhenInUseAuthorization];
}
break;
}
case kCLAuthorizationStatusAuthorizedAlways:
case kCLAuthorizationStatusAuthorizedWhenInUse: {
[_positionListener geolocationAuthorizationGranted];
break;
}
case kCLAuthorizationStatusRestricted:
case kCLAuthorizationStatusDenied:
[_positionListener geolocationAuthorizationDenied];
break;
}
#endif
}
static bool isAuthorizationGranted(CLAuthorizationStatus authorizationStatus)
{
return authorizationStatus == kCLAuthorizationStatusAuthorizedAlways || authorizationStatus == kCLAuthorizationStatusAuthorizedWhenInUse;
}
- (void)start
{
if (![getCLLocationManagerClass() locationServicesEnabled]
|| !isAuthorizationGranted([getCLLocationManagerClass() authorizationStatus])) {
[_locationManager stopUpdatingLocation];
[_positionListener resetGeolocation];
return;
}
[_locationManager startUpdatingLocation];
}
- (void)stop
{
[_locationManager stopUpdatingLocation];
}
- (void)locationManager:(CLLocationManager *)manager didChangeAuthorizationStatus:(CLAuthorizationStatus)status
{
if (_isWaitingForAuthorization) {
switch (status) {
case kCLAuthorizationStatusNotDetermined:
// This can happen after resume if the user has still not answered the dialog. We just have to wait for the permission.
break;
case kCLAuthorizationStatusDenied:
case kCLAuthorizationStatusRestricted:
_isWaitingForAuthorization = NO;
[_positionListener geolocationAuthorizationDenied];
break;
case kCLAuthorizationStatusAuthorizedAlways:
case kCLAuthorizationStatusAuthorizedWhenInUse:
_isWaitingForAuthorization = NO;
[_positionListener geolocationAuthorizationGranted];
break;
}
} else {
if (!(isAuthorizationGranted(_lastAuthorizationStatus) && isAuthorizationGranted(status))) {
[_locationManager stopUpdatingLocation];
[_positionListener resetGeolocation];
}
}
_lastAuthorizationStatus = status;
}
- (void)sendLocation:(CLLocation *)newLocation
{
[_positionListener positionChanged:GeolocationPositionData { newLocation }];
}
- (void)locationManager:(CLLocationManager *)manager didUpdateLocations:(NSArray *)locations
{
UNUSED_PARAM(manager);
for (CLLocation *location in locations)
[self sendLocation:location];
}
- (void)locationManager:(CLLocationManager *)manager didFailWithError:(NSError *)error
{
ASSERT(_positionListener);
ASSERT(error);
UNUSED_PARAM(manager);
if ([error code] == kCLErrorDenied) {
// Ignore the error here and let locationManager:didChangeAuthorizationStatus: handle the permission.
return;
}
NSString *errorMessage = [error localizedDescription];
[_positionListener errorOccurred:errorMessage];
}
- (void)setEnableHighAccuracy:(BOOL)flag
{
[_locationManager setDesiredAccuracy:flag ? kCLLocationAccuracyBest : kCLLocationAccuracyHundredMeters];
}
@end
#endif // PLATFORM(IOS_FAMILY)