[android] Remove track recorder

Make Google happy. This feature is broken anyway.

Signed-off-by: Roman Tsisyk <roman@tsisyk.com>
This commit is contained in:
Roman Tsisyk 2021-06-04 14:36:48 +03:00
parent 2274f6331f
commit 2bfb7ffc0b
10 changed files with 6 additions and 436 deletions

View file

@ -294,16 +294,6 @@
android:permission="android.permission.BIND_JOB_SERVICE"
android:exported="false"/>
<receiver android:name="com.mapswithme.maps.location.TrackRecorderWakeReceiver">
<intent-filter>
<action android:name="com.mapswithme.maps.TRACK_RECORDER_ALARM"/>
</intent-filter>
</receiver>
<service android:name="com.mapswithme.maps.location.TrackRecorderWakeService"
android:permission="android.permission.BIND_JOB_SERVICE"
android:exported="false"/>
<receiver
android:name="com.mapswithme.maps.background.SystemDownloadCompletedReceiver"
android:exported="true">

View file

@ -28,9 +28,6 @@
<string name="pref_autodownload" translatable="false">AutoDownloadMap</string>
<string name="pref_3d" translatable="false">3D</string>
<string name="pref_3d_buildings" translatable="false">3DBuildings</string>
<string name="pref_track_screen" translatable="false">TrackScreen</string>
<string name="pref_track_record" translatable="false">TrackRecord</string>
<string name="pref_track_record_time" translatable="false">TrackRecordTime</string>
<string name="pref_osm_profile" translatable="false">Osm profile</string>
<string name="pref_auto_zoom" translatable="false">AutoZoom</string>
<string name="pref_large_fonts_size" translatable="false">LargeFontSize</string>

View file

@ -40,28 +40,6 @@
android:title="@string/whatsnew_transliteration_title"
android:defaultValue="false"
android:order="7"/>
<androidx.preference.PreferenceScreen
android:key="@string/pref_track_screen"
android:title="@string/pref_track_record_title"
android:persistent="false"
android:order="8">
<SwitchPreferenceCompat
android:key="@string/pref_track_record_time"
android:title="@string/pref_track_record_title"
android:order="1"/>
<ListPreference
android:key="@string/pref_track_record"
android:title="@string/pref_track_record_title"
android:entries="@array/track_length"
android:entryValues="@array/track_length_values"
android:order="2"/>
<Preference
android:selectable="false"
android:persistent="false"
android:enabled="false"
android:summary="@string/recent_track_help_text"
android:order="3"/>
</androidx.preference.PreferenceScreen>
<Preference
android:key="@string/pref_storage"
android:title="@string/maps_storage"

View file

@ -21,7 +21,6 @@ import com.mapswithme.maps.downloader.MapDownloadManager;
import com.mapswithme.maps.downloader.MapManager;
import com.mapswithme.maps.editor.Editor;
import com.mapswithme.maps.location.LocationHelper;
import com.mapswithme.maps.location.TrackRecorder;
import com.mapswithme.maps.maplayer.isolines.IsolinesManager;
import com.mapswithme.maps.maplayer.subway.SubwayManager;
import com.mapswithme.maps.maplayer.traffic.TrafficManager;
@ -239,7 +238,6 @@ public class MwmApplication extends Application implements AppBackgroundTracker.
TrafficManager.INSTANCE.initialize(null);
SubwayManager.from(this).initialize(null);
IsolinesManager.from(this).initialize(null);
TrackRecorder.INSTANCE.initialize(this);
mBackgroundTracker.addListener(this);
getLogger().i(TAG, "Framework initialized");

View file

