#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; /** * static Point fromLngLat(double longitude,double latitude) */ jni::jobject* operator()(const mapbox::geometry::point &geometry) const { static jni::jclass* javaClass = jni::NewGlobalRef(env, &jni::FindClass(env, "com/mapbox/geojson/Point")).release(); static jni::jmethodID* fromLngLat = &jni::GetStaticMethodID(env, *javaClass, "fromLngLat", "(DD)Lcom/mapbox/geojson/Point;"); return reinterpret_cast(jni::CallStaticMethod(env, *javaClass, *fromLngLat, geometry.x, geometry.y)); } /** * static LineString fromLngLats(double [][]) */ jni::jobject* operator()(const mapbox::geometry::line_string &geometry) const { static jni::jclass* javaClass = jni::NewGlobalRef(env, &jni::FindClass(env, "com/mapbox/geojson/LineString")).release(); static jni::jmethodID* fromLngLats = &jni::GetStaticMethodID(env, *javaClass, "fromLngLats", "([[D)Lcom/mapbox/geojson/LineString;"); // Create jni::LocalObject> lngLatsArray = jni::NewLocalObject(env, toLngLatArray(env, geometry)); return reinterpret_cast(jni::CallStaticMethod(env, *javaClass, *fromLngLats, lngLatsArray.get())); } /** * static MultiPoint fromLngLats(double [][]) */ jni::jobject* operator()(const mapbox::geometry::multi_point &geometry) const { static jni::jclass* javaClass = jni::NewGlobalRef(env, &jni::FindClass(env, "com/mapbox/geojson/MultiPoint")).release(); static jni::jmethodID* fromLngLats = &jni::GetStaticMethodID(env, *javaClass, "fromLngLats", "([[D)Lcom/mapbox/geojson/MultiPoint;"); // Create jni::LocalObject> lngLatsArray = jni::NewLocalObject(env, toLngLatArray(env, geometry)); return reinterpret_cast(jni::CallStaticMethod(env, *javaClass, *fromLngLats, lngLatsArray.get())); } /** * static Polygon fromLngLats(double [][][]) */ jni::jobject* operator()(const mapbox::geometry::polygon &geometry) const { static jni::jclass* javaClass = jni::NewGlobalRef(env, &jni::FindClass(env, "com/mapbox/geojson/Polygon")).release(); static jni::jmethodID* fromLngLats = &jni::GetStaticMethodID(env, *javaClass, "fromLngLats", "([[[D)Lcom/mapbox/geojson/Polygon;"); // Create jni::LocalObject> shape = jni::NewLocalObject(env, shapeToLngLatArray<>(env, geometry)); return reinterpret_cast(jni::CallStaticMethod(env, *javaClass, *fromLngLats, shape.get())); } /** * static MultiLineString fromLngLats(double [][][]) */ jni::jobject* operator()(const mapbox::geometry::multi_line_string &geometry) const { static jni::jclass* javaClass = jni::NewGlobalRef(env, &jni::FindClass(env, "com/mapbox/geojson/MultiLineString")).release(); static jni::jmethodID* fromLngLats = &jni::GetStaticMethodID(env, *javaClass, "fromLngLats", "([[[D)Lcom/mapbox/geojson/MultiLineString;"); // Create jni::LocalObject> shape = jni::NewLocalObject(env,shapeToLngLatArray<>(env, geometry)); return reinterpret_cast(jni::CallStaticMethod(env, *javaClass, *fromLngLats, shape.get())); } /** * MultiPolygon (double[][][][]) -> [[[D + Object array == [[[[D */ jni::jobject* operator()(const mapbox::geometry::multi_polygon &geometry) const { static jni::jclass* arrayClass = jni::NewGlobalRef(env, &jni::FindClass(env, "[[[D")).release(); jni::LocalObject> jarray = jni::NewLocalObject(env, &jni::NewObjectArray(env, geometry.size(), *arrayClass)); for(size_t i = 0; i < geometry.size(); i = i + 1) { jni::LocalObject> shape = jni::NewLocalObject(env, shapeToLngLatArray<>(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/geojson/MultiPolygon")).release(); static jni::jmethodID* fromGeometries = &jni::GetStaticMethodID(env, *javaClass, "fromLngLats", "([[[[D)Lcom/mapbox/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/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/geojson/GeometryCollection")).release(); static jni::jmethodID* fromGeometries = &jni::GetStaticMethodID(env, *javaClass, "fromGeometries", "(Ljava/util/List;)Lcom/mapbox/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 *toLngLatArray(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 -> [D + Object array == [[D (double[][]) */ static jni::jarray* toLngLatArray(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::jarray *lngLatArray = toLngLatArray(env, point.x, point.y); jni::SetObjectArrayElement(env, jarray, i, lngLatArray); } return &jarray; } /** * polygon * multi_line_string -> jarray -> [[D + Object array == [[[D (double[][][]) */ template static jni::jarray* shapeToLngLatArray(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> lngLatsArray = jni::NewLocalObject(env, toLngLatArray(env, value.at(i))); jni::SetObjectArrayElement(env, jarray, i, lngLatsArray.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}; } }; } } }