Merge branch 'master' into dependency-injection
|  | @ -83,7 +83,6 @@ android { | ||||||
| 
 | 
 | ||||||
|     defaultConfig { |     defaultConfig { | ||||||
|         applicationId 'fr.free.nrw.commons' |         applicationId 'fr.free.nrw.commons' | ||||||
| 
 |  | ||||||
|         versionCode 76 |         versionCode 76 | ||||||
|         versionName '2.6.1' |         versionName '2.6.1' | ||||||
|         setProperty("archivesBaseName", "app-commons-v$versionName-" + getBranchName()) |         setProperty("archivesBaseName", "app-commons-v$versionName-" + getBranchName()) | ||||||
|  | @ -104,6 +103,7 @@ android { | ||||||
|             proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.txt' |             proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.txt' | ||||||
|         } |         } | ||||||
|         debug { |         debug { | ||||||
|  |             applicationIdSuffix ".debug" | ||||||
|             testCoverageEnabled true |             testCoverageEnabled true | ||||||
|             versionNameSuffix "-debug-" + getBranchName() + "~" + getBuildVersion() |             versionNameSuffix "-debug-" + getBranchName() + "~" + getBuildVersion() | ||||||
|         } |         } | ||||||
|  |  | ||||||
|  | @ -16,6 +16,9 @@ | ||||||
|     <uses-permission android:name="com.google.android.apps.photos.permission.GOOGLE_PHOTOS" /> |     <uses-permission android:name="com.google.android.apps.photos.permission.GOOGLE_PHOTOS" /> | ||||||
|     <uses-permission android:name="android.permission.READ_LOGS"/> |     <uses-permission android:name="android.permission.READ_LOGS"/> | ||||||
| 
 | 
 | ||||||
|  |     <!-- Needed only if your app targets Android 5.0 (API level 21) or higher. --> | ||||||
|  |     <uses-feature android:name="android.hardware.location.gps" /> | ||||||
|  | 
 | ||||||
|     <application |     <application | ||||||
|         android:name=".CommonsApplication" |         android:name=".CommonsApplication" | ||||||
|         android:icon="@drawable/ic_launcher" |         android:icon="@drawable/ic_launcher" | ||||||
|  |  | ||||||
|  | @ -1,5 +1,8 @@ | ||||||
| package fr.free.nrw.commons.location; | package fr.free.nrw.commons.location; | ||||||
| 
 | 
 | ||||||
|  | import android.location.Location; | ||||||
|  | import android.support.annotation.NonNull; | ||||||
|  | 
 | ||||||
| public class LatLng { | public class LatLng { | ||||||
| 
 | 
 | ||||||
|     private final double latitude; |     private final double latitude; | ||||||
|  | @ -22,6 +25,10 @@ public class LatLng { | ||||||
|         this.accuracy = accuracy; |         this.accuracy = accuracy; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     public static LatLng from(@NonNull Location location) { | ||||||
|  |         return new LatLng(location.getLatitude(), location.getLongitude(), location.getAccuracy()); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     public int hashCode() { |     public int hashCode() { | ||||||
|         boolean var1 = true; |         boolean var1 = true; | ||||||
|         byte var2 = 1; |         byte var2 = 1; | ||||||
|  |  | ||||||
|  | @ -1,11 +1,15 @@ | ||||||
| package fr.free.nrw.commons.location; | package fr.free.nrw.commons.location; | ||||||
| 
 | 
 | ||||||
|  | import android.Manifest; | ||||||
|  | import android.app.Activity; | ||||||
| import android.content.Context; | import android.content.Context; | ||||||
| import android.location.Criteria; | import android.content.pm.PackageManager; | ||||||
| import android.location.Location; | import android.location.Location; | ||||||
| import android.location.LocationListener; | import android.location.LocationListener; | ||||||
| import android.location.LocationManager; | import android.location.LocationManager; | ||||||
| import android.os.Bundle; | import android.os.Bundle; | ||||||
|  | import android.support.v4.app.ActivityCompat; | ||||||
|  | import android.support.v4.content.ContextCompat; | ||||||
| 
 | 
 | ||||||
| import java.util.List; | import java.util.List; | ||||||
| import java.util.concurrent.CopyOnWriteArrayList; | import java.util.concurrent.CopyOnWriteArrayList; | ||||||
|  | @ -16,58 +20,135 @@ import javax.inject.Singleton; | ||||||
| import timber.log.Timber; | import timber.log.Timber; | ||||||
| 
 | 
 | ||||||
| public class LocationServiceManager implements LocationListener { | public class LocationServiceManager implements LocationListener { | ||||||
|  |     public static final int LOCATION_REQUEST = 1; | ||||||
| 
 | 
 | ||||||
|     private String provider; |     private static final long MIN_LOCATION_UPDATE_REQUEST_TIME_IN_MILLIS = 2 * 60 * 1000; | ||||||
|  |     private static final long MIN_LOCATION_UPDATE_REQUEST_DISTANCE_IN_METERS = 10; | ||||||
|  | 
 | ||||||
|  |     private Context context; | ||||||
|     private LocationManager locationManager; |     private LocationManager locationManager; | ||||||
|     private LatLng lastLocation; |     private Location lastLocation; | ||||||
|     private Float latestLocationAccuracy; |  | ||||||
|     private final List<LocationUpdateListener> locationListeners = new CopyOnWriteArrayList<>(); |     private final List<LocationUpdateListener> locationListeners = new CopyOnWriteArrayList<>(); | ||||||
|  |     private boolean isLocationManagerRegistered = false; | ||||||
| 
 | 
 | ||||||
|     public LocationServiceManager(Context context) { |     public LocationServiceManager(Context context) { | ||||||
|  |         this.context = context; | ||||||
|         this.locationManager = (LocationManager) context.getSystemService(Context.LOCATION_SERVICE); |         this.locationManager = (LocationManager) context.getSystemService(Context.LOCATION_SERVICE); | ||||||
|         provider = locationManager.getBestProvider(new Criteria(), true); |  | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public boolean isProviderEnabled() { |     public boolean isProviderEnabled() { | ||||||
|         return locationManager.isProviderEnabled(LocationManager.GPS_PROVIDER); |         return locationManager.isProviderEnabled(LocationManager.GPS_PROVIDER); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     public LatLng getLastLocation() { |     public boolean isLocationPermissionGranted() { | ||||||
|         return lastLocation; |         return ContextCompat.checkSelfPermission(context, | ||||||
|  |                 Manifest.permission.ACCESS_FINE_LOCATION) == PackageManager.PERMISSION_GRANTED; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     /** |     public void requestPermissions(Activity activity) { | ||||||
|      * Returns the accuracy of the location. The measurement is |         if (activity.isFinishing()) { | ||||||
|      * given as a radius in meter of 68 % confidence. |             return; | ||||||
|      * |         } | ||||||
|      * @return Float |         ActivityCompat.requestPermissions(activity, | ||||||
|      */ |                 new String[]{Manifest.permission.ACCESS_FINE_LOCATION}, | ||||||
|     public Float getLatestLocationAccuracy() { |                 LOCATION_REQUEST); | ||||||
|         return latestLocationAccuracy; |     } | ||||||
|  | 
 | ||||||
|  |     public boolean isPermissionExplanationRequired(Activity activity) { | ||||||
|  |         if (activity.isFinishing()) { | ||||||
|  |             return false; | ||||||
|  |         } | ||||||
|  |         return ActivityCompat.shouldShowRequestPermissionRationale(activity, | ||||||
|  |                 Manifest.permission.ACCESS_FINE_LOCATION); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     public LatLng getLastLocation() { | ||||||
|  |         if (lastLocation == null) { | ||||||
|  |             return null; | ||||||
|  |         } | ||||||
|  |         return LatLng.from(lastLocation); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     /** Registers a LocationManager to listen for current location. |     /** Registers a LocationManager to listen for current location. | ||||||
|      */ |      */ | ||||||
|     public void registerLocationManager() { |     public void registerLocationManager() { | ||||||
|  |         if (!isLocationManagerRegistered) | ||||||
|  |             isLocationManagerRegistered = requestLocationUpdatesFromProvider(LocationManager.NETWORK_PROVIDER) | ||||||
|  |                     && requestLocationUpdatesFromProvider(LocationManager.GPS_PROVIDER); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private boolean requestLocationUpdatesFromProvider(String locationProvider) { | ||||||
|         try { |         try { | ||||||
|             locationManager.requestLocationUpdates(provider, 400, 1, this); |             locationManager.requestLocationUpdates(locationProvider, | ||||||
|             Location location = locationManager.getLastKnownLocation(provider); |                     MIN_LOCATION_UPDATE_REQUEST_TIME_IN_MILLIS, | ||||||
|             //Location works, just need to 'send' GPS coords |                     MIN_LOCATION_UPDATE_REQUEST_DISTANCE_IN_METERS, | ||||||
|             // via emulator extended controls if testing on emulator |                     this); | ||||||
|             Timber.d("Checking for location..."); |             return true; | ||||||
|             if (location != null) { |  | ||||||
|                 this.onLocationChanged(location); |  | ||||||
|             } |  | ||||||
|         } catch (IllegalArgumentException e) { |         } catch (IllegalArgumentException e) { | ||||||
|             Timber.e(e, "Illegal argument exception"); |             Timber.e(e, "Illegal argument exception"); | ||||||
|  |             return false; | ||||||
|         } catch (SecurityException e) { |         } catch (SecurityException e) { | ||||||
|             Timber.e(e, "Security exception"); |             Timber.e(e, "Security exception"); | ||||||
|  |             return false; | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     protected boolean isBetterLocation(Location location, Location currentBestLocation) { | ||||||
|  |         if (currentBestLocation == null) { | ||||||
|  |             // A new location is always better than no location | ||||||
|  |             return true; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         // Check whether the new location fix is newer or older | ||||||
|  |         long timeDelta = location.getTime() - currentBestLocation.getTime(); | ||||||
|  |         boolean isSignificantlyNewer = timeDelta > MIN_LOCATION_UPDATE_REQUEST_TIME_IN_MILLIS; | ||||||
|  |         boolean isSignificantlyOlder = timeDelta < -MIN_LOCATION_UPDATE_REQUEST_TIME_IN_MILLIS; | ||||||
|  |         boolean isNewer = timeDelta > 0; | ||||||
|  | 
 | ||||||
|  |         // If it's been more than two minutes since the current location, use the new location | ||||||
|  |         // because the user has likely moved | ||||||
|  |         if (isSignificantlyNewer) { | ||||||
|  |             return true; | ||||||
|  |             // If the new location is more than two minutes older, it must be worse | ||||||
|  |         } else if (isSignificantlyOlder) { | ||||||
|  |             return false; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         // Check whether the new location fix is more or less accurate | ||||||
|  |         int accuracyDelta = (int) (location.getAccuracy() - currentBestLocation.getAccuracy()); | ||||||
|  |         boolean isLessAccurate = accuracyDelta > 0; | ||||||
|  |         boolean isMoreAccurate = accuracyDelta < 0; | ||||||
|  |         boolean isSignificantlyLessAccurate = accuracyDelta > 200; | ||||||
|  | 
 | ||||||
|  |         // Check if the old and new location are from the same provider | ||||||
|  |         boolean isFromSameProvider = isSameProvider(location.getProvider(), | ||||||
|  |                 currentBestLocation.getProvider()); | ||||||
|  | 
 | ||||||
|  |         // Determine location quality using a combination of timeliness and accuracy | ||||||
|  |         if (isMoreAccurate) { | ||||||
|  |             return true; | ||||||
|  |         } else if (isNewer && !isLessAccurate) { | ||||||
|  |             return true; | ||||||
|  |         } else if (isNewer && !isSignificantlyLessAccurate && isFromSameProvider) { | ||||||
|  |             return true; | ||||||
|  |         } | ||||||
|  |         return false; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /** | ||||||
|  |      * Checks whether two providers are the same | ||||||
|  |      */ | ||||||
|  |     private boolean isSameProvider(String provider1, String provider2) { | ||||||
|  |         if (provider1 == null) { | ||||||
|  |             return provider2 == null; | ||||||
|  |         } | ||||||
|  |         return provider1.equals(provider2); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     /** Unregisters location manager. |     /** Unregisters location manager. | ||||||
|      */ |      */ | ||||||
|     public void unregisterLocationManager() { |     public void unregisterLocationManager() { | ||||||
|  |         isLocationManagerRegistered = false; | ||||||
|         try { |         try { | ||||||
|             locationManager.removeUpdates(this); |             locationManager.removeUpdates(this); | ||||||
|         } catch (SecurityException e) { |         } catch (SecurityException e) { | ||||||
|  | @ -87,15 +168,11 @@ public class LocationServiceManager implements LocationListener { | ||||||
| 
 | 
 | ||||||
|     @Override |     @Override | ||||||
|     public void onLocationChanged(Location location) { |     public void onLocationChanged(Location location) { | ||||||
|         double currentLatitude = location.getLatitude(); |         if (isBetterLocation(location, lastLocation)) { | ||||||
|         double currentLongitude = location.getLongitude(); |             lastLocation = location; | ||||||
|         latestLocationAccuracy = location.getAccuracy(); |             for (LocationUpdateListener listener : locationListeners) { | ||||||
|         Timber.d("Latitude: %f Longitude: %f Accuracy %f", |                 listener.onLocationChanged(LatLng.from(lastLocation)); | ||||||
|                 currentLatitude, currentLongitude, latestLocationAccuracy); |             } | ||||||
|         lastLocation = new LatLng(currentLatitude, currentLongitude, latestLocationAccuracy); |  | ||||||
| 
 |  | ||||||
|         for (LocationUpdateListener listener : locationListeners) { |  | ||||||
|             listener.onLocationChanged(lastLocation); |  | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -1,6 +1,5 @@ | ||||||
| package fr.free.nrw.commons.nearby; | package fr.free.nrw.commons.nearby; | ||||||
| 
 | 
 | ||||||
| import android.Manifest; |  | ||||||
| import android.content.Context; | import android.content.Context; | ||||||
| import android.content.Intent; | import android.content.Intent; | ||||||
| import android.content.SharedPreferences; | import android.content.SharedPreferences; | ||||||
|  | @ -10,10 +9,8 @@ import android.os.Build; | ||||||
| import android.os.Bundle; | import android.os.Bundle; | ||||||
| import android.preference.PreferenceManager; | import android.preference.PreferenceManager; | ||||||
| import android.support.annotation.NonNull; | import android.support.annotation.NonNull; | ||||||
| import android.support.v4.app.ActivityCompat; |  | ||||||
| import android.support.v4.app.Fragment; | import android.support.v4.app.Fragment; | ||||||
| import android.support.v4.app.FragmentTransaction; | import android.support.v4.app.FragmentTransaction; | ||||||
| import android.support.v4.content.ContextCompat; |  | ||||||
| import android.support.v7.app.AlertDialog; | import android.support.v7.app.AlertDialog; | ||||||
| import android.view.Menu; | import android.view.Menu; | ||||||
| import android.view.MenuInflater; | import android.view.MenuInflater; | ||||||
|  | @ -44,6 +41,8 @@ import io.reactivex.disposables.Disposable; | ||||||
| import io.reactivex.schedulers.Schedulers; | import io.reactivex.schedulers.Schedulers; | ||||||
| import timber.log.Timber; | import timber.log.Timber; | ||||||
| 
 | 
 | ||||||
|  | import static fr.free.nrw.commons.location.LocationServiceManager.LOCATION_REQUEST; | ||||||
|  | 
 | ||||||
| 
 | 
 | ||||||
| public class NearbyActivity extends NavigationBaseActivity implements LocationUpdateListener { | public class NearbyActivity extends NavigationBaseActivity implements LocationUpdateListener { | ||||||
| 
 | 
 | ||||||
|  | @ -71,7 +70,6 @@ public class NearbyActivity extends NavigationBaseActivity implements LocationUp | ||||||
|         sharedPreferences = PreferenceManager.getDefaultSharedPreferences(getApplicationContext()); |         sharedPreferences = PreferenceManager.getDefaultSharedPreferences(getApplicationContext()); | ||||||
|         setContentView(R.layout.activity_nearby); |         setContentView(R.layout.activity_nearby); | ||||||
|         ButterKnife.bind(this); |         ButterKnife.bind(this); | ||||||
|         checkLocationPermission(); |  | ||||||
|         bundle = new Bundle(); |         bundle = new Bundle(); | ||||||
|         initDrawer(); |         initDrawer(); | ||||||
|         initViewState(); |         initViewState(); | ||||||
|  | @ -103,7 +101,7 @@ public class NearbyActivity extends NavigationBaseActivity implements LocationUp | ||||||
|         // Handle item selection |         // Handle item selection | ||||||
|         switch (item.getItemId()) { |         switch (item.getItemId()) { | ||||||
|             case R.id.action_refresh: |             case R.id.action_refresh: | ||||||
|                 lockNearbyView = false; |                 lockNearbyView(false); | ||||||
|                 refreshView(true); |                 refreshView(true); | ||||||
|                 return true; |                 return true; | ||||||
|             case R.id.action_toggle_view: |             case R.id.action_toggle_view: | ||||||
|  | @ -116,52 +114,9 @@ public class NearbyActivity extends NavigationBaseActivity implements LocationUp | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     private void checkLocationPermission() { |     private void requestLocationPermissions() { | ||||||
|         if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) { |         if (!isFinishing()) { | ||||||
|             if (ContextCompat.checkSelfPermission(this, |             locationManager.requestPermissions(this); | ||||||
|                     Manifest.permission.ACCESS_FINE_LOCATION) == PackageManager.PERMISSION_GRANTED) { |  | ||||||
|                 refreshView(false); |  | ||||||
|             } else { |  | ||||||
|                 if (ContextCompat.checkSelfPermission(this, |  | ||||||
|                         Manifest.permission.ACCESS_FINE_LOCATION) |  | ||||||
|                         != PackageManager.PERMISSION_GRANTED) { |  | ||||||
| 
 |  | ||||||
|                     // Should we show an explanation? |  | ||||||
|                     if (ActivityCompat.shouldShowRequestPermissionRationale(this, |  | ||||||
|                             Manifest.permission.ACCESS_FINE_LOCATION)) { |  | ||||||
| 
 |  | ||||||
|                         // Show an explanation to the user *asynchronously* -- don't block |  | ||||||
|                         // this thread waiting for the user's response! After the user |  | ||||||
|                         // sees the explanation, try again to request the permission. |  | ||||||
| 
 |  | ||||||
|                         new AlertDialog.Builder(this) |  | ||||||
|                                 .setMessage(getString(R.string.location_permission_rationale)) |  | ||||||
|                                 .setPositiveButton("OK", (dialog, which) -> { |  | ||||||
|                                     ActivityCompat.requestPermissions(NearbyActivity.this, |  | ||||||
|                                             new String[]{Manifest.permission.ACCESS_FINE_LOCATION}, |  | ||||||
|                                             LOCATION_REQUEST); |  | ||||||
|                                     dialog.dismiss(); |  | ||||||
|                                 }) |  | ||||||
|                                 .setNegativeButton("Cancel", null) |  | ||||||
|                                 .create() |  | ||||||
|                                 .show(); |  | ||||||
| 
 |  | ||||||
|                     } else { |  | ||||||
| 
 |  | ||||||
|                         // No explanation needed, we can request the permission. |  | ||||||
| 
 |  | ||||||
|                         ActivityCompat.requestPermissions(this, |  | ||||||
|                                 new String[]{Manifest.permission.ACCESS_FINE_LOCATION}, |  | ||||||
|                                 LOCATION_REQUEST); |  | ||||||
| 
 |  | ||||||
|                         // MY_PERMISSIONS_REQUEST_READ_CONTACTS is an |  | ||||||
|                         // app-defined int constant. The callback method gets the |  | ||||||
|                         // result of the request. |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         } else { |  | ||||||
|             refreshView(false); |  | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  | @ -186,7 +141,7 @@ public class NearbyActivity extends NavigationBaseActivity implements LocationUp | ||||||
|                 .setCancelable(false) |                 .setCancelable(false) | ||||||
|                 .setPositiveButton(R.string.give_permission, (dialog, which) -> { |                 .setPositiveButton(R.string.give_permission, (dialog, which) -> { | ||||||
|                     //will ask for the location permission again |                     //will ask for the location permission again | ||||||
|                     checkLocationPermission(); |                     checkGps(); | ||||||
|                 }) |                 }) | ||||||
|                 .setNegativeButton(R.string.cancel, (dialog, which) -> { |                 .setNegativeButton(R.string.cancel, (dialog, which) -> { | ||||||
|                     //dismiss dialog and finish activity |                     //dismiss dialog and finish activity | ||||||
|  | @ -210,11 +165,48 @@ public class NearbyActivity extends NavigationBaseActivity implements LocationUp | ||||||
|                                 Timber.d("Loaded settings page"); |                                 Timber.d("Loaded settings page"); | ||||||
|                                 startActivityForResult(callGPSSettingIntent, 1); |                                 startActivityForResult(callGPSSettingIntent, 1); | ||||||
|                             }) |                             }) | ||||||
|                     .setNegativeButton(R.string.menu_cancel_upload, (dialog, id) -> dialog.cancel()) |                     .setNegativeButton(R.string.menu_cancel_upload, (dialog, id) -> { | ||||||
|  |                         showLocationPermissionDeniedErrorDialog(); | ||||||
|  |                         dialog.cancel(); | ||||||
|  |                     }) | ||||||
|                     .create() |                     .create() | ||||||
|                     .show(); |                     .show(); | ||||||
|         } else { |         } else { | ||||||
|             Timber.d("GPS is enabled"); |             Timber.d("GPS is enabled"); | ||||||
|  |             checkLocationPermission(); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     private void checkLocationPermission() { | ||||||
|  |         if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) { | ||||||
|  |             if (locationManager.isLocationPermissionGranted()) { | ||||||
|  |                 refreshView(false); | ||||||
|  |             } else { | ||||||
|  |                 // Should we show an explanation? | ||||||
|  |                 if (locationManager.isPermissionExplanationRequired(this)) { | ||||||
|  |                     // Show an explanation to the user *asynchronously* -- don't block | ||||||
|  |                     // this thread waiting for the user's response! After the user | ||||||
|  |                     // sees the explanation, try again to request the permission. | ||||||
|  |                     new AlertDialog.Builder(this) | ||||||
|  |                             .setMessage(getString(R.string.location_permission_rationale_nearby)) | ||||||
|  |                             .setPositiveButton("OK", (dialog, which) -> { | ||||||
|  |                                 requestLocationPermissions(); | ||||||
|  |                                 dialog.dismiss(); | ||||||
|  |                             }) | ||||||
|  |                             .setNegativeButton("Cancel", (dialog, id) -> { | ||||||
|  |                                 showLocationPermissionDeniedErrorDialog(); | ||||||
|  |                                 dialog.cancel(); | ||||||
|  |                             }) | ||||||
|  |                             .create() | ||||||
|  |                             .show(); | ||||||
|  | 
 | ||||||
|  |                 } else { | ||||||
|  |                     // No explanation needed, we can request the permission. | ||||||
|  |                     requestLocationPermissions(); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } else { | ||||||
|  |             refreshView(false); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  | @ -239,7 +231,6 @@ public class NearbyActivity extends NavigationBaseActivity implements LocationUp | ||||||
|     @Override |     @Override | ||||||
|     protected void onStart() { |     protected void onStart() { | ||||||
|         super.onStart(); |         super.onStart(); | ||||||
|         locationManager.registerLocationManager(); |  | ||||||
|         locationManager.addLocationListener(this); |         locationManager.addLocationListener(this); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  | @ -263,13 +254,18 @@ public class NearbyActivity extends NavigationBaseActivity implements LocationUp | ||||||
|         super.onResume(); |         super.onResume(); | ||||||
|         lockNearbyView = false; |         lockNearbyView = false; | ||||||
|         checkGps(); |         checkGps(); | ||||||
|         refreshView(false); |  | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     /** | ||||||
|  |      * This method should be the single point to load/refresh nearby places | ||||||
|  |      * | ||||||
|  |      * @param isHardRefresh | ||||||
|  |      */ | ||||||
|     private void refreshView(boolean isHardRefresh) { |     private void refreshView(boolean isHardRefresh) { | ||||||
|         if (lockNearbyView) { |         if (lockNearbyView) { | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|  |         locationManager.registerLocationManager(); | ||||||
|         LatLng lastLocation = locationManager.getLastLocation(); |         LatLng lastLocation = locationManager.getLastLocation(); | ||||||
|         if (curLatLang != null && curLatLang.equals(lastLocation)) { //refresh view only if location has changed |         if (curLatLang != null && curLatLang.equals(lastLocation)) { //refresh view only if location has changed | ||||||
|             if (isHardRefresh) { |             if (isHardRefresh) { | ||||||
|  | @ -309,7 +305,7 @@ public class NearbyActivity extends NavigationBaseActivity implements LocationUp | ||||||
|         bundle.putString("PlaceList", gsonPlaceList); |         bundle.putString("PlaceList", gsonPlaceList); | ||||||
|         bundle.putString("CurLatLng", gsonCurLatLng); |         bundle.putString("CurLatLng", gsonCurLatLng); | ||||||
| 
 | 
 | ||||||
|         lockNearbyView = true; |         lockNearbyView(true); | ||||||
|         // Begin the transaction |         // Begin the transaction | ||||||
|         if (viewMode.isMap()) { |         if (viewMode.isMap()) { | ||||||
|             setMapFragment(); |             setMapFragment(); | ||||||
|  | @ -320,6 +316,18 @@ public class NearbyActivity extends NavigationBaseActivity implements LocationUp | ||||||
|         hideProgressBar(); |         hideProgressBar(); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     private void lockNearbyView(boolean lock) { | ||||||
|  |         if (lock) { | ||||||
|  |             lockNearbyView = true; | ||||||
|  |             locationManager.unregisterLocationManager(); | ||||||
|  |             locationManager.removeLocationListener(this); | ||||||
|  |         } else { | ||||||
|  |             lockNearbyView = false; | ||||||
|  |             locationManager.registerLocationManager(); | ||||||
|  |             locationManager.addLocationListener(this); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     private void hideProgressBar() { |     private void hideProgressBar() { | ||||||
|         if (progressBar != null) { |         if (progressBar != null) { | ||||||
|             progressBar.setVisibility(View.GONE); |             progressBar.setVisibility(View.GONE); | ||||||
|  |  | ||||||
|  | @ -213,4 +213,5 @@ Tap this message (or hit back) to skip this step.</string> | ||||||
| 
 | 
 | ||||||
|   <string name="nearby_location_has_not_changed">Location has not changed.</string> |   <string name="nearby_location_has_not_changed">Location has not changed.</string> | ||||||
|   <string name="nearby_location_not_available">Location not available.</string> |   <string name="nearby_location_not_available">Location not available.</string> | ||||||
|  |   <string name="location_permission_rationale_nearby">Permission required to display a list of nearby places</string> | ||||||
| </resources> | </resources> | ||||||
|  |  | ||||||
|  | @ -1,5 +1,6 @@ | ||||||
| package fr.free.nrw.commons.nearby; | package fr.free.nrw.commons.nearby; | ||||||
| 
 | 
 | ||||||
|  | import android.app.Activity; | ||||||
| import org.junit.Before; | import org.junit.Before; | ||||||
| import org.junit.Test; | import org.junit.Test; | ||||||
| import org.junit.runner.RunWith; | import org.junit.runner.RunWith; | ||||||
|  |  | ||||||
							
								
								
									
										
											BIN
										
									
								
								design/screenshots/Chinese (Simplified) zh-CN/car-categories.png
									
										
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 2 MiB | 
| After Width: | Height: | Size: 2 MiB | 
							
								
								
									
										
											BIN
										
									
								
								design/screenshots/Chinese (Simplified) zh-CN/car-details.png
									
										
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 1.5 MiB | 
							
								
								
									
										
											BIN
										
									
								
								design/screenshots/Chinese (Simplified) zh-CN/drawer.png
									
										
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 621 KiB | 
							
								
								
									
										
											BIN
										
									
								
								design/screenshots/Chinese (Simplified) zh-CN/gallery.png
									
										
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 2.1 MiB | 
							
								
								
									
										
											BIN
										
									
								
								design/screenshots/Chinese (Simplified) zh-CN/nearby-list.png
									
										
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 128 KiB | 
							
								
								
									
										
											BIN
										
									
								
								design/screenshots/Chinese (Simplified) zh-CN/nearby-map.png
									
										
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 909 KiB | 
| After Width: | Height: | Size: 1.7 MiB | 
| After Width: | Height: | Size: 1.6 MiB | 
| After Width: | Height: | Size: 1.1 MiB | 
| After Width: | Height: | Size: 1.8 MiB | 
| After Width: | Height: | Size: 1.7 MiB | 
							
								
								
									
										
											BIN
										
									
								
								design/screenshots/Chinese (Simplified) zh-CN/school-details.png
									
										
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 1.1 MiB | 
							
								
								
									
										
											BIN
										
									
								
								design/screenshots/Chinese (Simplified) zh-CN/taking-picture.png
									
										
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 2.5 MiB | 
 Vivek Maskara
						Vivek Maskara