Skip to content

Instantly share code, notes, and snippets.

@christopherdro
Created January 7, 2016 22:22
Show Gist options
  • Save christopherdro/bdf685c20000b3776de1 to your computer and use it in GitHub Desktop.
Save christopherdro/bdf685c20000b3776de1 to your computer and use it in GitHub Desktop.
package com.mapbox.reactnativemapboxgl;
import android.graphics.Color;
import android.util.Log;
import android.os.StrictMode;
import android.location.Location;
import com.facebook.react.bridge.Arguments;
import com.facebook.react.bridge.ReactContext;
import com.facebook.react.bridge.ReadableArray;
import com.facebook.react.bridge.ReadableMap;
import com.facebook.react.bridge.WritableMap;
import com.facebook.react.uimanager.ReactProp;
import com.mapbox.mapboxsdk.constants.Style;
import android.graphics.RectF;
import com.mapbox.mapboxsdk.geometry.CoordinateBounds;
import com.facebook.react.uimanager.SimpleViewManager;
import com.facebook.react.uimanager.ThemedReactContext;
import com.facebook.react.uimanager.events.RCTEventEmitter;
import com.mapbox.mapboxsdk.annotations.MarkerOptions;
import com.mapbox.mapboxsdk.annotations.PolygonOptions;
import com.mapbox.mapboxsdk.annotations.PolylineOptions;
import com.mapbox.mapboxsdk.constants.MyLocationTracking;
import com.mapbox.mapboxsdk.geometry.LatLng;
import com.mapbox.mapboxsdk.views.MapView;
import com.mapbox.mapboxsdk.annotations.Sprite;
import com.mapbox.mapboxsdk.annotations.SpriteFactory;
import android.support.v4.content.ContextCompat;
import android.graphics.drawable.Drawable;
import android.graphics.BitmapFactory;
import android.graphics.Bitmap;
import java.io.InputStream;
import java.io.IOException;
import java.net.URL;
import java.net.HttpURLConnection;
import java.net.MalformedURLException;
import android.graphics.drawable.BitmapDrawable;
import javax.annotation.Nullable;
public class ReactNativeMapboxGLManager extends SimpleViewManager<MapView> {
public static final String REACT_CLASS = "RCTMapbox";
public static final String PROP_ACCESS_TOKEN = "accessToken";
public static final String PROP_ANNOTATIONS = "annotations";
public static final String PROP_CENTER_COORDINATE = "centerCoordinate";
public static final String PROP_DEBUG_ACTIVE = "debugActive";
public static final String PROP_DIRECTION = "direction";
public static final String PROP_ONREGIONCHANGE = "onRegionChange";
public static final String PROP_ONUSER_LOCATION_CHANGE = "onUserLocationChange";
public static final String PROP_ROTATION_ENABLED = "rotationEnabled";
public static final String PROP_SCROLL_ENABLED = "scrollEnabled";
public static final String PROP_USER_LOCATON = "showsUserLocation";
public static final String PROP_STYLE_URL = "styleUrl";
public static final String PROP_USER_TRACKING_MODE = "userTrackingMode";
public static final String PROP_ZOOM_ENABLED = "zoomEnabled";
public static final String PROP_ZOOM_LEVEL = "zoomLevel";
public static final String PROP_SET_TILT = "tilt";
public static final String PROP_COMPASS_IS_HIDDEN = "compassIsHidden";
private MapView mapView;
@Override
public String getName() {
return REACT_CLASS;
}
@Override
public MapView createViewInstance(ThemedReactContext context) {
mapView = new MapView(context, "pk.foo");
mapView.onCreate(null);
StrictMode.ThreadPolicy policy = new StrictMode.ThreadPolicy.Builder().permitAll().build();
StrictMode.setThreadPolicy(policy);
return mapView;
}
@ReactProp(name = PROP_ACCESS_TOKEN)
public void setAccessToken(MapView view, @Nullable String value) {
if (value == null || value.isEmpty()) {
Log.e(REACT_CLASS, "Error: No access token provided");
} else {
view.setAccessToken(value);
}
}
@ReactProp(name = PROP_SET_TILT)
public void setTilt(MapView view, @Nullable double pitch) {
mapView.setTilt(pitch, 1L);
}
public static Drawable drawableFromUrl(MapView view, String url) throws IOException {
Bitmap x;
HttpURLConnection connection = (HttpURLConnection) new URL(url).openConnection();
connection.connect();
InputStream input = connection.getInputStream();
x = BitmapFactory.decodeStream(input);
return new BitmapDrawable(view.getResources(), x);
}
public void addAnnotation(MapView view, @Nullable ReadableMap annotation) {
// view.addAnnotation(value);
switch (annotation.getString("type")) {
case "point":
view.addMarker(convertToMarker(view, annotation));
break;
case "polyline":
view.addPolyline(convertToPolyline(view, annotation));
break;
case "polygon":
view.addPolygon(convertToPolygon(view, annotation));
break;
}
}
@ReactProp(name = PROP_ANNOTATIONS)
public void setAnnotations(MapView view, @Nullable ReadableArray value) {
if (value == null || value.size() < 1) {
Log.e(REACT_CLASS, "Error: No annotations");
} else {
int size = value.size();
for (int i = 0; i < size; i++) {
ReadableMap annotation = value.getMap(i);
addAnnotation(view, annotation);
}
}
}
@ReactProp(name = PROP_DEBUG_ACTIVE, defaultBoolean = false)
public void setDebugActive(MapView view, Boolean value) {
view.setDebugActive(value);
}
@ReactProp(name = PROP_DIRECTION, defaultFloat = 0f)
public void setDirection(MapView view, float value) {
view.setDirection(value, true);
}
@ReactProp(name = PROP_ONREGIONCHANGE, defaultBoolean = true)
public void onMapChanged(final MapView view, Boolean value) {
view.addOnMapChangedListener(new MapView.OnMapChangedListener() {
@Override
public void onMapChanged(int change) {
if (change == MapView.REGION_DID_CHANGE || change == MapView.REGION_DID_CHANGE_ANIMATED) {
WritableMap event = Arguments.createMap();
WritableMap location = Arguments.createMap();
location.putDouble("latitude", view.getCenterCoordinate().getLatitude());
location.putDouble("longitude", view.getCenterCoordinate().getLongitude());
location.putDouble("zoom", view.getZoomLevel());
event.putMap("src", location);
ReactContext reactContext = (ReactContext) view.getContext();
reactContext.getJSModule(RCTEventEmitter.class).receiveEvent(view.getId(), "topChange", event);
}
}
});
}
@ReactProp(name = PROP_ONUSER_LOCATION_CHANGE, defaultBoolean = true)
public void onMyLocationChange(final MapView view, Boolean value) {
view.setOnMyLocationChangeListener(new MapView.OnMyLocationChangeListener() {
@Override
public void onMyLocationChange(@Nullable Location location) {
WritableMap event = Arguments.createMap();
WritableMap locationMap = Arguments.createMap();
locationMap.putDouble("latitude", location.getLatitude());
locationMap.putDouble("longitude", location.getLongitude());
locationMap.putDouble("accuracy", location.getAccuracy());
locationMap.putDouble("altitude", location.getAltitude());
locationMap.putDouble("bearing", location.getBearing());
locationMap.putDouble("speed", location.getSpeed());
locationMap.putString("provider", location.getProvider());
event.putMap("src", locationMap);
ReactContext reactContext = (ReactContext) view.getContext();
reactContext.getJSModule(RCTEventEmitter.class).receiveEvent(view.getId(), "topSelect", event);
}
});
}
@ReactProp(name = PROP_CENTER_COORDINATE)
public void setCenterCoordinate(MapView view, @Nullable ReadableMap center) {
if (center != null) {
double latitude = center.getDouble("latitude");
double longitude = center.getDouble("longitude");
view.setCenterCoordinate(new LatLng(latitude, longitude));
}else{
Log.w(REACT_CLASS, "No CenterCoordinate provided");
}
}
@ReactProp(name = PROP_ROTATION_ENABLED, defaultBoolean = true)
public void setRotateEnabled(MapView view, Boolean value) {
view.setRotateEnabled(value);
}
@ReactProp(name = PROP_USER_LOCATON, defaultBoolean = true)
public void setMyLocationEnabled(MapView view, Boolean value) {
view.setMyLocationEnabled(value);
}
@ReactProp(name = PROP_STYLE_URL)
public void setStyleUrl(MapView view, @Nullable String value) {
if (value != null && !value.isEmpty()) {
view.setStyleUrl(value);
}else{
Log.w(REACT_CLASS, "No StyleUrl provided");
}
}
@ReactProp(name = PROP_USER_TRACKING_MODE, defaultInt = 0)
public void setMyLocationTrackingMode(MapView view, @Nullable int mode) {
view.setMyLocationTrackingMode(mode);
}
@ReactProp(name = PROP_ZOOM_ENABLED, defaultBoolean = true)
public void setZoomEnabled(MapView view, Boolean value) {
view.setZoomEnabled(value);
}
@ReactProp(name = PROP_ZOOM_LEVEL, defaultFloat = 0f)
public void setZoomLevel(MapView view, float value) {
view.setZoomLevel(value);
}
@ReactProp(name = PROP_SCROLL_ENABLED, defaultBoolean = true)
public void setScrollEnabled(MapView view, Boolean value) {
view.setScrollEnabled(value);
}
@ReactProp(name = PROP_COMPASS_IS_HIDDEN)
public void setCompassIsHidden(MapView view, Boolean value) {
view.setCompassEnabled(!value);
}
public void setCenterCoordinateZoomLevel(MapView view, @Nullable ReadableMap center) {
if (center != null) {
double latitude = center.getDouble("latitude");
double longitude = center.getDouble("longitude");
double zoom = center.getDouble("zoom");
view.setCenterCoordinate(new LatLng(latitude, longitude, zoom), true);
}else{
Log.w(REACT_CLASS, "No CenterCoordinate provided");
}
}
public void setVisibleCoordinateBounds(MapView view, @Nullable ReadableMap info) {
final LatLng sw = new LatLng(info.getDouble("latSW"), info.getDouble("lngSW"));
final LatLng ne = new LatLng(info.getDouble("latNE"), info.getDouble("lngNE"));
view.setVisibleCoordinateBounds(new CoordinateBounds(sw, ne), new RectF((float) info.getDouble("paddingLeft"), (float) info.getDouble("paddingTop"), (float) info.getDouble("paddingRight"), (float) info.getDouble("paddingBottom")), true);
}
public void removeAllAnnotations(MapView view, @Nullable Boolean placeHolder) {
view.removeAllAnnotations();
}
public MapView getMapView() {
return mapView;
}
private MarkerOptions convertToMarker(MapView view, ReadableMap annotation) {
double latitude = annotation.getArray("coordinates").getDouble(0);
double longitude = annotation.getArray("coordinates").getDouble(1);
LatLng markerCenter = new LatLng(latitude, longitude);
MarkerOptions marker = new MarkerOptions();
marker.position(markerCenter);
if (annotation.hasKey("title")) {
String title = annotation.getString("title");
marker.title(title);
}
if (annotation.hasKey("subtitle")) {
String subtitle = annotation.getString("subtitle");
marker.snippet(subtitle);
}
if (annotation.hasKey("annotationImage")) {
ReadableMap annotationImage = annotation.getMap("annotationImage");
String annotationURL = annotationImage.getString("url");
try {
Drawable image = drawableFromUrl(mapView, annotationURL);
SpriteFactory spriteFactory = view.getSpriteFactory();
Sprite icon = spriteFactory.fromDrawable(image);
marker.icon(icon);
} catch (Exception e) {
e.printStackTrace();
}
}
return marker;
}
private PolylineOptions convertToPolyline(MapView view, ReadableMap annotation) {
int coordSize = annotation.getArray("coordinates").size();
PolylineOptions polyline = new PolylineOptions();
for (int p = 0; p < coordSize; p++) {
double latitude = annotation.getArray("coordinates").getArray(p).getDouble(0);
double longitude = annotation.getArray("coordinates").getArray(p).getDouble(1);
polyline.add(new LatLng(latitude, longitude));
}
if (annotation.hasKey("alpha")) {
double strokeAlpha = annotation.getDouble("alpha");
polyline.alpha((float) strokeAlpha);
}
if (annotation.hasKey("strokeColor")) {
int strokeColor = Color.parseColor(annotation.getString("strokeColor"));
polyline.color(strokeColor);
}
if (annotation.hasKey("strokeWidth")) {
float strokeWidth = annotation.getInt("strokeWidth");
polyline.width(strokeWidth);
}
return polyline;
}
private PolygonOptions convertToPolygon(MapView view, ReadableMap annotation) {
int coordSize = annotation.getArray("coordinates").size();
PolygonOptions polygon = new PolygonOptions();
for (int p = 0; p < coordSize; p++) {
double latitude = annotation.getArray("coordinates").getArray(p).getDouble(0);
double longitude = annotation.getArray("coordinates").getArray(p).getDouble(1);
polygon.add(new LatLng(latitude, longitude));
}
if (annotation.hasKey("alpha")) {
double fillAlpha = annotation.getDouble("alpha");
polygon.alpha((float) fillAlpha);
}
if (annotation.hasKey("fillColor")) {
int fillColor = Color.parseColor(annotation.getString("fillColor"));
polygon.fillColor(fillColor);
}
if (annotation.hasKey("strokeColor")) {
int strokeColor = Color.parseColor(annotation.getString("strokeColor"));
polygon.strokeColor(strokeColor);
}
return polygon;
}
}
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment