/*
 * Copyright (C) 2009 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. 
 */

#import "config.h"

#if ENABLE(GEOLOCATION) && !ENABLE(CLIENT_BASED_GEOLOCATION)

#import "GeolocationServiceMac.h"

#import "Geoposition.h"
#import "PositionError.h"
#import "PositionOptions.h"
#import "SoftLinking.h"
#import <CoreLocation/CoreLocation.h>
#import <objc/objc-runtime.h>
#import <wtf/RefPtr.h>
#import <wtf/UnusedParam.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 WebCoreCoreLocationObserver : NSObject<CLLocationManagerDelegate>
{
    GeolocationServiceMac* m_callback;
}

- (id)initWithCallback:(GeolocationServiceMac*)callback;

- (void)locationManager:(CLLocationManager *)manager didUpdateToLocation:(CLLocation *)newLocation fromLocation:(CLLocation *)oldLocation;
- (void)locationManager:(CLLocationManager *)manager didFailWithError:(NSError *)error;

@end

namespace WebCore {

GeolocationService* GeolocationServiceMac::create(GeolocationServiceClient* client)
{
    return new GeolocationServiceMac(client);
}

GeolocationService::FactoryFunction* GeolocationService::s_factoryFunction = &GeolocationServiceMac::create;

GeolocationServiceMac::GeolocationServiceMac(GeolocationServiceClient* client)
    : GeolocationService(client)
    , m_objcObserver(AdoptNS, [[WebCoreCoreLocationObserver alloc] initWithCallback:this])
{
}

GeolocationServiceMac::~GeolocationServiceMac()
{
    [m_locationManager.get() stopUpdatingLocation];
    m_locationManager.get().delegate = nil;
}

bool GeolocationServiceMac::startUpdating(PositionOptions* options)
{
    #define CLLocationManager getCLLocationManagerClass()
    if (!m_locationManager.get()) {
        m_locationManager.adoptNS([[CLLocationManager alloc] init]);
        m_locationManager.get().delegate = m_objcObserver.get();
    }

    if (!m_locationManager.get().locationServicesEnabled)
        return false;

    if (options) {
        // CLLocationAccuracy values suggested by Ron Huang.
        CLLocationAccuracy accuracy = options->enableHighAccuracy() ? kCLLocationAccuracyBest : kCLLocationAccuracyHundredMeters;
        m_locationManager.get().desiredAccuracy = accuracy;
    }
    
    // This can safely be called multiple times.
    [m_locationManager.get() startUpdatingLocation];
    
    return true;
    #undef CLLocationManager
}

void GeolocationServiceMac::stopUpdating()
{
    [m_locationManager.get() stopUpdatingLocation];
}

void GeolocationServiceMac::suspend()
{
    [m_locationManager.get() stopUpdatingLocation];
}

void GeolocationServiceMac::resume()
{
    [m_locationManager.get() startUpdatingLocation];
}

void GeolocationServiceMac::positionChanged(PassRefPtr<Geoposition> position)
{
    m_lastPosition = position;
    GeolocationService::positionChanged();
}
    
void GeolocationServiceMac::errorOccurred(PassRefPtr<PositionError> error)
{
    m_lastError = error;
    GeolocationService::errorOccurred();
}

} // namespace WebCore

@implementation WebCoreCoreLocationObserver

- (id)initWithCallback:(GeolocationServiceMac *)callback
{
    self = [super init];
    if (self)
        m_callback = callback;
    return self;
}

- (void)locationManager:(CLLocationManager *)manager didUpdateToLocation:(CLLocation *)newLocation fromLocation:(CLLocation *)oldLocation
{
    ASSERT(m_callback);
    ASSERT(newLocation);
    UNUSED_PARAM(manager);
    UNUSED_PARAM(oldLocation);

    // Normalize
    bool canProvideAltitude = true;
    bool canProvideAltitudeAccuracy = true;
    double altitude = newLocation.altitude;
    double altitudeAccuracy = newLocation.verticalAccuracy;
    if (altitudeAccuracy < 0.0) {
        canProvideAltitude = false;
        canProvideAltitudeAccuracy = false;
    }
    
    bool canProvideSpeed = true;
    double speed = newLocation.speed;
    if (speed < 0.0)
        canProvideSpeed = false;

    bool canProvideHeading = true;
    double heading = newLocation.course;
    if (heading < 0.0)
        canProvideHeading = false;
    
    WTF::RefPtr<WebCore::Coordinates> newCoordinates = WebCore::Coordinates::create(
                            newLocation.coordinate.latitude,
                            newLocation.coordinate.longitude,
                            canProvideAltitude,
                            altitude,
                            newLocation.horizontalAccuracy,
                            canProvideAltitudeAccuracy,
                            altitudeAccuracy,
                            canProvideHeading,
                            heading,
                            canProvideSpeed,
                            speed);
    WTF::RefPtr<WebCore::Geoposition> newPosition = WebCore::Geoposition::create(
                             newCoordinates.release(),
                             [newLocation.timestamp timeIntervalSince1970] * 1000.0); // seconds -> milliseconds
    
    m_callback->positionChanged(newPosition.release());
}

- (void)locationManager:(CLLocationManager *)manager didFailWithError:(NSError *)error
{
    ASSERT(m_callback);
    ASSERT(error);
 
    UNUSED_PARAM(manager);

    PositionError::ErrorCode code;
    switch ([error code]) {
        case kCLErrorDenied:
            code = PositionError::PERMISSION_DENIED;
            break;
        case kCLErrorLocationUnknown:
            code = PositionError::POSITION_UNAVAILABLE;
            break;
        default:
            code = PositionError::POSITION_UNAVAILABLE;
            break;
    }

    m_callback->errorOccurred(PositionError::create(code, [error localizedDescription]));
}

@end

#endif // ENABLE(GEOLOCATION)