@ -1,218 +0,0 @@
package com.mapswithme.maps.location;
import android.app.AlarmManager;
import android.app.PendingIntent;
import android.content.Context;
import android.content.Intent;
import android.location.Location;
import android.os.SystemClock;
import androidx.annotation.NonNull;
import androidx.annotation.Nullable;
import com.mapswithme.maps.MwmApplication;
import com.mapswithme.maps.base.Initializable;
import com.mapswithme.util.concurrency.UiThread;
import com.mapswithme.util.log.Logger;
import com.mapswithme.util.log.LoggerFactory;
public enum TrackRecorder implements Initializable<Context>
{
INSTANCE;
@NonNull
private static final Logger LOGGER = LoggerFactory.INSTANCE.getLogger(LoggerFactory.Type.TRACK_RECORDER);
private static final String TAG = TrackRecorder.class.getSimpleName();
private static final long WAKEUP_INTERVAL_MS = 20000;
private static final long STARTUP_AWAIT_INTERVAL_MS = 5000;
private static final String LOCATION_TIMEOUT_STORED_KEY = "TrackRecordLastAwaitTimeout";
private static final long LOCATION_TIMEOUT_MIN_MS = 5000;
private static final long LOCATION_TIMEOUT_MAX_MS = 80000;
private boolean mInitialized = false;
@SuppressWarnings("NotNullFieldNotInitialized")
@NonNull
private Context mContext;
@NonNull
private AlarmManager mAlarmManager;
@NonNull
private final Runnable mStartupAwaitProc = this::restartAlarmIfEnabled;
@NonNull
private final LocationListener mLocationListener = new LocationListener.Simple()
{
@Override
public void onLocationUpdated(Location location)
{
LOGGER.d(TAG, "onLocationUpdated()");
setAwaitTimeout(LOCATION_TIMEOUT_MIN_MS);
LocationHelper.INSTANCE.onLocationUpdated(location);
TrackRecorderWakeService.stop();
}
@Override
public void onLocationError(int errorCode)
{
LOGGER.e(TAG, "onLocationError() errorCode: " + errorCode);
// Unrecoverable error occured: GPS disabled or inaccessible
setEnabled(false);
}
};
@Override
public void initialize(@Nullable Context context)
{
LOGGER.d(TAG, "Initialization of track recorder and setting the listener for track changes");
mContext = context;
mAlarmManager = (AlarmManager) MwmApplication.from(context)
.getSystemService(Context.ALARM_SERVICE);
MwmApplication.backgroundTracker(context).addListener(foreground -> {
LOGGER.d(TAG, "Transit to foreground: " + foreground);
UiThread.cancelDelayedTasks(mStartupAwaitProc);
if (foreground)
TrackRecorderWakeService.stop();
else
restartAlarmIfEnabled();
});
if (nativeIsEnabled())
UiThread.runLater(mStartupAwaitProc, STARTUP_AWAIT_INTERVAL_MS);
else
stop();
mInitialized = true;
}
@Override
public void destroy()
{
// No op.
}
private void checkInitialization()
{
if (!mInitialized)
throw new AssertionError("Track recorder is not initialized!");
}
private PendingIntent getAlarmIntent()
{
Intent intent = new Intent(MwmApplication.from(mContext), TrackRecorderWakeReceiver.class);
return PendingIntent.getBroadcast(MwmApplication.from(mContext), 0, intent, 0);
}
private void restartAlarmIfEnabled()
{
LOGGER.d(TAG, "restartAlarmIfEnabled()");
if (nativeIsEnabled())
mAlarmManager.set(AlarmManager.ELAPSED_REALTIME_WAKEUP, SystemClock.elapsedRealtime() + WAKEUP_INTERVAL_MS, getAlarmIntent());
}
private void stop()
{
LOGGER.d(TAG, "stop(). Cancel awake timer");
mAlarmManager.cancel(getAlarmIntent());
TrackRecorderWakeService.stop();
}
public boolean isEnabled()
{
checkInitialization();
return nativeIsEnabled();
}
public void setEnabled(boolean enabled)
{
checkInitialization();
LOGGER.d(TAG, "setEnabled(): " + enabled);
setAwaitTimeout(LOCATION_TIMEOUT_MIN_MS);
nativeSetEnabled(enabled);
if (enabled)
restartAlarmIfEnabled();
else
stop();
}
public int getDuration()
{
checkInitialization();
return nativeGetDuration();
}
public void setDuration(int hours)
{
checkInitialization();
nativeSetDuration(hours);
}
void onWakeAlarm()
{
LOGGER.d(TAG, "onWakeAlarm(). Enabled: " + nativeIsEnabled());
UiThread.cancelDelayedTasks(mStartupAwaitProc);
if (nativeIsEnabled() && !MwmApplication.backgroundTracker(mContext).isForeground())
TrackRecorderWakeService.start(mContext);
else
stop();
}
long getAwaitTimeout()
{
return MwmApplication.prefs(mContext).getLong(LOCATION_TIMEOUT_STORED_KEY, LOCATION_TIMEOUT_MIN_MS);
}
private void setAwaitTimeout(long timeout)
{
LOGGER.d(TAG, "setAwaitTimeout(): " + timeout);
if (timeout != getAwaitTimeout())
MwmApplication.prefs(mContext).edit().putLong(LOCATION_TIMEOUT_STORED_KEY, timeout).apply();
}
void incrementAwaitTimeout()
{
long current = getAwaitTimeout();
long next = current * 2;
if (next > LOCATION_TIMEOUT_MAX_MS)
next = LOCATION_TIMEOUT_MAX_MS;
if (next != current)
setAwaitTimeout(next);
}
void onServiceStarted()
{
LOGGER.d(TAG, "onServiceStarted(). Scheduled to be run on UI thread...");
UiThread.run(() -> {
LOGGER.d(TAG, "onServiceStarted(): actually runs here");
LocationHelper.INSTANCE.addListener(mLocationListener, false);
});
}
void onServiceStopped()
{
LOGGER.d(TAG, "onServiceStopped(). Scheduled to be run on UI thread...");
UiThread.run(() -> {
LOGGER.d(TAG, "onServiceStopped(): actually runs here");
LocationHelper.INSTANCE.removeListener(mLocationListener);
if (!MwmApplication.backgroundTracker(mContext).isForeground())
restartAlarmIfEnabled();
});
}
private native void nativeSetEnabled(boolean enable);
private native boolean nativeIsEnabled();
private native void nativeSetDuration(int hours);
private native int nativeGetDuration();
}

