summaryrefslogtreecommitdiff
path: root/android
diff options
context:
space:
mode:
authorLeith Bade <leith@mapbox.com>2015-09-24 17:18:30 +1000
committerLeith Bade <leith@mapbox.com>2015-09-24 18:04:26 +1000
commit85ec3eb0a5739c4f7b160d82715d3be51e10a02b (patch)
tree1524ad5c8786d0ba0487f6a05f5e46d05e42f4ae /android
parente5ff4638b38c497cf37cffde30b46d9af54cc60d (diff)
downloadqtlocation-mapboxgl-85ec3eb0a5739c4f7b160d82715d3be51e10a02b.tar.gz
Refactor Android render loop to limit frame rate to 60 FPS
Move expensive cache size calulation to initialisation Fix setGestureInProgress bug Renamed several functions to match current usage Refactor map updates to reduce GPS marker lag Fixes #1676 Fixes #2290 Fixes #2396
Diffstat (limited to 'android')
-rw-r--r--android/cpp/jni.cpp8
-rw-r--r--android/cpp/native_map_view.cpp33
-rw-r--r--android/java/MapboxGLAndroidSDK/src/main/java/com/mapbox/mapboxgl/views/MapView.java137
-rw-r--r--android/java/MapboxGLAndroidSDK/src/main/java/com/mapbox/mapboxgl/views/NativeMapView.java6
4 files changed, 96 insertions, 88 deletions
diff --git a/android/cpp/jni.cpp b/android/cpp/jni.cpp
index 5ed0a730c3..dcf64eed20 100644
--- a/android/cpp/jni.cpp
+++ b/android/cpp/jni.cpp
@@ -470,11 +470,11 @@ void JNICALL nativeUpdate(JNIEnv *env, jobject obj, jlong nativeMapViewPtr) {
nativeMapView->getMap().update(mbgl::Update::Repaint);
}
-void JNICALL nativeOnInvalidate(JNIEnv *env, jobject obj, jlong nativeMapViewPtr) {
- mbgl::Log::Debug(mbgl::Event::JNI, "nativeOnInvalidate");
+void JNICALL nativeRenderSync(JNIEnv *env, jobject obj, jlong nativeMapViewPtr) {
+ mbgl::Log::Debug(mbgl::Event::JNI, "nativeRenderSync");
assert(nativeMapViewPtr != 0);
NativeMapView *nativeMapView = reinterpret_cast<NativeMapView *>(nativeMapViewPtr);
- nativeMapView->onInvalidate();
+ nativeMapView->renderSync();
}
void JNICALL nativeViewResize(JNIEnv *env, jobject obj, jlong nativeMapViewPtr, jint width, jint height) {
@@ -1625,7 +1625,7 @@ extern "C" JNIEXPORT jint JNICALL JNI_OnLoad(JavaVM *vm, void *reserved) {
{"nativePause", "(J)V", reinterpret_cast<void *>(&nativePause)},
{"nativeResume", "(J)V", reinterpret_cast<void *>(&nativeResume)},
{"nativeUpdate", "(J)V", reinterpret_cast<void *>(&nativeUpdate)},
- {"nativeOnInvalidate", "(J)V", reinterpret_cast<void *>(&nativeOnInvalidate)},
+ {"nativeRenderSync", "(J)V", reinterpret_cast<void *>(&nativeRenderSync)},
{"nativeViewResize", "(JII)V",
reinterpret_cast<void *>(static_cast<void JNICALL (
*)(JNIEnv *, jobject, jlong, jint, jint)>(&nativeViewResize))},
diff --git a/android/cpp/native_map_view.cpp b/android/cpp/native_map_view.cpp
index cc04f089e6..31fb5453a3 100644
--- a/android/cpp/native_map_view.cpp
+++ b/android/cpp/native_map_view.cpp
@@ -78,6 +78,16 @@ NativeMapView::NativeMapView(JNIEnv *env, jobject obj_, float pixelRatio_, int a
fileSource = std::make_unique<mbgl::DefaultFileSource>(fileCache.get());
map = std::make_unique<mbgl::Map>(*this, *fileSource, MapMode::Continuous);
+ float zoomFactor = map->getMaxZoom() - map->getMinZoom() + 1;
+ float cpuFactor = availableProcessors;
+ float memoryFactor = static_cast<float>(totalMemory) / 1000.0f / 1000.0f / 1000.0f;
+ float sizeFactor = (static_cast<float>(map->getWidth()) / mbgl::util::tileSize) *
+ (static_cast<float>(map->getHeight()) / mbgl::util::tileSize);
+
+ size_t cacheSize = zoomFactor * cpuFactor * memoryFactor * sizeFactor * 0.5f;
+
+ map->setSourceTileCacheSize(cacheSize);
+
map->pause();
}
@@ -148,8 +158,6 @@ void NativeMapView::deactivate() {
void NativeMapView::invalidate() {
mbgl::Log::Debug(mbgl::Event::Android, "NativeMapView::invalidate()");
- clean.clear();
-
assert(vm != nullptr);
assert(obj != nullptr);
@@ -169,7 +177,7 @@ void NativeMapView::beforeRender() {
}
void NativeMapView::afterRender() {
- mbgl::Log::Debug(mbgl::Event::Android, "NativeMapView::swap");
+ mbgl::Log::Debug(mbgl::Event::Android, "NativeMapView::afterRender");
if ((display != EGL_NO_DISPLAY) && (surface != EGL_NO_SURFACE)) {
if (!eglSwapBuffers(display, surface)) {
@@ -701,28 +709,15 @@ void NativeMapView::updateFps() {
detach_jni_thread(vm, &env, detach);
}
-void NativeMapView::onInvalidate() {
- mbgl::Log::Debug(mbgl::Event::Android, "NativeMapView::onInvalidate()");
+void NativeMapView::renderSync() {
+ mbgl::Log::Debug(mbgl::Event::Android, "NativeMapView::renderSync()");
if (map->isPaused()) {
mbgl::Log::Debug(mbgl::Event::Android, "Not rendering as map is paused");
return;
}
- const bool dirty = !clean.test_and_set();
- if (dirty) {
- float zoomFactor = map->getMaxZoom() - map->getMinZoom() + 1;
- float cpuFactor = availableProcessors;
- float memoryFactor = static_cast<float>(totalMemory) / 1000.0f / 1000.0f / 1000.0f;
- float sizeFactor = (static_cast<float>(map->getWidth()) / mbgl::util::tileSize) *
- (static_cast<float>(map->getHeight()) / mbgl::util::tileSize);
-
- size_t cacheSize = zoomFactor * cpuFactor * memoryFactor * sizeFactor * 0.5f;
-
- map->setSourceTileCacheSize(cacheSize);
-
- map->renderSync();
- }
+ map->renderSync();
}
void NativeMapView::resizeView(int w, int h) {
diff --git a/android/java/MapboxGLAndroidSDK/src/main/java/com/mapbox/mapboxgl/views/MapView.java b/android/java/MapboxGLAndroidSDK/src/main/java/com/mapbox/mapboxgl/views/MapView.java
index b4eece5ec8..70fb5e1383 100644
--- a/android/java/MapboxGLAndroidSDK/src/main/java/com/mapbox/mapboxgl/views/MapView.java
+++ b/android/java/MapboxGLAndroidSDK/src/main/java/com/mapbox/mapboxgl/views/MapView.java
@@ -31,6 +31,7 @@ import android.support.v4.view.GestureDetectorCompat;
import android.support.v4.view.ScaleGestureDetectorCompat;
import android.support.v7.app.AlertDialog;
import android.text.TextUtils;
+import android.util.Log;
import android.view.GestureDetector;
import android.view.Gravity;
import android.view.ScaleGestureDetector;
@@ -63,7 +64,7 @@ import com.mapzen.android.lost.api.LocationListener;
import com.mapzen.android.lost.api.LocationRequest;
import com.mapzen.android.lost.api.LocationServices;
import com.mapzen.android.lost.api.LostApiClient;
-import com.squareup.okhttp.HttpUrl;
+
import java.nio.ByteBuffer;
import java.util.ArrayList;
import java.util.Arrays;
@@ -138,6 +139,9 @@ public class MapView extends FrameLayout implements LocationListener {
// Used to call JNI NativeMapView
private NativeMapView mNativeMapView;
+ // Used to track rendering
+ private Boolean mDirty = false;
+
// Used to handle DPI scaling
private float mScreenDensity = 1.0f;
@@ -400,6 +404,12 @@ public class MapView extends FrameLayout implements LocationListener {
.setSmallestDisplacement(1)
.setPriority(LocationRequest.PRIORITY_HIGH_ACCURACY);
+ // Setup User Location UI
+ mGpsMarker = new ImageView(getContext());
+ mGpsMarker.setImageResource(R.drawable.location_marker);
+ mGpsMarker.setVisibility(View.INVISIBLE);
+ addView(mGpsMarker);
+
// Setup Compass
mSensorManager = (SensorManager) mContext.getSystemService(Context.SENSOR_SERVICE);
mSensorAccelerometer = mSensorManager.getDefaultSensor(Sensor.TYPE_ACCELEROMETER);
@@ -411,6 +421,7 @@ public class MapView extends FrameLayout implements LocationListener {
mCompassView.setContentDescription(getResources().getString(R.string.compassContentDescription));
LayoutParams lp = new FrameLayout.LayoutParams((int)(48 * mScreenDensity), (int)(48 * mScreenDensity));
mCompassView.setLayoutParams(lp);
+ mCompassView.setVisibility(View.INVISIBLE);
addView(mCompassView);
mCompassView.setOnClickListener(new CompassOnClickListener());
@@ -883,7 +894,9 @@ public class MapView extends FrameLayout implements LocationListener {
addOnMapChangedListener(new OnMapChangedListener() {
@Override
public void onMapChanged(MapChange change) {
- updateMap(change);
+ if (change.equals(MapChange.MapChangeRegionWillChange) || change.equals(MapChange.MapChangeRegionWillChangeAnimated)) {
+ deselectAnnotation();
+ }
}
});
}
@@ -957,7 +970,7 @@ public class MapView extends FrameLayout implements LocationListener {
if (mIsMyLocationEnabled) {
toggleGps(false);
- };
+ }
mNativeMapView.pause();
}
@@ -1074,7 +1087,8 @@ public class MapView extends FrameLayout implements LocationListener {
@Override
public boolean onTouchEvent(@NonNull MotionEvent event) {
// Check and ignore non touch or left clicks
- if ((android.os.Build.VERSION.SDK_INT >= Build.VERSION_CODES.ICE_CREAM_SANDWICH) && (event.getButtonState() != 0) && (event.getButtonState() != MotionEvent.BUTTON_PRIMARY)) {
+
+ if ((event.getButtonState() != 0) && (event.getButtonState() != MotionEvent.BUTTON_PRIMARY)) {
return false;
}
@@ -1103,7 +1117,6 @@ public class MapView extends FrameLayout implements LocationListener {
long tapInterval = event.getEventTime() - event.getDownTime();
boolean isTap = tapInterval <= ViewConfiguration.getTapTimeout();
boolean inProgress = mRotateGestureDetector.isInProgress() || mScaleGestureDetector.isInProgress();
- mNativeMapView.setGestureInProgress(false);
if (mTwoTap && isTap && !inProgress) {
PointF focalPoint = TwoFingerGestureDetector.determineFocalPoint(event);
@@ -1113,6 +1126,7 @@ public class MapView extends FrameLayout implements LocationListener {
}
mTwoTap = false;
+ mNativeMapView.setGestureInProgress(false);
break;
case MotionEvent.ACTION_CANCEL:
@@ -1780,12 +1794,30 @@ public class MapView extends FrameLayout implements LocationListener {
// Called when the map needs to be rerendered
// Called via JNI from NativeMapView
protected void onInvalidate() {
- post(new Runnable() {
+ synchronized (mDirty) {
+ if (!mDirty) {
+ mDirty = true;
+ postRender();
+ }
+ }
+ }
+
+ private void postRender() {
+ Runnable mRunnable = new Runnable() {
@Override
public void run() {
- mNativeMapView.invalidate();
+ updateCompass();
+ updateGpsMarker();
+ mNativeMapView.renderSync();
+ mDirty = false;
}
- });
+ };
+
+ if (android.os.Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN) {
+ postOnAnimation(mRunnable);
+ } else {
+ postDelayed(mRunnable, 1000 / 60);
+ }
}
/**
@@ -1876,7 +1908,6 @@ public class MapView extends FrameLayout implements LocationListener {
public final void setMyLocationEnabled (boolean enabled) {
mIsMyLocationEnabled = enabled;
toggleGps(enabled);
- updateMap(MapChange.MapChangeNullChange);
}
/**
@@ -1898,18 +1929,20 @@ public class MapView extends FrameLayout implements LocationListener {
mLocationClient.connect();
updateLocation(LocationServices.FusedLocationApi.getLastLocation());
LocationServices.FusedLocationApi.requestLocationUpdates(mLocationRequest, this);
- mSensorManager.registerListener(mCompassListener, mSensorAccelerometer, SensorManager.SENSOR_DELAY_UI);
- mSensorManager.registerListener(mCompassListener, mSensorMagneticField, SensorManager.SENSOR_DELAY_UI);
+ //mSensorManager.registerListener(mCompassListener, mSensorAccelerometer, SensorManager.SENSOR_DELAY_UI);
+ //mSensorManager.registerListener(mCompassListener, mSensorMagneticField, SensorManager.SENSOR_DELAY_UI);
}
} else {
if (mLocationClient.isConnected()) {
LocationServices.FusedLocationApi.removeLocationUpdates(this);
mLocationClient.disconnect();
mGpsLocation = null;
- mSensorManager.unregisterListener(mCompassListener, mSensorAccelerometer);
- mSensorManager.unregisterListener(mCompassListener, mSensorMagneticField);
+ //mSensorManager.unregisterListener(mCompassListener, mSensorAccelerometer);
+ //mSensorManager.unregisterListener(mCompassListener, mSensorMagneticField);
}
}
+
+ onInvalidate();
}
/**
@@ -1931,18 +1964,8 @@ public class MapView extends FrameLayout implements LocationListener {
* @param compassEnabled true to enable the compass; false to disable the compass.
*/
public void setCompassEnabled (boolean compassEnabled) {
- // Set value
this.mIsCompassEnabled = compassEnabled;
-
- // Toggle UI
- if (mIsCompassEnabled) {
- mCompassView.setVisibility(View.VISIBLE);
- } else {
- mCompassView.setVisibility(View.GONE);
- }
-
- // Update Map
- updateMap(MapChange.MapChangeNullChange);
+ onInvalidate();
}
public void setCompassGravity(int gravity){
@@ -2022,7 +2045,7 @@ public class MapView extends FrameLayout implements LocationListener {
}
}
- updateMap(MapChange.MapChangeNullChange);
+ onInvalidate();
}
@Override
@@ -2088,7 +2111,7 @@ public class MapView extends FrameLayout implements LocationListener {
private void updateLocation(Location location) {
if (location != null) {
mGpsLocation = location;
- updateMap(MapChange.MapChangeNullChange);
+ updateGpsMarker();
}
}
@@ -2102,22 +2125,17 @@ public class MapView extends FrameLayout implements LocationListener {
}
// Updates the UI to match the current map's position
- private void updateMap(MapChange change) {
- // Using direct access to mIsCompassEnabled instead of isCompassEnabled() for
- // small performance boost as this method is called rapidly.
- if (mIsCompassEnabled) {
+ private void updateCompass() {
+ if (isCompassEnabled()) {
+ mCompassView.setVisibility(VISIBLE);
rotateImageView(mCompassView, (float) getDirection());
+ } else {
+ mCompassView.setVisibility(INVISIBLE);
}
+ }
+ private void updateGpsMarker() {
if (isMyLocationEnabled() && mGpsLocation != null) {
- if (mGpsMarker == null) {
- // Setup User Location UI
- // NOTE: mIsMyLocationEnabled == false to begin with
- mGpsMarker = new ImageView(getContext());
- mGpsMarker.setImageResource(R.drawable.location_marker);
- addView(mGpsMarker);
- }
-
mGpsMarker.setVisibility(View.VISIBLE);
LatLng coordinate = new LatLng(mGpsLocation);
PointF screenLocation = toScreenLocation(coordinate);
@@ -2130,26 +2148,6 @@ public class MapView extends FrameLayout implements LocationListener {
mGpsMarker.setLayoutParams(lp);
rotateImageView(mGpsMarker, 0.0f);
mGpsMarker.requestLayout();
-
- // Update direction if tracking mode
- if(mUserLocationTrackingMode == UserLocationTrackingMode.FOLLOW_BEARING && mCompassValid){
- // TODO need to do proper filtering (see branch filter-compass) or else map will lock up because of all the compass events
- long t = new Date().getTime();
- if((t-t0)>1000){
- t0 = t;
- setDirection(-mCompassBearing, true);
- }
- }
-
-/*
- // TODO - Too much overhead on main thread. Needs to be refactored before it
- // can be re-enabled
- // Update map position if NOT in NONE mode
- if (mUserLocationTrackingMode != UserLocationTrackingMode.NONE) {
- setCenterCoordinate(new LatLng(mGpsLocation));
- }
-*/
-
/*
// Used For User Location Bearing UI
if (mGpsLocation.hasBearing() || mCompassValid) {
@@ -2169,12 +2167,27 @@ public class MapView extends FrameLayout implements LocationListener {
mGpsMarker.setVisibility(View.INVISIBLE);
}
}
+ }
- if (change.equals(MapChange.MapChangeRegionWillChange) || change.equals(MapChange.MapChangeRegionWillChangeAnimated)) {
- deselectAnnotation();
- }
+ // Old tracking code
+ // Update direction if tracking mode
+ /*if(mUserLocationTrackingMode == UserLocationTrackingMode.FOLLOW_BEARING && mCompassValid){
+ // TODO need to do proper filtering (see branch filter-compass) or else map will lock up because of all the compass events
+ long t = new Date().getTime();
+ if((t-t0)>1000){
+ t0 = t;
+ setDirection(-mCompassBearing, true);
+ }
+ }*/
- }
+/*
+ // TODO - Too much overhead on main thread. Needs to be refactored before it
+ // can be re-enabled
+ // Update map position if NOT in NONE mode
+ if (mUserLocationTrackingMode != UserLocationTrackingMode.NONE) {
+ setCenterCoordinate(new LatLng(mGpsLocation));
+ }
+*/
private void selectAnnotation(Annotation annotation) {
diff --git a/android/java/MapboxGLAndroidSDK/src/main/java/com/mapbox/mapboxgl/views/NativeMapView.java b/android/java/MapboxGLAndroidSDK/src/main/java/com/mapbox/mapboxgl/views/NativeMapView.java
index 3cf56a8dea..1023fa1ff6 100644
--- a/android/java/MapboxGLAndroidSDK/src/main/java/com/mapbox/mapboxgl/views/NativeMapView.java
+++ b/android/java/MapboxGLAndroidSDK/src/main/java/com/mapbox/mapboxgl/views/NativeMapView.java
@@ -96,8 +96,8 @@ class NativeMapView {
nativeUpdate(mNativeMapViewPtr);
}
- public void invalidate() {
- nativeOnInvalidate(mNativeMapViewPtr);
+ public void renderSync() {
+ nativeRenderSync(mNativeMapViewPtr);
}
public void resizeView(int width, int height) {
@@ -475,7 +475,7 @@ class NativeMapView {
private native void nativeUpdate(long nativeMapViewPtr);
- private native void nativeOnInvalidate(long nativeMapViewPtr);
+ private native void nativeRenderSync(long nativeMapViewPtr);
private native void nativeViewResize(long nativeMapViewPtr, int width, int height);