#pragma once #include "../../conversion/constant.hpp" #include "../../conversion/collection.hpp" #include #include #include "../../jni/local_object.hpp" namespace mbgl { namespace android { namespace conversion { /** * Turn mapbox::geometry type into Java GeoJson Geometries */ template class GeometryEvaluator { public: jni::JNIEnv& env; /** * Point (double[]) */ jni::jobject* operator()(const mapbox::geometry::point &geometry) const { static jni::jclass* javaClass = jni::NewGlobalRef(env, &jni::FindClass(env, "com/mapbox/services/commons/geojson/Point")).release(); static jni::jmethodID* fromCoordinates = &jni::GetStaticMethodID(env, *javaClass, "fromCoordinates", "([D)Lcom/mapbox/services/commons/geojson/Point;"); // Create Point jni::LocalObject> position = jni::NewLocalObject(env, toGeoJsonPosition(env, geometry.x, geometry.y)); return reinterpret_cast(jni::CallStaticMethod(env, *javaClass, *fromCoordinates, position.get())); } /** * LineString (double[][]) */ jni::jobject* operator()(const mapbox::geometry::line_string &geometry) const { static jni::jclass* javaClass = jni::NewGlobalRef(env, &jni::FindClass(env, "com/mapbox/services/commons/geojson/LineString")).release(); static jni::jmethodID* fromCoordinates = &jni::GetStaticMethodID(env, *javaClass, "fromCoordinates", "([[D)Lcom/mapbox/services/commons/geojson/LineString;"); // Create jni::LocalObject> coordinates = jni::NewLocalObject(env, toGeoJsonCoordinates(env, geometry)); return reinterpret_cast(jni::CallStaticMethod(env, *javaClass, *fromCoordinates, coordinates.get())); } /** * MultiPoint (double[][]) */ jni::jobject* operator()(const mapbox::geometry::multi_point &geometry) const { static jni::jclass* javaClass = jni::NewGlobalRef(env, &jni::FindClass(env, "com/mapbox/services/commons/geojson/MultiPoint")).release(); static jni::jmethodID* fromCoordinates = &jni::GetStaticMethodID(env, *javaClass, "fromCoordinates", "([[D)Lcom/mapbox/services/commons/geojson/MultiPoint;"); // Create jni::LocalObject> coordinates = jni::NewLocalObject(env, toGeoJsonCoordinates(env, geometry)); return reinterpret_cast(jni::CallStaticMethod(env, *javaClass, *fromCoordinates, coordinates.get())); } /** * Polygon (double[][][]) */ jni::jobject* operator()(const mapbox::geometry::polygon &geometry) const { static jni::jclass* javaClass = jni::NewGlobalRef(env, &jni::FindClass(env, "com/mapbox/services/commons/geojson/Polygon")).release(); static jni::jmethodID* fromCoordinates = &jni::GetStaticMethodID(env, *javaClass, "fromCoordinates", "([[[D)Lcom/mapbox/services/commons/geojson/Polygon;"); // Create jni::LocalObject> shape = jni::NewLocalObject(env, toShape<>(env, geometry)); return reinterpret_cast(jni::CallStaticMethod(env, *javaClass, *fromCoordinates, shape.get())); } /** * MultiLineString (double[][][]) */ jni::jobject* operator()(const mapbox::geometry::multi_line_string &geometry) const { static jni::jclass* javaClass = jni::NewGlobalRef(env, &jni::FindClass(env, "com/mapbox/services/commons/geojson/MultiLineString")).release(); static jni::jmethodID* fromCoordinates = &jni::GetStaticMethodID(env, *javaClass, "fromCoordinates", "([[[D)Lcom/mapbox/services/commons/geojson/MultiLineString;"); // Create jni::LocalObject> shape = jni::NewLocalObject(env, toShape<>(env, geometry)); return reinterpret_cast(jni::CallStaticMethod(env, *javaClass, *fromCoordinates, shape.get())); } /** * MultiPolygon (double[][][][]) -> [[[D + Object array == [[[[D */ jni::jobject* operator()(const mapbox::geometry::multi_polygon &geometry) const { static jni::jclass* listClass = jni::NewGlobalRef(env, &jni::FindClass(env, "[[[D")).release(); jni::LocalObject> jarray = jni::NewLocalObject(env, &jni::NewObjectArray(env, geometry.size(), *listClass)); for(size_t i = 0; i < geometry.size(); i = i + 1) { jni::LocalObject> shape = jni::NewLocalObject(env, toShape<>(env, geometry.at(i))); jni::SetObjectArrayElement(env, *jarray, i, shape.get()); } // Create the MultiPolygon static jni::jclass* javaClass = jni::NewGlobalRef(env, &jni::FindClass(env, "com/mapbox/services/commons/geojson/MultiPolygon")).release(); static jni::jmethodID* fromGeometries = &jni::GetStaticMethodID(env, *javaClass, "fromCoordinates", "([[[[D)Lcom/mapbox/services/commons/geojson/MultiPolygon;"); return reinterpret_cast(jni::CallStaticMethod(env, *javaClass, *fromGeometries, jarray.get())); } /** * GeometryCollection */ jni::jobject* operator()(const mapbox::geometry::geometry_collection &collection) const { static jni::jclass* geometryClass = jni::NewGlobalRef(env, &jni::FindClass(env, "com/mapbox/services/commons/geojson/Geometry")).release(); jni::LocalObject> jarray = jni::NewLocalObject(env, &jni::NewObjectArray(env, collection.size(), *geometryClass)); for(size_t i = 0; i < collection.size(); i = i + 1) { auto& geometry = collection.at(i); jni::LocalObject converted = jni::NewLocalObject(env, mapbox::geometry::geometry::visit(geometry, *this)); jni::SetObjectArrayElement(env, *jarray, i, converted.get()); } // Turn into array list and create the GeometryCollection static jni::jclass* javaClass = jni::NewGlobalRef(env, &jni::FindClass(env, "com/mapbox/services/commons/geojson/GeometryCollection")).release(); static jni::jmethodID* fromGeometries = &jni::GetStaticMethodID(env, *javaClass, "fromGeometries", "(Ljava/util/List;)Lcom/mapbox/services/commons/geojson/GeometryCollection;"); jni::LocalObject list = jni::NewLocalObject(env, toArrayList<>(env, *jarray)); return reinterpret_cast(jni::CallStaticMethod(env, *javaClass, *fromGeometries, list.get())); } private: /** * x, y -> jarray ([x,y]) */ static jni::jarray* toGeoJsonPosition(JNIEnv& env, double x, double y) { jni::jarray& jarray = jni::NewArray(env, 2); jni::jdouble array[] = {x, y}; jni::SetArrayRegion(env, jarray, 0, 2, array); return &jarray; } /** * vector> -> jarray (double[][]) -> [D + Object array == [[D */ static jni::jarray* toGeoJsonCoordinates(JNIEnv& env, std::vector> points) { static jni::jclass* javaClass = jni::NewGlobalRef(env, &jni::FindClass(env, "[D")).release(); jni::jarray& jarray = jni::NewObjectArray(env, points.size(), *javaClass); for(size_t i = 0; i < points.size(); i = i + 1) { mapbox::geometry::point point = points.at(i); jni::LocalObject> position = jni::NewLocalObject(env, toGeoJsonPosition(env, point.x, point.y)); jni::SetObjectArrayElement(env, jarray, i, position.get()); } return &jarray; } /** * polygon * multi_line_string * -> jarray (double[][][]) -> [[D + Object array == [[[D */ template static jni::jarray* toShape(JNIEnv& env, SHAPE value) { static jni::jclass* javaClass = jni::NewGlobalRef(env, &jni::FindClass(env, "[[D")).release(); jni::jarray& jarray = jni::NewObjectArray(env, value.size(), *javaClass); for(size_t i = 0; i < value.size(); i = i + 1) { jni::LocalObject> coordinates = jni::NewLocalObject(env, toGeoJsonCoordinates(env, value.at(i))); jni::SetObjectArrayElement(env, jarray, i, coordinates.get()); } return &jarray; } }; /** * mapbox::geometry::geometry -> Java GeoJson Geometry<> */ template struct Converter> { Result operator()(jni::JNIEnv& env, const mapbox::geometry::geometry& value) const { GeometryEvaluator evaluator { env } ; jni::jobject* converted = mapbox::geometry::geometry::visit(value, evaluator); return {converted}; } }; } } }