/*
 * Copyright 2009, The Android Open Source Project
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions
 * are met:
 *  * Redistributions of source code must retain the above copyright
 *    notice, this list of conditions and the following disclaimer.
 *  * 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 THE COPYRIGHT HOLDERS ``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 THE COPYRIGHT OWNER 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.
 */

#include "config.h"
#include "GeolocationServiceBridge.h"

#include "GeolocationServiceAndroid.h"
#include "Geoposition.h"
#include "PositionError.h"
#include "WebViewCore.h"
#include <JNIHelp.h>

namespace WebCore {

using JSC::Bindings::getJNIEnv;

static const char* javaGeolocationServiceClassName = "android/webkit/GeolocationService";
enum javaGeolocationServiceClassMethods {
    GeolocationServiceMethodInit = 0,
    GeolocationServiceMethodStart,
    GeolocationServiceMethodStop,
    GeolocationServiceMethodSetEnableGps,
    GeolocationServiceMethodCount,
};
static jmethodID javaGeolocationServiceClassMethodIDs[GeolocationServiceMethodCount];

static const JNINativeMethod javaGeolocationServiceClassNativeMethods[] = {
    { "nativeNewLocationAvailable", "(JLandroid/location/Location;)V",
        (void*) GeolocationServiceBridge::newLocationAvailable },
    { "nativeNewErrorAvailable", "(JLjava/lang/String;)V",
        (void*) GeolocationServiceBridge::newErrorAvailable }
};

static const char *javaLocationClassName = "android/location/Location";
enum javaLocationClassMethods {
    LocationMethodGetLatitude = 0,
    LocationMethodGetLongitude,
    LocationMethodHasAltitude,
    LocationMethodGetAltitude,
    LocationMethodHasAccuracy,
    LocationMethodGetAccuracy,
    LocationMethodHasBearing,
    LocationMethodGetBearing,
    LocationMethodHasSpeed,
    LocationMethodGetSpeed,
    LocationMethodGetTime,
    LocationMethodCount,
};
static jmethodID javaLocationClassMethodIDs[LocationMethodCount];

GeolocationServiceBridge::GeolocationServiceBridge(ListenerInterface* listener)
    : m_listener(listener)
    , m_javaGeolocationServiceObject(0)
{
    ASSERT(m_listener);
    startJavaImplementation();
}

GeolocationServiceBridge::~GeolocationServiceBridge()
{
    stop();
    stopJavaImplementation();
}

void GeolocationServiceBridge::start()
{
    ASSERT(m_javaGeolocationServiceObject);
    getJNIEnv()->CallVoidMethod(m_javaGeolocationServiceObject,
                                javaGeolocationServiceClassMethodIDs[GeolocationServiceMethodStart]);
}

void GeolocationServiceBridge::stop()
{
    ASSERT(m_javaGeolocationServiceObject);
    getJNIEnv()->CallVoidMethod(m_javaGeolocationServiceObject,
                                javaGeolocationServiceClassMethodIDs[GeolocationServiceMethodStop]);
}

void GeolocationServiceBridge::setEnableGps(bool enable)
{
    ASSERT(m_javaGeolocationServiceObject);
    getJNIEnv()->CallVoidMethod(m_javaGeolocationServiceObject,
                                javaGeolocationServiceClassMethodIDs[GeolocationServiceMethodSetEnableGps],
                                enable);
}

void GeolocationServiceBridge::newLocationAvailable(JNIEnv* env, jclass, jlong nativeObject, jobject location)
{
    ASSERT(nativeObject);
    ASSERT(location);
    GeolocationServiceBridge* object = reinterpret_cast<GeolocationServiceBridge*>(nativeObject);
    object->m_listener->newPositionAvailable(toGeoposition(env, location));
}

void GeolocationServiceBridge::newErrorAvailable(JNIEnv* env, jclass, jlong nativeObject, jstring message)
{
    GeolocationServiceBridge* object = reinterpret_cast<GeolocationServiceBridge*>(nativeObject);
    RefPtr<PositionError> error =
        PositionError::create(PositionError::POSITION_UNAVAILABLE, android::to_string(env, message));
    object->m_listener->newErrorAvailable(error.release());
}

PassRefPtr<Geoposition> GeolocationServiceBridge::toGeoposition(JNIEnv *env, const jobject &location)
{
    // Altitude is optional and may not be supplied.
    bool hasAltitude =
        env->CallBooleanMethod(location, javaLocationClassMethodIDs[LocationMethodHasAltitude]);
    double Altitude =
        hasAltitude ?
        env->CallDoubleMethod(location, javaLocationClassMethodIDs[LocationMethodGetAltitude]) :
        0.0;
    // Accuracy is required, but is not supplied by the emulator.
    double Accuracy =
        env->CallBooleanMethod(location, javaLocationClassMethodIDs[LocationMethodHasAccuracy]) ?
        env->CallFloatMethod(location, javaLocationClassMethodIDs[LocationMethodGetAccuracy]) :
        0.0;
    // heading is optional and may not be supplied.
    bool hasHeading =
        env->CallBooleanMethod(location, javaLocationClassMethodIDs[LocationMethodHasBearing]);
    double heading =
        hasHeading ?
        env->CallFloatMethod(location, javaLocationClassMethodIDs[LocationMethodGetBearing]) :
        0.0;
    // speed is optional and may not be supplied.
    bool hasSpeed =
        env->CallBooleanMethod(location, javaLocationClassMethodIDs[LocationMethodHasSpeed]);
    double speed =
        hasSpeed ?
        env->CallFloatMethod(location, javaLocationClassMethodIDs[LocationMethodGetSpeed]) :
        0.0;

    RefPtr<Coordinates> newCoordinates = WebCore::Coordinates::create(
        env->CallDoubleMethod(location, javaLocationClassMethodIDs[LocationMethodGetLatitude]),
        env->CallDoubleMethod(location, javaLocationClassMethodIDs[LocationMethodGetLongitude]),
        hasAltitude, Altitude,
        Accuracy,
        false, 0.0,  // AltitudeAccuracy not provided.
        hasHeading, heading,
        hasSpeed, speed);

    return WebCore::Geoposition::create(
         newCoordinates.release(),
         env->CallLongMethod(location, javaLocationClassMethodIDs[LocationMethodGetTime]));
}

void GeolocationServiceBridge::startJavaImplementation()
{
    JNIEnv* env = getJNIEnv();

    // Get the Java GeolocationService class.
    jclass javaGeolocationServiceClass = env->FindClass(javaGeolocationServiceClassName);
    ASSERT(javaGeolocationServiceClass);

    // Set up the methods we wish to call on the Java GeolocationService class.
    javaGeolocationServiceClassMethodIDs[GeolocationServiceMethodInit] =
            env->GetMethodID(javaGeolocationServiceClass, "<init>", "(J)V");
    javaGeolocationServiceClassMethodIDs[GeolocationServiceMethodStart] =
            env->GetMethodID(javaGeolocationServiceClass, "start", "()V");
    javaGeolocationServiceClassMethodIDs[GeolocationServiceMethodStop] =
            env->GetMethodID(javaGeolocationServiceClass, "stop", "()V");
    javaGeolocationServiceClassMethodIDs[GeolocationServiceMethodSetEnableGps] =
            env->GetMethodID(javaGeolocationServiceClass, "setEnableGps", "(Z)V");

    // Create the Java GeolocationService object.
    jlong nativeObject = reinterpret_cast<jlong>(this);
    jobject object = env->NewObject(javaGeolocationServiceClass,
                                    javaGeolocationServiceClassMethodIDs[GeolocationServiceMethodInit],
                                    nativeObject);

    m_javaGeolocationServiceObject = getJNIEnv()->NewGlobalRef(object);
    ASSERT(m_javaGeolocationServiceObject);

    // Register to handle calls to native methods of the Java GeolocationService
    // object. We register once only.
    static int registered = jniRegisterNativeMethods(env,
                                                     javaGeolocationServiceClassName,
                                                     javaGeolocationServiceClassNativeMethods,
                                                     NELEM(javaGeolocationServiceClassNativeMethods));
    ASSERT(registered == JNI_OK);

    // Set up the methods we wish to call on the Java Location class.
    jclass javaLocationClass = env->FindClass(javaLocationClassName);
    ASSERT(javaLocationClass);
    javaLocationClassMethodIDs[LocationMethodGetLatitude] =
        env->GetMethodID(javaLocationClass, "getLatitude", "()D");
    javaLocationClassMethodIDs[LocationMethodGetLongitude] =
        env->GetMethodID(javaLocationClass, "getLongitude", "()D");
    javaLocationClassMethodIDs[LocationMethodHasAltitude] =
        env->GetMethodID(javaLocationClass, "hasAltitude", "()Z");
    javaLocationClassMethodIDs[LocationMethodGetAltitude] =
        env->GetMethodID(javaLocationClass, "getAltitude", "()D");
    javaLocationClassMethodIDs[LocationMethodHasAccuracy] =
        env->GetMethodID(javaLocationClass, "hasAccuracy", "()Z");
    javaLocationClassMethodIDs[LocationMethodGetAccuracy] =
        env->GetMethodID(javaLocationClass, "getAccuracy", "()F");
    javaLocationClassMethodIDs[LocationMethodHasBearing] =
        env->GetMethodID(javaLocationClass, "hasBearing", "()Z");
    javaLocationClassMethodIDs[LocationMethodGetBearing] =
        env->GetMethodID(javaLocationClass, "getBearing", "()F");
    javaLocationClassMethodIDs[LocationMethodHasSpeed] =
        env->GetMethodID(javaLocationClass, "hasSpeed", "()Z");
    javaLocationClassMethodIDs[LocationMethodGetSpeed] =
        env->GetMethodID(javaLocationClass, "getSpeed", "()F");
    javaLocationClassMethodIDs[LocationMethodGetTime] =
        env->GetMethodID(javaLocationClass, "getTime", "()J");
}

void GeolocationServiceBridge::stopJavaImplementation()
{
    // Called by GeolocationServiceAndroid on WebKit thread.
    ASSERT(m_javaGeolocationServiceObject);
    getJNIEnv()->DeleteGlobalRef(m_javaGeolocationServiceObject);
}

} // namespace WebCore