View file

@ -1,17 +0,0 @@
package com.mapswithme.maps.location;
import android.content.Context;
import android.content.Intent;
import androidx.annotation.NonNull;
import com.mapswithme.maps.MwmBroadcastReceiver;
public class TrackRecorderWakeReceiver extends MwmBroadcastReceiver
{
@Override
public void onReceiveInitialized(@NonNull Context context, @NonNull Intent intent)
{
TrackRecorder.INSTANCE.onWakeAlarm();
}
}

View file

@ -1,95 +0,0 @@
package com.mapswithme.maps.location;
import android.app.job.JobInfo;
import android.app.job.JobScheduler;
import android.content.Context;
import android.content.Intent;
import android.util.Log;
import androidx.annotation.NonNull;
import androidx.core.app.JobIntentService;
import com.mapswithme.maps.MwmJobIntentService;
import com.mapswithme.maps.scheduling.JobIdMap;
import com.mapswithme.util.CrashlyticsUtils;
import com.mapswithme.util.log.Logger;
import com.mapswithme.util.log.LoggerFactory;
import java.util.Arrays;
import java.util.List;
import java.util.Objects;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.TimeUnit;
public class TrackRecorderWakeService extends MwmJobIntentService
{
private static final String TAG = TrackRecorderWakeService.class.getSimpleName();
private static final Logger LOGGER = LoggerFactory.INSTANCE.getLogger(LoggerFactory.Type.TRACK_RECORDER);
private static final Object sLock = new Object();
private static TrackRecorderWakeService sService;
private final CountDownLatch mWaitMonitor = new CountDownLatch(1);
@Override
protected void onHandleWorkInitialized(@NonNull Intent intent)
{
synchronized (sLock)
{
sService = this;
}
TrackRecorder.INSTANCE.onServiceStarted();
try
{
long timeout = TrackRecorder.INSTANCE.getAwaitTimeout();
LOGGER.d(TAG, "Timeout: " + timeout);
if (!mWaitMonitor.await(timeout, TimeUnit.MILLISECONDS))
{
LOGGER.d(TAG, "TIMEOUT awaiting coordinates");
TrackRecorder.INSTANCE.incrementAwaitTimeout();
}
} catch (InterruptedException ignored) {}
synchronized (sLock)
{
sService = null;
}
TrackRecorder.INSTANCE.onServiceStopped();
}
public static void start(@NonNull Context context)
{
Context app = context.getApplicationContext();
Intent intent = new Intent(app, TrackRecorderWakeService.class);
final int jobId = JobIdMap.getId(TrackRecorderWakeService.class);
JobScheduler scheduler = (JobScheduler) context.getSystemService(Context.JOB_SCHEDULER_SERVICE);
Objects.requireNonNull(scheduler);
List<JobInfo> pendingJobs = scheduler.getAllPendingJobs();
String jobsRepresentation = Arrays.toString(pendingJobs.toArray());
for (JobInfo each : pendingJobs)
{
if (TrackRecorderWakeService.class.getName().equals(each.getService().getClassName()))
{
scheduler.cancel(each.getId());
String logMsg = "Canceled job: " + each + ". All jobs: " + jobsRepresentation;
CrashlyticsUtils.INSTANCE.log(Log.INFO, TAG, logMsg);
}
}
JobIntentService.enqueueWork(app, TrackRecorderWakeService.class, jobId, intent);
}
public static void stop()
{
LOGGER.d(TAG, "SVC.stop()");
synchronized (sLock)
{
if (sService != null)
sService.mWaitMonitor.countDown();
else
LOGGER.d(TAG, "SVC.stop() SKIPPED because (sService == null)");
}
}
}

View file

@ -2,7 +2,6 @@ package com.mapswithme.maps.scheduling;
import com.mapswithme.maps.background.OsmUploadService;
import com.mapswithme.maps.background.SystemDownloadCompletedService;
import com.mapswithme.maps.location.TrackRecorderWakeService;
import java.util.HashMap;
import java.util.Map;
@ -12,7 +11,6 @@ public class JobIdMap
private static final Map<Class<?>, Integer> MAP = new HashMap<>();
static {
MAP.put(TrackRecorderWakeService.class, calcIdentifier(MAP.size()));
MAP.put(SystemDownloadCompletedService.class, calcIdentifier(MAP.size()));
MAP.put(OsmUploadService.class, calcIdentifier(MAP.size()));
}

View file

@ -25,13 +25,11 @@ import androidx.preference.TwoStatePreference;
import com.mapswithme.maps.Framework;
import com.mapswithme.maps.R;
import com.mapswithme.maps.bookmarks.data.BookmarkManager;
import com.mapswithme.maps.downloader.MapManager;
import com.mapswithme.maps.downloader.OnmapDownloader;
import com.mapswithme.maps.editor.ProfileActivity;
import com.mapswithme.maps.location.LocationHelper;
import com.mapswithme.maps.location.LocationProviderFactory;
import com.mapswithme.maps.location.TrackRecorder;
import com.mapswithme.maps.sound.LanguageData;
import com.mapswithme.maps.sound.TtsPlayer;
import com.mapswithme.util.Config;
@ -298,7 +296,6 @@ public class SettingsPrefsFragment extends BaseXmlSettingsFragment
initTransliterationPrefsCallbacks();
init3dModePrefsCallbacks();
initPerspectivePrefsCallbacks();
initTrackRecordPrefsCallbacks();
initPlayServicesPrefsCallbacks();
initAutoZoomPrefsCallbacks();
initLoggingEnabledPrefsCallbacks();
@ -347,7 +344,6 @@ public class SettingsPrefsFragment extends BaseXmlSettingsFragment
{
super.onResume();
initTrackRecordPrefsCallbacks();
updateTts();
}
@ -599,68 +595,6 @@ public class SettingsPrefsFragment extends BaseXmlSettingsFragment
}
}
private void initTrackRecordPrefsCallbacks()
{
final ListPreference trackPref = findPreference(getString(R.string.pref_track_record));
final Preference pref = findPreference(getString(R.string.pref_track_record_time));
final Preference root = findPreference(getString(R.string.pref_track_screen));
if (trackPref == null || pref == null)
return;
boolean enabled = TrackRecorder.INSTANCE.isEnabled();
((TwoStatePreference)pref).setChecked(enabled);
trackPref.setEnabled(enabled);
if (root != null)
root.setSummary(enabled ? R.string.on : R.string.off);
pref.setTitle(enabled ? R.string.on : R.string.off);
pref.setOnPreferenceChangeListener(new Preference.OnPreferenceChangeListener()
{
@Override
public boolean onPreferenceChange(Preference preference, Object newValue)
{
boolean enabled = (Boolean) newValue;
TrackRecorder.INSTANCE.setEnabled(enabled);
trackPref.setEnabled(enabled);
if (root != null)
root.setSummary(enabled ? R.string.on : R.string.off);
pref.setTitle(enabled ? R.string.on : R.string.off);
trackPref.performClick();
return true;
}
});
String value = (enabled ? String.valueOf(TrackRecorder.INSTANCE.getDuration()) : "0");
trackPref.setValue(value);
trackPref.setSummary(trackPref.getEntry());
trackPref.setOnPreferenceChangeListener(new Preference.OnPreferenceChangeListener()
{
@Override
public boolean onPreferenceChange(final Preference preference, Object newValue)
{
int value = Integer.valueOf((String)newValue);
boolean enabled = value != 0;
if (enabled)
TrackRecorder.INSTANCE.setDuration(value);
TrackRecorder.INSTANCE.setEnabled(enabled);
((TwoStatePreference) pref).setChecked(enabled);
trackPref.setEnabled(enabled);
if (root != null)
root.setSummary(enabled ? R.string.on : R.string.off);
pref.setTitle(enabled ? R.string.on : R.string.off);
UiThread.runLater(new Runnable()
{
@Override
public void run()
{
trackPref.setSummary(trackPref.getEntry());
}
});
return true;
}
});
}
private void init3dModePrefsCallbacks()
{
final TwoStatePreference pref = findPreference(getString(R.string.pref_3d_buildings));

View file

@ -5020,7 +5020,7 @@
[pref_track_record_title]
comment = Settings «Map» category: «Record track» title
tags = ios,android
tags = ios
en = Recent track
ru = Недавний путь
ar = المسار الأخير
@ -5282,6 +5282,7 @@
fa = 1 روز
[recent_track_help_text]
tags = ios
en = This option allows you to record traveled path for a certain period and see it on the map. Please note: activation of this function causes increased battery usage. The track will be removed automatically from the map after the time interval will expire.
ru = Эта функция позволяет записывать пройденный путь за определенный период времени и видеть его на карте. Внимание: активация этой функции может привести к повышенному расходу батареи. Записанный трек будет удален с карты по истечении этого срока.
ar = يتيح لك تسجيل مسار السفر لفترة معينة وعرضه على الخريطة. الرجاء ملاحظة: يؤدي تنشيط هذه الوظيفة إلى زيادة استهلاك البطارية. سوف تتم إزالة المسار تلقائيًا من الخريطة بعد انتهاء الفاصل الزمني.
@ -18930,6 +18931,7 @@
fa = خواندن
[recent_track_background_dialog_title]
tags = ios
comment = iOS dialog for the case when recent track recording is on and the app comes back from background
en = Disable recording of your recently traveled route?
ru = Выключить запись недавно пройденого пути?
@ -18962,6 +18964,7 @@
fa = غیرفعال سازی ذخیره مسیر اخیرا طی شده
[off_recent_track_background_button]
tags = ios
en = Disable
ru = Выключить
ar = تعطيل
@ -19025,6 +19028,7 @@
fa = بررسی کنید
[continue_recent_track_background_button]
tags = ios
en = Continue
ru = Продолжить
ar = استمرار
@ -19055,6 +19059,7 @@
fa = ادامه
[recent_track_background_dialog_message]
tags = ios
en = Organic Maps uses your geoposition in the background for recording your recently traveled route.
ru = Organic Maps использует вашу геопозицию в фоновом режиме для записи недавно пройденного пути.
ar = يستخدم Organic Maps موقعك الجغرافي في الخلفية لتسجيل مسار سفرك الأخير.