2024-01-10 18:54:00 +01:00
/ * Copyright ( C ) 2015 - 2024 Andreas Shimokawa , Arjan Schrijver , Carsten
Pfeiffer , Damien Gaignon , Daniel Dakhno , Daniele Gobbetti , Davis Mosenkovs ,
Dmitriy Bogdanov , Joel Beckmeyer , José Rebelo , Kornél Schmidt , Ludovic
Jozeau , Martin , Martin . JM , mvn23 , Normano64 , odavo32nof , Pauli Salmenrinne ,
Pavel Elagin , Petr Vaněk , Saul Nunez , Taavi Eomäe , x29a
2017-03-10 14:53:19 +01:00
This file is part of Gadgetbridge .
Gadgetbridge is free software : you can redistribute it and / or modify
it under the terms of the GNU Affero General Public License as published
by the Free Software Foundation , either version 3 of the License , or
( at your option ) any later version .
Gadgetbridge is distributed in the hope that it will be useful ,
but WITHOUT ANY WARRANTY ; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
GNU Affero General Public License for more details .
You should have received a copy of the GNU Affero General Public License
2024-01-10 18:54:00 +01:00
along with this program . If not , see < https : //www.gnu.org/licenses/>. */
2015-05-01 09:36:10 +02:00
package nodomain.freeyourgadget.gadgetbridge ;
2016-05-20 21:49:25 +02:00
import android.annotation.TargetApi ;
2015-05-01 09:36:10 +02:00
import android.app.Application ;
2016-05-19 23:58:13 +02:00
import android.app.NotificationManager ;
2016-05-19 16:34:59 +02:00
import android.app.NotificationManager.Policy ;
2018-05-16 19:03:06 +02:00
import android.bluetooth.BluetoothAdapter ;
2015-05-01 09:36:10 +02:00
import android.content.Context ;
2015-12-07 23:33:32 +01:00
import android.content.Intent ;
2018-05-16 19:03:06 +02:00
import android.content.IntentFilter ;
2015-05-23 00:45:12 +02:00
import android.content.SharedPreferences ;
2017-10-19 21:52:38 +02:00
import android.content.pm.ApplicationInfo ;
import android.content.pm.PackageInfo ;
import android.content.pm.PackageManager ;
2017-07-31 22:49:05 +02:00
import android.content.res.Configuration ;
2016-04-23 23:24:56 +02:00
import android.content.res.Resources ;
2016-05-19 16:34:59 +02:00
import android.database.Cursor ;
2016-05-13 23:47:47 +02:00
import android.database.sqlite.SQLiteDatabase ;
2016-05-19 16:34:59 +02:00
import android.net.Uri ;
2015-06-06 00:10:38 +02:00
import android.os.Build ;
import android.os.Build.VERSION ;
2015-05-23 00:45:12 +02:00
import android.preference.PreferenceManager ;
2016-05-19 16:34:59 +02:00
import android.provider.ContactsContract.PhoneLookup ;
2015-06-21 10:18:41 +02:00
import android.util.Log ;
2016-04-23 23:24:56 +02:00
import android.util.TypedValue ;
2015-05-23 00:45:12 +02:00
2020-07-28 01:09:59 +02:00
import androidx.core.app.NotificationCompat ;
2019-05-09 14:55:37 +02:00
import androidx.localbroadcastmanager.content.LocalBroadcastManager ;
2015-05-13 23:15:20 +02:00
import java.io.File ;
2015-07-08 23:03:34 +02:00
import java.io.IOException ;
2023-10-22 01:33:08 +02:00
import java.io.InputStream ;
2019-11-24 18:40:44 +01:00
import java.util.ArrayList ;
import java.util.Arrays ;
import java.util.Collections ;
2015-09-25 00:53:40 +02:00
import java.util.HashSet ;
2016-11-24 21:11:28 +01:00
import java.util.List ;
2017-07-31 22:49:05 +02:00
import java.util.Locale ;
2023-12-05 12:45:32 +01:00
import java.util.Map ;
2017-04-22 16:59:55 +02:00
import java.util.Set ;
2015-08-03 01:17:02 +02:00
import java.util.concurrent.TimeUnit ;
import java.util.concurrent.locks.Lock ;
import java.util.concurrent.locks.ReentrantLock ;
2015-05-13 23:15:20 +02:00
2021-11-06 10:07:16 +01:00
import nodomain.freeyourgadget.gadgetbridge.activities.devicesettings.DeviceSettingsPreferenceConst ;
2015-08-03 01:17:02 +02:00
import nodomain.freeyourgadget.gadgetbridge.database.DBHandler ;
2016-08-27 22:51:00 +02:00
import nodomain.freeyourgadget.gadgetbridge.database.DBHelper ;
2016-05-16 23:54:51 +02:00
import nodomain.freeyourgadget.gadgetbridge.database.DBOpenHelper ;
2022-02-27 18:43:53 +01:00
import nodomain.freeyourgadget.gadgetbridge.database.PeriodicExporter ;
2016-06-18 23:35:34 +02:00
import nodomain.freeyourgadget.gadgetbridge.devices.DeviceManager ;
2016-04-17 19:52:51 +02:00
import nodomain.freeyourgadget.gadgetbridge.entities.DaoMaster ;
2019-05-20 21:53:53 +02:00
import nodomain.freeyourgadget.gadgetbridge.entities.DaoSession ;
2019-05-20 16:36:06 +02:00
import nodomain.freeyourgadget.gadgetbridge.entities.Device ;
2018-05-16 19:03:06 +02:00
import nodomain.freeyourgadget.gadgetbridge.externalevents.BluetoothStateChangeReceiver ;
2023-09-02 11:13:14 +02:00
import nodomain.freeyourgadget.gadgetbridge.externalevents.TimeChangeReceiver ;
2022-05-22 18:11:45 +02:00
import nodomain.freeyourgadget.gadgetbridge.externalevents.opentracks.OpenTracksContentObserver ;
2016-11-24 21:11:28 +01:00
import nodomain.freeyourgadget.gadgetbridge.impl.GBDevice ;
2015-08-21 00:58:18 +02:00
import nodomain.freeyourgadget.gadgetbridge.impl.GBDeviceService ;
2016-02-02 17:33:24 +01:00
import nodomain.freeyourgadget.gadgetbridge.model.ActivityUser ;
2015-08-21 00:58:18 +02:00
import nodomain.freeyourgadget.gadgetbridge.model.DeviceService ;
2019-05-23 23:24:49 +02:00
import nodomain.freeyourgadget.gadgetbridge.model.DeviceType ;
2022-08-18 23:03:28 +02:00
import nodomain.freeyourgadget.gadgetbridge.model.Weather ;
2017-04-19 13:23:13 +02:00
import nodomain.freeyourgadget.gadgetbridge.service.NotificationCollectorMonitorService ;
2017-09-03 01:02:31 +02:00
import nodomain.freeyourgadget.gadgetbridge.util.AndroidUtils ;
2015-07-08 23:03:34 +02:00
import nodomain.freeyourgadget.gadgetbridge.util.FileUtils ;
2015-08-22 01:08:46 +02:00
import nodomain.freeyourgadget.gadgetbridge.util.GB ;
2016-04-28 23:17:13 +02:00
import nodomain.freeyourgadget.gadgetbridge.util.GBPrefs ;
2016-01-09 16:07:22 +01:00
import nodomain.freeyourgadget.gadgetbridge.util.LimitedQueue ;
2016-04-25 23:18:55 +02:00
import nodomain.freeyourgadget.gadgetbridge.util.Prefs ;
2015-05-30 17:28:03 +02:00
2019-05-23 23:24:49 +02:00
import static nodomain.freeyourgadget.gadgetbridge.model.DeviceType.AMAZFITBIP ;
import static nodomain.freeyourgadget.gadgetbridge.model.DeviceType.AMAZFITCOR ;
import static nodomain.freeyourgadget.gadgetbridge.model.DeviceType.AMAZFITCOR2 ;
2022-03-10 00:06:41 +01:00
import static nodomain.freeyourgadget.gadgetbridge.model.DeviceType.FITPRO ;
2021-11-06 10:07:16 +01:00
import static nodomain.freeyourgadget.gadgetbridge.model.DeviceType.GALAXY_BUDS ;
2022-03-10 00:06:41 +01:00
import static nodomain.freeyourgadget.gadgetbridge.model.DeviceType.LEFUN ;
2019-06-18 22:07:43 +02:00
import static nodomain.freeyourgadget.gadgetbridge.model.DeviceType.MIBAND ;
2019-05-24 23:46:20 +02:00
import static nodomain.freeyourgadget.gadgetbridge.model.DeviceType.MIBAND2 ;
2023-08-27 23:47:19 +02:00
import static nodomain.freeyourgadget.gadgetbridge.model.DeviceType.MIBAND2_HRX ;
2019-05-24 23:46:20 +02:00
import static nodomain.freeyourgadget.gadgetbridge.model.DeviceType.MIBAND3 ;
2020-10-13 00:25:43 +02:00
import static nodomain.freeyourgadget.gadgetbridge.model.DeviceType.PEBBLE ;
2022-03-10 00:06:41 +01:00
import static nodomain.freeyourgadget.gadgetbridge.model.DeviceType.TLW64 ;
import static nodomain.freeyourgadget.gadgetbridge.model.DeviceType.WATCHXPLUS ;
2020-03-03 18:01:39 +01:00
import static nodomain.freeyourgadget.gadgetbridge.util.GB.NOTIFICATION_CHANNEL_HIGH_PRIORITY_ID ;
2021-05-14 18:30:54 +02:00
import static nodomain.freeyourgadget.gadgetbridge.util.GB.NOTIFICATION_ID_ERROR ;
2020-07-28 01:09:59 +02:00
2022-09-28 07:51:10 +02:00
import com.jakewharton.threetenabp.AndroidThreeTen ;
2023-06-17 18:28:11 +02:00
import org.apache.commons.lang3.StringUtils ;
2023-10-22 01:33:08 +02:00
import org.json.JSONObject ;
2023-06-17 18:28:11 +02:00
2015-10-26 23:32:03 +01:00
/ * *
* Main Application class that initializes and provides access to certain things like
* logging and DB access .
* /
2015-05-01 09:36:10 +02:00
public class GBApplication extends Application {
2015-08-03 01:17:02 +02:00
// Since this class must not log to slf4j, we use plain android.util.Log
private static final String TAG = " GBApplication " ;
2016-08-27 22:51:00 +02:00
public static final String DATABASE_NAME = " Gadgetbridge " ;
2015-05-01 09:36:10 +02:00
private static GBApplication context ;
2015-08-16 00:17:16 +02:00
private static final Lock dbLock = new ReentrantLock ( ) ;
2015-08-21 00:58:18 +02:00
private static DeviceService deviceService ;
2015-09-25 00:53:40 +02:00
private static SharedPreferences sharedPrefs ;
2016-02-02 17:33:24 +01:00
private static final String PREFS_VERSION = " shared_preferences_version " ;
//if preferences have to be migrated, increment the following and add the migration logic in migratePrefs below; see http://stackoverflow.com/questions/16397848/how-can-i-migrate-android-preferences-with-a-new-version
2023-12-09 12:40:28 +01:00
private static final int CURRENT_PREFS_VERSION = 28 ;
2020-07-28 01:09:59 +02:00
2023-12-10 11:30:27 +01:00
private static final LimitedQueue < Integer , String > mIDSenderLookup = new LimitedQueue < > ( 16 ) ;
2016-04-25 23:18:55 +02:00
private static Prefs prefs ;
2016-04-28 23:17:13 +02:00
private static GBPrefs gbPrefs ;
2016-06-14 20:13:08 +02:00
private static LockHandler lockHandler ;
2016-05-20 22:04:30 +02:00
/ * *
2022-09-09 19:58:34 +02:00
* Note : is null on Lollipop
2016-05-20 22:04:30 +02:00
* /
2016-05-19 23:58:13 +02:00
private static NotificationManager notificationManager ;
2015-05-01 09:36:10 +02:00
2015-12-07 23:33:32 +01:00
public static final String ACTION_QUIT
= " nodomain.freeyourgadget.gadgetbridge.gbapplication.action.quit " ;
2017-07-31 22:49:05 +02:00
public static final String ACTION_LANGUAGE_CHANGE = " nodomain.freeyourgadget.gadgetbridge.gbapplication.action.language_change " ;
2023-07-22 21:23:27 +02:00
public static final String ACTION_THEME_CHANGE = " nodomain.freeyourgadget.gadgetbridge.gbapplication.action.theme_change " ;
2019-09-01 22:09:09 +02:00
public static final String ACTION_NEW_DATA = " nodomain.freeyourgadget.gadgetbridge.action.new_data " ;
2017-04-27 07:52:11 +02:00
private static GBApplication app ;
2016-05-26 23:46:21 +02:00
private static Logging logging = new Logging ( ) {
@Override
protected String createLogDirectory ( ) throws IOException {
2017-04-27 07:52:11 +02:00
if ( GBEnvironment . env ( ) . isLocalTest ( ) ) {
return System . getProperty ( Logging . PROP_LOGFILES_DIR ) ;
} else {
File dir = FileUtils . getExternalFilesDir ( ) ;
return dir . getAbsolutePath ( ) ;
}
2016-05-26 23:46:21 +02:00
}
} ;
2017-07-31 22:49:05 +02:00
private static Locale language ;
2016-12-07 23:13:51 +01:00
private DeviceManager deviceManager ;
2018-05-16 19:03:06 +02:00
private BluetoothStateChangeReceiver bluetoothStateChangeReceiver ;
2015-12-07 23:33:32 +01:00
2022-02-08 13:38:23 +01:00
private OpenTracksContentObserver openTracksObserver ;
2023-10-06 22:38:35 +02:00
2022-02-27 18:43:53 +01:00
private long lastAutoExportTimestamp = 0 ;
private long autoExportScheduledTimestamp = 0 ;
2022-02-08 13:38:23 +01:00
2016-07-08 22:35:52 +02:00
public static void quit ( ) {
GB . log ( " Quitting Gadgetbridge... " , GB . INFO , null ) ;
Intent quitIntent = new Intent ( GBApplication . ACTION_QUIT ) ;
LocalBroadcastManager . getInstance ( context ) . sendBroadcast ( quitIntent ) ;
GBApplication . deviceService ( ) . quit ( ) ;
2021-07-20 20:59:58 +02:00
System . exit ( 0 ) ;
2015-12-07 23:33:32 +01:00
}
2015-05-01 09:36:10 +02:00
public GBApplication ( ) {
context = this ;
2015-07-25 00:07:33 +02:00
// don't do anything here, add it to onCreate instead
2023-10-06 22:38:35 +02:00
//if (BuildConfig.DEBUG) {
// // detect everything
// //StrictMode.enableDefaults();
// // detect closeable objects
// //StrictMode.setVmPolicy(
// // new StrictMode.VmPolicy.Builder()
// // .detectLeakedClosableObjects()
// // .penaltyLog()
// // .build()
// //);
//}
2015-05-01 09:36:10 +02:00
}
2017-04-27 07:52:11 +02:00
public static Logging getLogging ( ) {
return logging ;
}
2015-08-21 00:58:18 +02:00
protected DeviceService createDeviceService ( ) {
2015-08-22 01:08:46 +02:00
return new GBDeviceService ( this ) ;
2015-08-21 00:58:18 +02:00
}
2015-05-13 23:15:20 +02:00
@Override
public void onCreate ( ) {
2017-04-27 07:52:11 +02:00
app = this ;
2015-05-13 23:15:20 +02:00
super . onCreate ( ) ;
2016-08-25 00:00:53 +02:00
if ( lockHandler ! = null ) {
// guard against multiple invocations (robolectric)
return ;
}
2022-09-28 07:51:10 +02:00
// Initialize the timezones library
AndroidThreeTen . init ( this ) ;
2015-09-25 00:53:40 +02:00
sharedPrefs = PreferenceManager . getDefaultSharedPreferences ( context ) ;
2016-04-25 23:18:55 +02:00
prefs = new Prefs ( sharedPrefs ) ;
2016-04-28 23:17:13 +02:00
gbPrefs = new GBPrefs ( prefs ) ;
2015-09-25 00:53:40 +02:00
2017-04-27 07:57:37 +02:00
if ( ! GBEnvironment . isEnvironmentSetup ( ) ) {
GBEnvironment . setupEnvironment ( GBEnvironment . createDeviceEnvironment ( ) ) ;
// setup db after the environment is set up, but don't do it in test mode
// in test mode, it's done individually, see TestBase
setupDatabase ( ) ;
}
2015-07-25 00:07:33 +02:00
// don't do anything here before we set up logging, otherwise
// slf4j may be implicitly initialized before we properly configured it.
2016-04-18 00:20:40 +02:00
setupLogging ( isFileLoggingEnabled ( ) ) ;
2015-10-07 23:32:58 +02:00
2019-05-20 21:53:53 +02:00
if ( getPrefsFileVersion ( ) ! = CURRENT_PREFS_VERSION ) {
2016-02-02 17:33:24 +01:00
migratePrefs ( getPrefsFileVersion ( ) ) ;
}
2023-10-27 21:14:41 +02:00
// Uncomment the line below to force a device key migration, after you updated
// the devicetype.json file
//migrateDeviceTypes();
2015-10-07 23:32:58 +02:00
setupExceptionHandler ( ) ;
2015-06-05 21:46:56 +02:00
2022-08-18 23:03:28 +02:00
Weather . getInstance ( ) . setCacheFile ( getCacheDir ( ) , prefs . getBoolean ( " cache_weather " , true ) ) ;
2016-06-18 23:35:34 +02:00
deviceManager = new DeviceManager ( this ) ;
2017-07-31 22:49:05 +02:00
String language = prefs . getString ( " language " , " default " ) ;
2017-07-31 23:00:02 +02:00
setLanguage ( language ) ;
2016-06-18 23:35:34 +02:00
2015-10-03 23:26:36 +02:00
deviceService = createDeviceService ( ) ;
2018-06-18 20:38:37 +02:00
loadAppsNotifBlackList ( ) ;
loadAppsPebbleBlackList ( ) ;
2015-12-07 23:33:32 +01:00
2022-02-27 18:43:53 +01:00
PeriodicExporter . enablePeriodicExport ( context ) ;
2023-09-02 11:13:14 +02:00
TimeChangeReceiver . scheduleNextDstChange ( context ) ;
2022-02-27 18:43:53 +01:00
2016-05-19 23:58:13 +02:00
if ( isRunningMarshmallowOrLater ( ) ) {
2016-05-20 22:04:30 +02:00
notificationManager = ( NotificationManager ) context . getSystemService ( Context . NOTIFICATION_SERVICE ) ;
2019-06-05 16:00:18 +02:00
if ( isRunningOreoOrLater ( ) ) {
2018-05-16 19:03:06 +02:00
bluetoothStateChangeReceiver = new BluetoothStateChangeReceiver ( ) ;
registerReceiver ( bluetoothStateChangeReceiver , new IntentFilter ( BluetoothAdapter . ACTION_STATE_CHANGED ) ) ;
2018-02-14 21:27:24 +01:00
}
2020-07-28 01:09:59 +02:00
try {
2021-05-14 18:30:54 +02:00
//the following will ensure the notification manager is kept alive
2020-07-28 01:09:59 +02:00
startService ( new Intent ( this , NotificationCollectorMonitorService . class ) ) ;
} catch ( IllegalStateException e ) {
String message = e . toString ( ) ;
if ( message = = null ) {
message = getString ( R . string . _unknown_ ) ;
}
2021-05-14 18:30:54 +02:00
GB . notify ( NOTIFICATION_ID_ERROR ,
2020-07-28 01:09:59 +02:00
new NotificationCompat . Builder ( this , NOTIFICATION_CHANNEL_HIGH_PRIORITY_ID )
. setSmallIcon ( R . drawable . gadgetbridge_img )
. setContentTitle ( getString ( R . string . error_background_service ) )
. setContentText ( getString ( R . string . error_background_service_reason_truncated ) )
. setStyle ( new NotificationCompat . BigTextStyle ( )
. bigText ( getString ( R . string . error_background_service_reason ) + " \" " + message + " \" " ) )
. setPriority ( NotificationCompat . PRIORITY_DEFAULT )
2021-05-14 18:30:54 +02:00
. build ( ) , context ) ;
2020-07-28 01:09:59 +02:00
}
2016-05-19 23:58:13 +02:00
}
2015-05-13 23:15:20 +02:00
}
2016-11-24 21:11:28 +01:00
@Override
public void onTrimMemory ( int level ) {
super . onTrimMemory ( level ) ;
if ( level > = TRIM_MEMORY_BACKGROUND ) {
if ( ! hasBusyDevice ( ) ) {
DBHelper . clearSession ( ) ;
}
}
}
/ * *
* Returns true if at least a single device is busy , e . g synchronizing activity data
* or something similar .
* Note : busy is not the same as connected or initialized !
* /
private boolean hasBusyDevice ( ) {
List < GBDevice > devices = getDeviceManager ( ) . getDevices ( ) ;
2016-12-07 23:13:51 +01:00
for ( GBDevice device : devices ) {
2016-11-24 21:11:28 +01:00
if ( device . isBusy ( ) ) {
return true ;
}
}
2016-11-25 14:53:12 +01:00
return false ;
2016-11-24 21:11:28 +01:00
}
2016-05-26 23:46:21 +02:00
public static void setupLogging ( boolean enabled ) {
logging . setupLogging ( enabled ) ;
}
2019-06-05 16:00:18 +02:00
public static String getLogPath ( ) {
2018-08-28 14:03:57 +02:00
return logging . getLogPath ( ) ;
}
2015-10-07 23:32:58 +02:00
private void setupExceptionHandler ( ) {
LoggingExceptionHandler handler = new LoggingExceptionHandler ( Thread . getDefaultUncaughtExceptionHandler ( ) ) ;
Thread . setDefaultUncaughtExceptionHandler ( handler ) ;
}
2015-05-23 00:45:12 +02:00
public static boolean isFileLoggingEnabled ( ) {
2016-04-25 23:18:55 +02:00
return prefs . getBoolean ( " log_to_file " , false ) ;
2015-05-23 00:45:12 +02:00
}
2017-01-08 15:51:56 +01:00
public static boolean minimizeNotification ( ) {
return prefs . getBoolean ( " minimize_priority " , false ) ;
}
2017-04-27 07:52:11 +02:00
public void setupDatabase ( ) {
DaoMaster . OpenHelper helper ;
GBEnvironment env = GBEnvironment . env ( ) ;
if ( env . isTest ( ) ) {
helper = new DaoMaster . DevOpenHelper ( this , null , null ) ;
} else {
helper = new DBOpenHelper ( this , DATABASE_NAME , null ) ;
}
2016-04-17 19:52:51 +02:00
SQLiteDatabase db = helper . getWritableDatabase ( ) ;
DaoMaster daoMaster = new DaoMaster ( db ) ;
2016-06-14 20:13:08 +02:00
if ( lockHandler = = null ) {
lockHandler = new LockHandler ( ) ;
}
lockHandler . init ( daoMaster , helper ) ;
2016-04-17 19:52:51 +02:00
}
2015-05-01 09:36:10 +02:00
public static Context getContext ( ) {
return context ;
}
2015-05-30 17:28:03 +02:00
2015-10-26 23:32:03 +01:00
/ * *
* Returns the facade for talking to devices . Devices are managed by
* an Android Service and this facade provides access to its functionality .
*
* @return the facade for talking to the service / devices .
* /
2015-08-21 00:58:18 +02:00
public static DeviceService deviceService ( ) {
return deviceService ;
}
2022-08-01 23:04:41 +02:00
/ * *
* Returns the facade for talking to a specific device . Devices are managed by
* an Android Service and this facade provides access to its functionality .
*
* @return the facade for talking to the service / device .
* /
public static DeviceService deviceService ( GBDevice device ) {
return deviceService . forDevice ( device ) ;
}
2015-08-03 01:17:02 +02:00
/ * *
* Returns the DBHandler instance for reading / writing or throws GBException
* when that was not successful
* If acquiring was successful , callers must call # releaseDB when they
* are done ( from the same thread that acquired the lock !
2016-12-07 23:13:51 +01:00
* < p >
2016-06-14 20:13:08 +02:00
* Callers must not hold a reference to the returned instance because it
* will be invalidated at some point .
*
2015-08-03 01:17:02 +02:00
* @return the DBHandler
* @throws GBException
2015-09-24 14:45:21 +02:00
* @see # releaseDB ( )
2015-08-03 01:17:02 +02:00
* /
public static DBHandler acquireDB ( ) throws GBException {
try {
if ( dbLock . tryLock ( 30 , TimeUnit . SECONDS ) ) {
2016-05-16 23:00:04 +02:00
return lockHandler ;
2015-08-03 01:17:02 +02:00
}
} catch ( InterruptedException ex ) {
Log . i ( TAG , " Interrupted while waiting for DB lock " ) ;
}
throw new GBException ( " Unable to access the database. " ) ;
}
/ * *
* Releases the database lock .
2015-09-24 14:45:21 +02:00
*
2015-08-03 01:17:02 +02:00
* @throws IllegalMonitorStateException if the current thread is not owning the lock
* @see # acquireDB ( )
* /
public static void releaseDB ( ) {
dbLock . unlock ( ) ;
2015-05-30 17:28:03 +02:00
}
2016-04-14 16:44:44 +02:00
2016-05-19 16:34:59 +02:00
public static boolean isRunningMarshmallowOrLater ( ) {
return VERSION . SDK_INT > = Build . VERSION_CODES . M ;
}
2019-06-05 16:00:18 +02:00
2018-04-12 22:06:45 +02:00
public static boolean isRunningNougatOrLater ( ) {
return VERSION . SDK_INT > = Build . VERSION_CODES . N ;
}
2016-05-19 16:34:59 +02:00
2019-06-05 16:00:18 +02:00
public static boolean isRunningOreoOrLater ( ) {
2018-02-14 21:27:24 +01:00
return VERSION . SDK_INT > = Build . VERSION_CODES . O ;
}
2021-09-26 21:59:11 +02:00
public static boolean isRunningTenOrLater ( ) {
return VERSION . SDK_INT > = Build . VERSION_CODES . Q ;
}
2022-04-30 21:24:34 +02:00
public static boolean isRunningTwelveOrLater ( ) {
return VERSION . SDK_INT > = 31 ; // Build.VERSION_CODES.S, but our target SDK is lower
}
2021-09-26 21:59:11 +02:00
2020-02-22 10:12:33 +01:00
public static boolean isRunningPieOrLater ( ) {
return VERSION . SDK_INT > = Build . VERSION_CODES . P ;
}
2016-05-19 23:58:13 +02:00
private static boolean isPrioritySender ( int prioritySenders , String number ) {
if ( prioritySenders = = Policy . PRIORITY_SENDERS_ANY ) {
return true ;
} else {
Uri uri = Uri . withAppendedPath ( PhoneLookup . CONTENT_FILTER_URI , Uri . encode ( number ) ) ;
String [ ] projection = new String [ ] { PhoneLookup . _ID , PhoneLookup . STARRED } ;
Cursor cursor = context . getContentResolver ( ) . query ( uri , projection , null , null , null ) ;
boolean exists = false ;
int starred = 0 ;
try {
2016-05-20 22:04:30 +02:00
if ( cursor ! = null & & cursor . moveToFirst ( ) ) {
2016-05-19 23:58:13 +02:00
exists = true ;
starred = cursor . getInt ( cursor . getColumnIndexOrThrow ( PhoneLookup . STARRED ) ) ;
}
} finally {
if ( cursor ! = null ) {
cursor . close ( ) ;
}
2016-05-19 16:34:59 +02:00
}
2016-05-19 23:58:13 +02:00
if ( prioritySenders = = Policy . PRIORITY_SENDERS_CONTACTS & & exists ) {
return true ;
} else if ( prioritySenders = = Policy . PRIORITY_SENDERS_STARRED & & starred = = 1 ) {
return true ;
2016-05-19 16:34:59 +02:00
}
2016-05-19 23:58:13 +02:00
return false ;
2016-05-19 16:34:59 +02:00
}
2016-05-19 23:58:13 +02:00
}
2016-05-20 21:49:25 +02:00
@TargetApi ( Build . VERSION_CODES . M )
2016-05-19 23:58:13 +02:00
public static boolean isPriorityNumber ( int priorityType , String number ) {
NotificationManager . Policy notificationPolicy = notificationManager . getNotificationPolicy ( ) ;
2016-12-07 23:13:51 +01:00
if ( priorityType = = Policy . PRIORITY_CATEGORY_MESSAGES ) {
2016-05-19 23:58:13 +02:00
if ( ( notificationPolicy . priorityCategories & Policy . PRIORITY_CATEGORY_MESSAGES ) = = Policy . PRIORITY_CATEGORY_MESSAGES ) {
return isPrioritySender ( notificationPolicy . priorityMessageSenders , number ) ;
}
} else if ( priorityType = = Policy . PRIORITY_CATEGORY_CALLS ) {
if ( ( notificationPolicy . priorityCategories & Policy . PRIORITY_CATEGORY_CALLS ) = = Policy . PRIORITY_CATEGORY_CALLS ) {
return isPrioritySender ( notificationPolicy . priorityCallSenders , number ) ;
}
2016-05-19 16:34:59 +02:00
}
return false ;
}
2016-05-20 21:49:25 +02:00
@TargetApi ( Build . VERSION_CODES . M )
2016-05-19 23:58:13 +02:00
public static int getGrantedInterruptionFilter ( ) {
2022-05-15 16:51:41 +02:00
if ( GBApplication . isRunningMarshmallowOrLater ( ) & & notificationManager . isNotificationPolicyAccessGranted ( ) ) {
return notificationManager . getCurrentInterruptionFilter ( ) ;
2016-05-19 23:58:13 +02:00
}
return NotificationManager . INTERRUPTION_FILTER_ALL ;
}
2018-06-18 20:38:37 +02:00
private static HashSet < String > apps_notification_blacklist = null ;
2017-04-22 16:59:55 +02:00
2018-06-18 20:38:37 +02:00
public static boolean appIsNotifBlacklisted ( String packageName ) {
if ( apps_notification_blacklist = = null ) {
GB . log ( " appIsNotifBlacklisted: apps_notification_blacklist is null! " , GB . INFO , null ) ;
2017-05-21 21:02:23 +02:00
}
2018-06-18 20:38:37 +02:00
return apps_notification_blacklist ! = null & & apps_notification_blacklist . contains ( packageName ) ;
2017-04-22 16:59:55 +02:00
}
2018-06-18 20:38:37 +02:00
public static void setAppsNotifBlackList ( Set < String > packageNames ) {
2023-03-18 04:27:53 +01:00
setAppsNotifBlackList ( packageNames , sharedPrefs . edit ( ) ) ;
}
public static void setAppsNotifBlackList ( Set < String > packageNames , SharedPreferences . Editor editor ) {
2017-04-22 16:59:55 +02:00
if ( packageNames = = null ) {
2018-06-18 20:38:37 +02:00
GB . log ( " Set null apps_notification_blacklist " , GB . INFO , null ) ;
apps_notification_blacklist = new HashSet < > ( ) ;
2017-04-22 16:59:55 +02:00
} else {
2018-06-18 20:38:37 +02:00
apps_notification_blacklist = new HashSet < > ( packageNames ) ;
2017-04-22 16:59:55 +02:00
}
2018-06-18 20:38:37 +02:00
GB . log ( " New apps_notification_blacklist has " + apps_notification_blacklist . size ( ) + " entries " , GB . INFO , null ) ;
2023-03-18 04:27:53 +01:00
saveAppsNotifBlackList ( editor ) ;
2017-04-22 16:59:55 +02:00
}
2015-09-25 00:53:40 +02:00
2018-06-18 20:38:37 +02:00
private static void loadAppsNotifBlackList ( ) {
GB . log ( " Loading apps_notification_blacklist " , GB . INFO , null ) ;
2019-02-28 21:57:45 +01:00
apps_notification_blacklist = ( HashSet < String > ) sharedPrefs . getStringSet ( GBPrefs . PACKAGE_BLACKLIST , null ) ; // lgtm [java/abstract-to-concrete-cast]
2018-06-18 20:38:37 +02:00
if ( apps_notification_blacklist = = null ) {
apps_notification_blacklist = new HashSet < > ( ) ;
2015-09-25 00:53:40 +02:00
}
2018-06-18 20:38:37 +02:00
GB . log ( " Loaded apps_notification_blacklist has " + apps_notification_blacklist . size ( ) + " entries " , GB . INFO , null ) ;
2015-09-25 00:53:40 +02:00
}
2018-06-18 20:38:37 +02:00
private static void saveAppsNotifBlackList ( ) {
2023-03-18 04:27:53 +01:00
saveAppsNotifBlackList ( sharedPrefs . edit ( ) ) ;
}
private static void saveAppsNotifBlackList ( SharedPreferences . Editor editor ) {
2018-06-18 20:38:37 +02:00
GB . log ( " Saving apps_notification_blacklist with " + apps_notification_blacklist . size ( ) + " entries " , GB . INFO , null ) ;
if ( apps_notification_blacklist . isEmpty ( ) ) {
2017-04-22 16:59:55 +02:00
editor . putStringSet ( GBPrefs . PACKAGE_BLACKLIST , null ) ;
2015-09-25 00:53:40 +02:00
} else {
2018-06-18 20:38:37 +02:00
Prefs . putStringSet ( editor , GBPrefs . PACKAGE_BLACKLIST , apps_notification_blacklist ) ;
2015-09-25 00:53:40 +02:00
}
editor . apply ( ) ;
}
2018-06-18 20:38:37 +02:00
public static void addAppToNotifBlacklist ( String packageName ) {
if ( apps_notification_blacklist . add ( packageName ) ) {
saveAppsNotifBlackList ( ) ;
2015-09-25 00:53:40 +02:00
}
}
2018-06-18 20:38:37 +02:00
public static synchronized void removeFromAppsNotifBlacklist ( String packageName ) {
GB . log ( " Removing from apps_notification_blacklist: " + packageName , GB . INFO , null ) ;
apps_notification_blacklist . remove ( packageName ) ;
saveAppsNotifBlackList ( ) ;
2017-08-18 10:30:19 +02:00
}
2018-06-18 20:38:37 +02:00
private static HashSet < String > apps_pebblemsg_blacklist = null ;
public static boolean appIsPebbleBlacklisted ( String sender ) {
if ( apps_pebblemsg_blacklist = = null ) {
GB . log ( " appIsPebbleBlacklisted: apps_pebblemsg_blacklist is null! " , GB . INFO , null ) ;
}
return apps_pebblemsg_blacklist ! = null & & apps_pebblemsg_blacklist . contains ( sender ) ;
}
public static void setAppsPebbleBlackList ( Set < String > packageNames ) {
2023-03-18 04:27:53 +01:00
setAppsPebbleBlackList ( packageNames , sharedPrefs . edit ( ) ) ;
}
public static void setAppsPebbleBlackList ( Set < String > packageNames , SharedPreferences . Editor editor ) {
2018-06-18 20:38:37 +02:00
if ( packageNames = = null ) {
GB . log ( " Set null apps_pebblemsg_blacklist " , GB . INFO , null ) ;
apps_pebblemsg_blacklist = new HashSet < > ( ) ;
} else {
apps_pebblemsg_blacklist = new HashSet < > ( packageNames ) ;
}
GB . log ( " New apps_pebblemsg_blacklist has " + apps_pebblemsg_blacklist . size ( ) + " entries " , GB . INFO , null ) ;
2023-03-18 04:27:53 +01:00
saveAppsPebbleBlackList ( editor ) ;
2018-06-18 20:38:37 +02:00
}
private static void loadAppsPebbleBlackList ( ) {
GB . log ( " Loading apps_pebblemsg_blacklist " , GB . INFO , null ) ;
2019-02-28 21:57:45 +01:00
apps_pebblemsg_blacklist = ( HashSet < String > ) sharedPrefs . getStringSet ( GBPrefs . PACKAGE_PEBBLEMSG_BLACKLIST , null ) ; // lgtm [java/abstract-to-concrete-cast]
2018-06-18 20:38:37 +02:00
if ( apps_pebblemsg_blacklist = = null ) {
apps_pebblemsg_blacklist = new HashSet < > ( ) ;
}
GB . log ( " Loaded apps_pebblemsg_blacklist has " + apps_pebblemsg_blacklist . size ( ) + " entries " , GB . INFO , null ) ;
}
private static void saveAppsPebbleBlackList ( ) {
2023-03-18 04:27:53 +01:00
saveAppsPebbleBlackList ( sharedPrefs . edit ( ) ) ;
}
private static void saveAppsPebbleBlackList ( SharedPreferences . Editor editor ) {
2018-06-18 20:38:37 +02:00
GB . log ( " Saving apps_pebblemsg_blacklist with " + apps_pebblemsg_blacklist . size ( ) + " entries " , GB . INFO , null ) ;
if ( apps_pebblemsg_blacklist . isEmpty ( ) ) {
editor . putStringSet ( GBPrefs . PACKAGE_PEBBLEMSG_BLACKLIST , null ) ;
} else {
Prefs . putStringSet ( editor , GBPrefs . PACKAGE_PEBBLEMSG_BLACKLIST , apps_pebblemsg_blacklist ) ;
}
editor . apply ( ) ;
}
public static void addAppToPebbleBlacklist ( String packageName ) {
if ( apps_pebblemsg_blacklist . add ( packageNameToPebbleMsgSender ( packageName ) ) ) {
saveAppsPebbleBlackList ( ) ;
}
}
public static synchronized void removeFromAppsPebbleBlacklist ( String packageName ) {
GB . log ( " Removing from apps_pebblemsg_blacklist: " + packageName , GB . INFO , null ) ;
apps_pebblemsg_blacklist . remove ( packageNameToPebbleMsgSender ( packageName ) ) ;
saveAppsPebbleBlackList ( ) ;
}
2019-06-05 16:00:18 +02:00
public static String packageNameToPebbleMsgSender ( String packageName ) {
if ( " eu.siacs.conversations " . equals ( packageName ) ) {
return ( " Conversations " ) ;
} else if ( " net.osmand.plus " . equals ( packageName ) ) {
return ( " OsmAnd " ) ;
}
return packageName ;
2018-06-18 20:38:37 +02:00
}
2015-12-08 23:42:58 +01:00
/ * *
2016-08-31 17:35:28 +02:00
* Deletes both the old Activity database and the new one recreates it with empty tables .
2015-12-14 23:31:31 +01:00
*
2015-12-08 23:42:58 +01:00
* @return true on successful deletion
* /
2016-08-27 22:55:00 +02:00
public static synchronized boolean deleteActivityDatabase ( Context context ) {
2016-05-13 23:47:47 +02:00
// TODO: flush, close, reopen db
2016-05-17 00:51:00 +02:00
if ( lockHandler ! = null ) {
lockHandler . closeDb ( ) ;
}
2017-02-23 22:44:44 +01:00
boolean result = deleteOldActivityDatabase ( context ) ;
2016-08-27 22:51:00 +02:00
result & = getContext ( ) . deleteDatabase ( DATABASE_NAME ) ;
2015-12-08 23:42:58 +01:00
return result ;
}
2015-12-14 23:31:31 +01:00
2016-08-31 17:35:28 +02:00
/ * *
* Deletes the legacy ( pre 0 . 12 ) Activity database
*
* @return true on successful deletion
* /
public static synchronized boolean deleteOldActivityDatabase ( Context context ) {
DBHelper dbHelper = new DBHelper ( context ) ;
boolean result = true ;
2017-02-23 22:44:44 +01:00
if ( dbHelper . existsDB ( " ActivityDatabase " ) ) {
result = getContext ( ) . deleteDatabase ( " ActivityDatabase " ) ;
2016-08-31 17:35:28 +02:00
}
return result ;
}
2016-02-02 17:33:24 +01:00
private int getPrefsFileVersion ( ) {
2016-02-21 13:04:32 +01:00
try {
return Integer . parseInt ( sharedPrefs . getString ( PREFS_VERSION , " 0 " ) ) ; //0 is legacy
} catch ( Exception e ) {
//in version 1 this was an int
return 1 ;
}
2016-02-02 17:33:24 +01:00
}
2019-11-24 18:40:44 +01:00
private void migrateStringPrefToPerDevicePref ( String globalPref , String globalPrefDefault , String perDevicePref , ArrayList < DeviceType > deviceTypes ) {
SharedPreferences . Editor editor = sharedPrefs . edit ( ) ;
String globalPrefValue = prefs . getString ( globalPref , globalPrefDefault ) ;
try ( DBHandler db = acquireDB ( ) ) {
DaoSession daoSession = db . getDaoSession ( ) ;
List < Device > activeDevices = DBHelper . getActiveDevices ( daoSession ) ;
for ( Device dbDevice : activeDevices ) {
SharedPreferences deviceSpecificSharedPrefs = GBApplication . getDeviceSpecificSharedPrefs ( dbDevice . getIdentifier ( ) ) ;
if ( deviceSpecificSharedPrefs ! = null ) {
SharedPreferences . Editor deviceSharedPrefsEdit = deviceSpecificSharedPrefs . edit ( ) ;
2023-10-22 01:33:08 +02:00
DeviceType deviceType = DeviceType . fromName ( dbDevice . getTypeName ( ) ) ;
2019-11-24 18:40:44 +01:00
if ( deviceTypes . contains ( deviceType ) ) {
Log . i ( TAG , " migrating global string preference " + globalPref + " for " + deviceType . name ( ) + " " + dbDevice . getIdentifier ( ) ) ;
deviceSharedPrefsEdit . putString ( perDevicePref , globalPrefValue ) ;
}
deviceSharedPrefsEdit . apply ( ) ;
}
}
editor . remove ( globalPref ) ;
editor . apply ( ) ;
} catch ( Exception e ) {
Log . w ( TAG , " error acquiring DB lock " ) ;
}
}
private void migrateBooleanPrefToPerDevicePref ( String globalPref , Boolean globalPrefDefault , String perDevicePref , ArrayList < DeviceType > deviceTypes ) {
SharedPreferences . Editor editor = sharedPrefs . edit ( ) ;
boolean globalPrefValue = prefs . getBoolean ( globalPref , globalPrefDefault ) ;
try ( DBHandler db = acquireDB ( ) ) {
DaoSession daoSession = db . getDaoSession ( ) ;
List < Device > activeDevices = DBHelper . getActiveDevices ( daoSession ) ;
for ( Device dbDevice : activeDevices ) {
SharedPreferences deviceSpecificSharedPrefs = GBApplication . getDeviceSpecificSharedPrefs ( dbDevice . getIdentifier ( ) ) ;
if ( deviceSpecificSharedPrefs ! = null ) {
SharedPreferences . Editor deviceSharedPrefsEdit = deviceSpecificSharedPrefs . edit ( ) ;
2023-10-22 01:33:08 +02:00
DeviceType deviceType = DeviceType . fromName ( dbDevice . getTypeName ( ) ) ;
2019-11-24 18:40:44 +01:00
if ( deviceTypes . contains ( deviceType ) ) {
Log . i ( TAG , " migrating global boolean preference " + globalPref + " for " + deviceType . name ( ) + " " + dbDevice . getIdentifier ( ) ) ;
deviceSharedPrefsEdit . putBoolean ( perDevicePref , globalPrefValue ) ;
}
deviceSharedPrefsEdit . apply ( ) ;
}
}
editor . remove ( globalPref ) ;
editor . apply ( ) ;
} catch ( Exception e ) {
Log . w ( TAG , " error acquiring DB lock " ) ;
}
}
2023-10-27 21:14:41 +02:00
private void migrateDeviceTypes ( ) {
try ( DBHandler db = acquireDB ( ) ) {
final InputStream inputStream = getAssets ( ) . open ( " migrations/devicetype.json " ) ;
final byte [ ] buffer = new byte [ inputStream . available ( ) ] ;
inputStream . read ( buffer ) ;
inputStream . close ( ) ;
final JSONObject deviceMapping = new JSONObject ( new String ( buffer ) ) ;
final JSONObject deviceIdNameMapping = deviceMapping . getJSONObject ( " by-id " ) ;
final DaoSession daoSession = db . getDaoSession ( ) ;
final List < Device > activeDevices = DBHelper . getActiveDevices ( daoSession ) ;
for ( Device dbDevice : activeDevices ) {
String deviceTypeName = dbDevice . getTypeName ( ) ;
if ( deviceTypeName . isEmpty ( ) | | deviceTypeName . equals ( " UNKNOWN " ) ) {
deviceTypeName = deviceIdNameMapping . optString (
String . valueOf ( dbDevice . getType ( ) ) ,
" UNKNOWN "
) ;
dbDevice . setTypeName ( deviceTypeName ) ;
daoSession . getDeviceDao ( ) . update ( dbDevice ) ;
}
}
} catch ( Exception e ) {
Log . w ( TAG , " error acquiring DB lock " ) ;
}
}
2016-02-02 17:33:24 +01:00
private void migratePrefs ( int oldVersion ) {
2016-02-21 13:04:32 +01:00
SharedPreferences . Editor editor = sharedPrefs . edit ( ) ;
2023-10-22 01:33:08 +02:00
// this comes before all other migrations since the new column DeviceTypeName was added as non-null
if ( oldVersion < 25 ) {
2023-10-27 21:14:41 +02:00
migrateDeviceTypes ( ) ;
2023-10-22 01:33:08 +02:00
}
2019-05-20 16:36:06 +02:00
if ( oldVersion = = 0 ) {
String legacyGender = sharedPrefs . getString ( " mi_user_gender " , null ) ;
String legacyHeight = sharedPrefs . getString ( " mi_user_height_cm " , null ) ;
String legacyWeight = sharedPrefs . getString ( " mi_user_weight_kg " , null ) ;
String legacyYOB = sharedPrefs . getString ( " mi_user_year_of_birth " , null ) ;
if ( legacyGender ! = null ) {
int gender = " male " . equals ( legacyGender ) ? 1 : " female " . equals ( legacyGender ) ? 0 : 2 ;
editor . putString ( ActivityUser . PREF_USER_GENDER , Integer . toString ( gender ) ) ;
editor . remove ( " mi_user_gender " ) ;
}
if ( legacyHeight ! = null ) {
editor . putString ( ActivityUser . PREF_USER_HEIGHT_CM , legacyHeight ) ;
editor . remove ( " mi_user_height_cm " ) ;
}
if ( legacyWeight ! = null ) {
editor . putString ( ActivityUser . PREF_USER_WEIGHT_KG , legacyWeight ) ;
editor . remove ( " mi_user_weight_kg " ) ;
}
if ( legacyYOB ! = null ) {
editor . putString ( ActivityUser . PREF_USER_YEAR_OF_BIRTH , legacyYOB ) ;
editor . remove ( " mi_user_year_of_birth " ) ;
}
}
if ( oldVersion < 2 ) {
//migrate the integer version of gender introduced in version 1 to a string value, needed for the way Android accesses the shared preferences
int legacyGender_1 = 2 ;
try {
legacyGender_1 = sharedPrefs . getInt ( ActivityUser . PREF_USER_GENDER , 2 ) ;
} catch ( Exception e ) {
Log . e ( TAG , " Could not access legacy activity gender " , e ) ;
}
editor . putString ( ActivityUser . PREF_USER_GENDER , Integer . toString ( legacyGender_1 ) ) ;
}
if ( oldVersion < 3 ) {
2019-05-20 21:53:53 +02:00
try ( DBHandler db = acquireDB ( ) ) {
DaoSession daoSession = db . getDaoSession ( ) ;
List < Device > activeDevices = DBHelper . getActiveDevices ( daoSession ) ;
for ( Device dbDevice : activeDevices ) {
2019-09-19 23:18:17 +02:00
SharedPreferences deviceSpecificSharedPrefs = GBApplication . getDeviceSpecificSharedPrefs ( dbDevice . getIdentifier ( ) ) ;
if ( deviceSpecificSharedPrefs ! = null ) {
SharedPreferences . Editor deviceSharedPrefsEdit = deviceSpecificSharedPrefs . edit ( ) ;
2019-05-20 21:53:53 +02:00
String preferenceKey = dbDevice . getIdentifier ( ) + " _lastSportsActivityTimeMillis " ;
long lastSportsActivityTimeMillis = sharedPrefs . getLong ( preferenceKey , 0 ) ;
if ( lastSportsActivityTimeMillis ! = 0 ) {
deviceSharedPrefsEdit . putLong ( " lastSportsActivityTimeMillis " , lastSportsActivityTimeMillis ) ;
editor . remove ( preferenceKey ) ;
}
preferenceKey = dbDevice . getIdentifier ( ) + " _lastSyncTimeMillis " ;
long lastSyncTimeMillis = sharedPrefs . getLong ( preferenceKey , 0 ) ;
if ( lastSyncTimeMillis ! = 0 ) {
deviceSharedPrefsEdit . putLong ( " lastSyncTimeMillis " , lastSyncTimeMillis ) ;
editor . remove ( preferenceKey ) ;
}
2019-05-23 23:24:49 +02:00
String newLanguage = null ;
Set < String > displayItems = null ;
2023-10-22 01:33:08 +02:00
DeviceType deviceType = DeviceType . fromName ( dbDevice . getTypeName ( ) ) ;
2019-05-23 23:24:49 +02:00
if ( deviceType = = AMAZFITBIP | | deviceType = = AMAZFITCOR | | deviceType = = AMAZFITCOR2 ) {
int oldLanguage = prefs . getInt ( " amazfitbip_language " , - 1 ) ;
newLanguage = " auto " ;
String [ ] oldLanguageLookup = { " zh_CN " , " zh_TW " , " en_US " , " es_ES " , " ru_RU " , " de_DE " , " it_IT " , " fr_FR " , " tr_TR " } ;
if ( oldLanguage > = 0 & & oldLanguage < oldLanguageLookup . length ) {
newLanguage = oldLanguageLookup [ oldLanguage ] ;
}
}
if ( deviceType = = AMAZFITBIP | | deviceType = = AMAZFITCOR ) {
deviceSharedPrefsEdit . putString ( " disconnect_notification " , prefs . getString ( " disconnect_notification " , " off " ) ) ;
deviceSharedPrefsEdit . putString ( " disconnect_notification_start " , prefs . getString ( " disconnect_notification_start " , " 8:00 " ) ) ;
deviceSharedPrefsEdit . putString ( " disconnect_notification_end " , prefs . getString ( " disconnect_notification_end " , " 22:00 " ) ) ;
}
2023-08-27 23:47:19 +02:00
if ( deviceType = = MIBAND2 | | deviceType = = MIBAND2_HRX | | deviceType = = MIBAND3 ) {
2019-06-05 12:48:33 +02:00
deviceSharedPrefsEdit . putString ( " do_not_disturb " , prefs . getString ( " mi2_do_not_disturb " , " off " ) ) ;
deviceSharedPrefsEdit . putString ( " do_not_disturb_start " , prefs . getString ( " mi2_do_not_disturb_start " , " 1:00 " ) ) ;
deviceSharedPrefsEdit . putString ( " do_not_disturb_end " , prefs . getString ( " mi2_do_not_disturb_end " , " 6:00 " ) ) ;
2019-05-24 23:46:20 +02:00
}
2019-06-05 16:00:18 +02:00
if ( dbDevice . getManufacturer ( ) . equals ( " Huami " ) ) {
deviceSharedPrefsEdit . putString ( " activate_display_on_lift_wrist " , prefs . getString ( " activate_display_on_lift_wrist " , " off " ) ) ;
deviceSharedPrefsEdit . putString ( " display_on_lift_start " , prefs . getString ( " display_on_lift_start " , " 0:00 " ) ) ;
deviceSharedPrefsEdit . putString ( " display_on_lift_end " , prefs . getString ( " display_on_lift_end " , " 0:00 " ) ) ;
}
2019-05-23 23:24:49 +02:00
switch ( deviceType ) {
2019-06-05 12:48:33 +02:00
case MIBAND :
deviceSharedPrefsEdit . putBoolean ( " low_latency_fw_update " , prefs . getBoolean ( " mi_low_latency_fw_update " , true ) ) ;
2019-11-24 18:40:44 +01:00
deviceSharedPrefsEdit . putString ( " device_time_offset_hours " , String . valueOf ( prefs . getInt ( " mi_device_time_offset_hours " , 0 ) ) ) ;
2019-06-05 12:48:33 +02:00
break ;
2019-05-20 21:53:53 +02:00
case AMAZFITCOR :
2019-05-23 23:24:49 +02:00
displayItems = prefs . getStringSet ( " cor_display_items " , null ) ;
break ;
2019-05-20 21:53:53 +02:00
case AMAZFITBIP :
2019-05-23 23:24:49 +02:00
displayItems = prefs . getStringSet ( " bip_display_items " , null ) ;
break ;
case MIBAND2 :
2023-08-27 23:47:19 +02:00
case MIBAND2_HRX :
2019-05-23 23:24:49 +02:00
displayItems = prefs . getStringSet ( " mi2_display_items " , null ) ;
deviceSharedPrefsEdit . putBoolean ( " mi2_enable_text_notifications " , prefs . getBoolean ( " mi2_enable_text_notifications " , true ) ) ;
deviceSharedPrefsEdit . putString ( " mi2_dateformat " , prefs . getString ( " mi2_dateformat " , " dateformat_time " ) ) ;
2019-06-05 16:00:18 +02:00
deviceSharedPrefsEdit . putBoolean ( " rotate_wrist_to_cycle_info " , prefs . getBoolean ( " mi2_rotate_wrist_to_switch_info " , false ) ) ;
2019-05-20 21:53:53 +02:00
break ;
case MIBAND3 :
2019-05-23 23:24:49 +02:00
newLanguage = prefs . getString ( " miband3_language " , " auto " ) ;
displayItems = prefs . getStringSet ( " miband3_display_items " , null ) ;
deviceSharedPrefsEdit . putBoolean ( " swipe_unlock " , prefs . getBoolean ( " mi3_band_screen_unlock " , false ) ) ;
deviceSharedPrefsEdit . putString ( " night_mode " , prefs . getString ( " mi3_night_mode " , " off " ) ) ;
deviceSharedPrefsEdit . putString ( " night_mode_start " , prefs . getString ( " mi3_night_mode_start " , " 16:00 " ) ) ;
deviceSharedPrefsEdit . putString ( " night_mode_end " , prefs . getString ( " mi3_night_mode_end " , " 7:00 " ) ) ;
}
if ( displayItems ! = null ) {
deviceSharedPrefsEdit . putStringSet ( " display_items " , displayItems ) ;
}
if ( newLanguage ! = null ) {
deviceSharedPrefsEdit . putString ( " language " , newLanguage ) ;
2019-05-20 21:53:53 +02:00
}
2019-09-19 23:18:17 +02:00
deviceSharedPrefsEdit . apply ( ) ;
2019-05-20 16:36:06 +02:00
}
2019-05-20 21:53:53 +02:00
}
editor . remove ( " amazfitbip_language " ) ;
2019-05-23 23:24:49 +02:00
editor . remove ( " bip_display_items " ) ;
editor . remove ( " cor_display_items " ) ;
editor . remove ( " disconnect_notification " ) ;
editor . remove ( " disconnect_notification_start " ) ;
editor . remove ( " disconnect_notification_end " ) ;
2019-06-05 16:00:18 +02:00
editor . remove ( " activate_display_on_lift_wrist " ) ;
editor . remove ( " display_on_lift_start " ) ;
editor . remove ( " display_on_lift_end " ) ;
2019-06-05 12:48:33 +02:00
editor . remove ( " mi_low_latency_fw_update " ) ;
2019-06-05 14:11:44 +02:00
editor . remove ( " mi_device_time_offset_hours " ) ;
2019-05-24 23:46:20 +02:00
editor . remove ( " mi2_do_not_disturb " ) ;
editor . remove ( " mi2_do_not_disturb_start " ) ;
editor . remove ( " mi2_do_not_disturb_end " ) ;
2019-05-23 23:24:49 +02:00
editor . remove ( " mi2_dateformat " ) ;
editor . remove ( " mi2_display_items " ) ;
2019-06-05 16:00:18 +02:00
editor . remove ( " mi2_rotate_wrist_to_switch_info " ) ;
2019-05-23 23:24:49 +02:00
editor . remove ( " mi2_enable_text_notifications " ) ;
editor . remove ( " mi3_band_screen_unlock " ) ;
editor . remove ( " mi3_night_mode " ) ;
editor . remove ( " mi3_night_mode_start " ) ;
editor . remove ( " mi3_night_mode_end " ) ;
editor . remove ( " miband3_language " ) ;
2019-05-20 21:53:53 +02:00
} catch ( Exception e ) {
Log . w ( TAG , " error acquiring DB lock " ) ;
2019-05-20 16:36:06 +02:00
}
2016-02-02 17:33:24 +01:00
}
2019-06-18 22:07:43 +02:00
if ( oldVersion < 4 ) {
try ( DBHandler db = acquireDB ( ) ) {
DaoSession daoSession = db . getDaoSession ( ) ;
List < Device > activeDevices = DBHelper . getActiveDevices ( daoSession ) ;
for ( Device dbDevice : activeDevices ) {
SharedPreferences deviceSharedPrefs = GBApplication . getDeviceSpecificSharedPrefs ( dbDevice . getIdentifier ( ) ) ;
SharedPreferences . Editor deviceSharedPrefsEdit = deviceSharedPrefs . edit ( ) ;
2023-10-22 01:33:08 +02:00
DeviceType deviceType = DeviceType . fromName ( dbDevice . getTypeName ( ) ) ;
2019-06-18 22:07:43 +02:00
if ( deviceType = = MIBAND ) {
int deviceTimeOffsetHours = deviceSharedPrefs . getInt ( " device_time_offset_hours " , 0 ) ;
deviceSharedPrefsEdit . putString ( " device_time_offset_hours " , Integer . toString ( deviceTimeOffsetHours ) ) ;
}
deviceSharedPrefsEdit . apply ( ) ;
}
} catch ( Exception e ) {
Log . w ( TAG , " error acquiring DB lock " ) ;
}
}
2019-10-05 22:35:30 +02:00
if ( oldVersion < 5 ) {
try ( DBHandler db = acquireDB ( ) ) {
DaoSession daoSession = db . getDaoSession ( ) ;
List < Device > activeDevices = DBHelper . getActiveDevices ( daoSession ) ;
for ( Device dbDevice : activeDevices ) {
SharedPreferences deviceSpecificSharedPrefs = GBApplication . getDeviceSpecificSharedPrefs ( dbDevice . getIdentifier ( ) ) ;
if ( deviceSpecificSharedPrefs ! = null ) {
SharedPreferences . Editor deviceSharedPrefsEdit = deviceSpecificSharedPrefs . edit ( ) ;
2023-10-22 01:33:08 +02:00
DeviceType deviceType = DeviceType . fromName ( dbDevice . getTypeName ( ) ) ;
2019-10-05 22:35:30 +02:00
String newWearside = null ;
String newOrientation = null ;
String newTimeformat = null ;
switch ( deviceType ) {
case AMAZFITBIP :
case AMAZFITCOR :
case AMAZFITCOR2 :
case MIBAND :
case MIBAND2 :
2023-08-27 23:47:19 +02:00
case MIBAND2_HRX :
2019-10-05 22:35:30 +02:00
case MIBAND3 :
case MIBAND4 :
newWearside = prefs . getString ( " mi_wearside " , " left " ) ;
break ;
2020-07-28 18:16:56 +02:00
case MIBAND5 :
newWearside = prefs . getString ( " mi_wearside " , " left " ) ;
break ;
2019-10-05 22:35:30 +02:00
case HPLUS :
newWearside = prefs . getString ( " hplus_wrist " , " left " ) ;
newTimeformat = prefs . getString ( " hplus_timeformat " , " 24h " ) ;
break ;
case ID115 :
newWearside = prefs . getString ( " id115_wrist " , " left " ) ;
newOrientation = prefs . getString ( " id115_screen_orientation " , " horizontal " ) ;
break ;
case ZETIME :
newWearside = prefs . getString ( " zetime_wrist " , " left " ) ;
newTimeformat = prefs . getInt ( " zetime_timeformat " , 1 ) = = 2 ? " am/pm " : " 24h " ;
break ;
}
if ( newWearside ! = null ) {
deviceSharedPrefsEdit . putString ( " wearlocation " , newWearside ) ;
}
if ( newOrientation ! = null ) {
deviceSharedPrefsEdit . putString ( " screen_orientation " , newOrientation ) ;
}
if ( newTimeformat ! = null ) {
deviceSharedPrefsEdit . putString ( " timeformat " , newTimeformat ) ;
}
deviceSharedPrefsEdit . apply ( ) ;
}
}
editor . remove ( " hplus_timeformat " ) ;
editor . remove ( " hplus_wrist " ) ;
editor . remove ( " id115_wrist " ) ;
editor . remove ( " id115_screen_orientation " ) ;
editor . remove ( " mi_wearside " ) ;
editor . remove ( " zetime_timeformat " ) ;
editor . remove ( " zetime_wrist " ) ;
} catch ( Exception e ) {
Log . w ( TAG , " error acquiring DB lock " ) ;
}
}
2019-11-24 18:40:44 +01:00
if ( oldVersion < 6 ) {
migrateBooleanPrefToPerDevicePref ( " mi2_enable_button_action " , false , " button_action_enable " , new ArrayList < > ( Collections . singletonList ( MIBAND2 ) ) ) ;
migrateBooleanPrefToPerDevicePref ( " mi2_button_action_vibrate " , false , " button_action_vibrate " , new ArrayList < > ( Collections . singletonList ( MIBAND2 ) ) ) ;
migrateStringPrefToPerDevicePref ( " mi_button_press_count " , " 6 " , " button_action_press_count " , new ArrayList < > ( Collections . singletonList ( MIBAND2 ) ) ) ;
migrateStringPrefToPerDevicePref ( " mi_button_press_count_max_delay " , " 2000 " , " button_action_press_max_interval " , new ArrayList < > ( Collections . singletonList ( MIBAND2 ) ) ) ;
migrateStringPrefToPerDevicePref ( " mi_button_press_count_match_delay " , " 0 " , " button_action_broadcast_delay " , new ArrayList < > ( Collections . singletonList ( MIBAND2 ) ) ) ;
migrateStringPrefToPerDevicePref ( " mi_button_press_broadcast " , " nodomain.freeyourgadget.gadgetbridge.ButtonPressed " , " button_action_broadcast " , new ArrayList < > ( Collections . singletonList ( MIBAND2 ) ) ) ;
}
2019-12-14 23:43:54 +01:00
if ( oldVersion < 7 ) {
2020-10-13 00:25:43 +02:00
migrateStringPrefToPerDevicePref ( " mi_reserve_alarm_calendar " , " 0 " , " reserve_alarms_calendar " , new ArrayList < > ( Arrays . asList ( MIBAND , MIBAND2 ) ) ) ;
}
if ( oldVersion < 8 ) {
for ( int i = 1 ; i < = 16 ; i + + ) {
String message = prefs . getString ( " canned_message_dismisscall_ " + i , null ) ;
if ( message ! = null ) {
migrateStringPrefToPerDevicePref ( " canned_message_dismisscall_ " + i , " " , " canned_message_dismisscall_ " + i , new ArrayList < > ( Collections . singletonList ( PEBBLE ) ) ) ;
}
}
for ( int i = 1 ; i < = 16 ; i + + ) {
String message = prefs . getString ( " canned_reply_ " + i , null ) ;
if ( message ! = null ) {
migrateStringPrefToPerDevicePref ( " canned_reply_ " + i , " " , " canned_reply_ " + i , new ArrayList < > ( Collections . singletonList ( PEBBLE ) ) ) ;
}
}
2019-12-14 23:43:54 +01:00
}
2021-03-21 19:46:30 +01:00
if ( oldVersion < 9 ) {
try ( DBHandler db = acquireDB ( ) ) {
DaoSession daoSession = db . getDaoSession ( ) ;
List < Device > activeDevices = DBHelper . getActiveDevices ( daoSession ) ;
migrateBooleanPrefToPerDevicePref ( " transliteration " , false , " pref_transliteration_enabled " , ( ArrayList ) activeDevices ) ;
Log . w ( TAG , " migrating transliteration settings " ) ;
} catch ( Exception e ) {
Log . w ( TAG , " error acquiring DB lock and migrating prefs " ) ;
}
}
2021-11-06 10:07:16 +01:00
if ( oldVersion < 10 ) {
//migrate the string version of pref_galaxy_buds_ambient_volume to int due to transition to SeekBarPreference
try ( DBHandler db = acquireDB ( ) ) {
DaoSession daoSession = db . getDaoSession ( ) ;
List < Device > activeDevices = DBHelper . getActiveDevices ( daoSession ) ;
for ( Device dbDevice : activeDevices ) {
SharedPreferences deviceSharedPrefs = GBApplication . getDeviceSpecificSharedPrefs ( dbDevice . getIdentifier ( ) ) ;
SharedPreferences . Editor deviceSharedPrefsEdit = deviceSharedPrefs . edit ( ) ;
2023-10-22 01:33:08 +02:00
DeviceType deviceType = DeviceType . fromName ( dbDevice . getTypeName ( ) ) ;
2019-10-05 22:35:30 +02:00
2021-11-06 10:07:16 +01:00
if ( deviceType = = GALAXY_BUDS ) {
GB . log ( " migrating Galaxy Buds volume " , GB . INFO , null ) ;
String volume = deviceSharedPrefs . getString ( DeviceSettingsPreferenceConst . PREF_GALAXY_BUDS_AMBIENT_VOLUME , " 1 " ) ;
deviceSharedPrefsEdit . putInt ( DeviceSettingsPreferenceConst . PREF_GALAXY_BUDS_AMBIENT_VOLUME , Integer . parseInt ( volume ) ) ;
}
deviceSharedPrefsEdit . apply ( ) ;
}
} catch ( Exception e ) {
Log . w ( TAG , " error acquiring DB lock " ) ;
}
}
2022-03-10 00:06:41 +01:00
if ( oldVersion < 11 ) {
try ( DBHandler db = acquireDB ( ) ) {
DaoSession daoSession = db . getDaoSession ( ) ;
List < Device > activeDevices = DBHelper . getActiveDevices ( daoSession ) ;
for ( Device dbDevice : activeDevices ) {
SharedPreferences deviceSharedPrefs = GBApplication . getDeviceSpecificSharedPrefs ( dbDevice . getIdentifier ( ) ) ;
SharedPreferences . Editor deviceSharedPrefsEdit = deviceSharedPrefs . edit ( ) ;
2023-10-22 01:33:08 +02:00
DeviceType deviceType = DeviceType . fromName ( dbDevice . getTypeName ( ) ) ;
2022-03-10 00:06:41 +01:00
if ( deviceType = = WATCHXPLUS | | deviceType = = FITPRO | | deviceType = = LEFUN ) {
deviceSharedPrefsEdit . putBoolean ( " inactivity_warnings_enable " , deviceSharedPrefs . getBoolean ( " pref_longsit_switch " , false ) ) ;
deviceSharedPrefsEdit . remove ( " pref_longsit_switch " ) ;
}
if ( deviceType = = WATCHXPLUS | | deviceType = = FITPRO ) {
deviceSharedPrefsEdit . putString ( " inactivity_warnings_start " , deviceSharedPrefs . getString ( " pref_longsit_start " , " 06:00 " ) ) ;
deviceSharedPrefsEdit . putString ( " inactivity_warnings_end " , deviceSharedPrefs . getString ( " pref_longsit_end " , " 23:00 " ) ) ;
deviceSharedPrefsEdit . remove ( " pref_longsit_start " ) ;
deviceSharedPrefsEdit . remove ( " pref_longsit_end " ) ;
}
if ( deviceType = = WATCHXPLUS | | deviceType = = LEFUN ) {
deviceSharedPrefsEdit . putString ( " inactivity_warnings_threshold " , deviceSharedPrefs . getString ( " pref_longsit_period " , " 60 " ) ) ;
deviceSharedPrefsEdit . remove ( " pref_longsit_period " ) ;
}
if ( deviceType = = TLW64 ) {
deviceSharedPrefsEdit . putBoolean ( " inactivity_warnings_enable_noshed " , deviceSharedPrefs . getBoolean ( " screen_longsit_noshed " , false ) ) ;
deviceSharedPrefsEdit . remove ( " screen_longsit_noshed " ) ;
}
if ( dbDevice . getManufacturer ( ) . equals ( " Huami " ) ) {
editor . putBoolean ( " inactivity_warnings_dnd " , prefs . getBoolean ( " mi2_inactivity_warnings_dnd " , false ) ) ;
editor . putString ( " inactivity_warnings_dnd_start " , prefs . getString ( " mi2_inactivity_warnings_dnd_start " , " 12:00 " ) ) ;
editor . putString ( " inactivity_warnings_dnd_end " , prefs . getString ( " mi2_inactivity_warnings_dnd_end " , " 14:00 " ) ) ;
editor . putBoolean ( " inactivity_warnings_enable " , prefs . getBoolean ( " mi2_inactivity_warnings " , false ) ) ;
editor . putInt ( " inactivity_warnings_threshold " , prefs . getInt ( " mi2_inactivity_warnings_threshold " , 60 ) ) ;
editor . putString ( " inactivity_warnings_start " , prefs . getString ( " mi2_inactivity_warnings_start " , " 06:00 " ) ) ;
editor . putString ( " inactivity_warnings_end " , prefs . getString ( " mi2_inactivity_warnings_end " , " 22:00 " ) ) ;
}
switch ( deviceType ) {
case LEFUN :
deviceSharedPrefsEdit . putString ( " language " , deviceSharedPrefs . getString ( " pref_lefun_interface_language " , " 0 " ) ) ;
deviceSharedPrefsEdit . remove ( " pref_lefun_interface_language " ) ;
break ;
case FITPRO :
deviceSharedPrefsEdit . putString ( " inactivity_warnings_threshold " , deviceSharedPrefs . getString ( " pref_longsit_period " , " 4 " ) ) ;
deviceSharedPrefsEdit . remove ( " pref_longsit_period " ) ;
break ;
case ZETIME :
editor . putString ( " do_not_disturb " , prefs . getString ( " zetime_do_not_disturb " , " off " ) ) ;
editor . putString ( " do_not_disturb_start " , prefs . getString ( " zetime_do_not_disturb_start " , " 22:00 " ) ) ;
editor . putString ( " do_not_disturb_end " , prefs . getString ( " zetime_do_not_disturb_end " , " 07:00 " ) ) ;
editor . putBoolean ( " inactivity_warnings_enable " , prefs . getBoolean ( " zetime_inactivity_warnings " , false ) ) ;
editor . putString ( " inactivity_warnings_start " , prefs . getString ( " zetime_inactivity_warnings_start " , " 06:00 " ) ) ;
editor . putString ( " inactivity_warnings_end " , prefs . getString ( " zetime_inactivity_warnings_end " , " 22:00 " ) ) ;
editor . putInt ( " inactivity_warnings_threshold " , prefs . getInt ( " zetime_inactivity_warnings_threshold " , 60 ) ) ;
editor . putBoolean ( " inactivity_warnings_mo " , prefs . getBoolean ( " zetime_prefs_inactivity_repetitions_mo " , false ) ) ;
editor . putBoolean ( " inactivity_warnings_tu " , prefs . getBoolean ( " zetime_prefs_inactivity_repetitions_tu " , false ) ) ;
editor . putBoolean ( " inactivity_warnings_we " , prefs . getBoolean ( " zetime_prefs_inactivity_repetitions_we " , false ) ) ;
editor . putBoolean ( " inactivity_warnings_th " , prefs . getBoolean ( " zetime_prefs_inactivity_repetitions_th " , false ) ) ;
editor . putBoolean ( " inactivity_warnings_fr " , prefs . getBoolean ( " zetime_prefs_inactivity_repetitions_fr " , false ) ) ;
editor . putBoolean ( " inactivity_warnings_sa " , prefs . getBoolean ( " zetime_prefs_inactivity_repetitions_sa " , false ) ) ;
editor . putBoolean ( " inactivity_warnings_su " , prefs . getBoolean ( " zetime_prefs_inactivity_repetitions_su " , false ) ) ;
break ;
}
deviceSharedPrefsEdit . apply ( ) ;
}
editor . putInt ( " fitness_goal " , prefs . getInt ( " mi_fitness_goal " , 8000 ) ) ;
editor . remove ( " zetime_do_not_disturb " ) ;
editor . remove ( " zetime_do_not_disturb_start " ) ;
editor . remove ( " zetime_do_not_disturb_end " ) ;
editor . remove ( " zetime_inactivity_warnings " ) ;
editor . remove ( " zetime_inactivity_warnings_start " ) ;
editor . remove ( " zetime_inactivity_warnings_end " ) ;
editor . remove ( " zetime_inactivity_warnings_threshold " ) ;
editor . remove ( " zetime_prefs_inactivity_repetitions_mo " ) ;
editor . remove ( " zetime_prefs_inactivity_repetitions_tu " ) ;
editor . remove ( " zetime_prefs_inactivity_repetitions_we " ) ;
editor . remove ( " zetime_prefs_inactivity_repetitions_th " ) ;
editor . remove ( " zetime_prefs_inactivity_repetitions_fr " ) ;
editor . remove ( " zetime_prefs_inactivity_repetitions_sa " ) ;
editor . remove ( " zetime_prefs_inactivity_repetitions_su " ) ;
editor . remove ( " mi2_inactivity_warnings_dnd " ) ;
editor . remove ( " mi2_inactivity_warnings_dnd_start " ) ;
editor . remove ( " mi2_inactivity_warnings_dnd_end " ) ;
editor . remove ( " mi2_inactivity_warnings " ) ;
editor . remove ( " mi2_inactivity_warnings_threshold " ) ;
editor . remove ( " mi2_inactivity_warnings_start " ) ;
editor . remove ( " mi2_inactivity_warnings_end " ) ;
editor . remove ( " mi_fitness_goal " ) ;
} catch ( Exception e ) {
Log . w ( TAG , " error acquiring DB lock " ) ;
}
}
2022-05-07 21:45:33 +02:00
if ( oldVersion < 12 ) {
// Convert preferences that were wrongly migrated to int, since Android saves them as Strings internally
editor . putString ( " inactivity_warnings_threshold " , String . valueOf ( prefs . getInt ( " inactivity_warnings_threshold " , 60 ) ) ) ;
editor . putString ( " fitness_goal " , String . valueOf ( prefs . getInt ( " fitness_goal " , 8000 ) ) ) ;
}
2022-05-14 22:08:32 +02:00
if ( oldVersion < 13 ) {
try ( DBHandler db = acquireDB ( ) ) {
final DaoSession daoSession = db . getDaoSession ( ) ;
final List < Device > activeDevices = DBHelper . getActiveDevices ( daoSession ) ;
for ( Device dbDevice : activeDevices ) {
final SharedPreferences deviceSharedPrefs = GBApplication . getDeviceSpecificSharedPrefs ( dbDevice . getIdentifier ( ) ) ;
final SharedPreferences . Editor deviceSharedPrefsEdit = deviceSharedPrefs . edit ( ) ;
if ( dbDevice . getManufacturer ( ) . equals ( " Huami " ) ) {
deviceSharedPrefsEdit . putBoolean ( " inactivity_warnings_enable " , prefs . getBoolean ( " inactivity_warnings_enable " , false ) ) ;
deviceSharedPrefsEdit . putString ( " inactivity_warnings_threshold " , prefs . getString ( " inactivity_warnings_threshold " , " 60 " ) ) ;
deviceSharedPrefsEdit . putString ( " inactivity_warnings_start " , prefs . getString ( " inactivity_warnings_start " , " 06:00 " ) ) ;
deviceSharedPrefsEdit . putString ( " inactivity_warnings_end " , prefs . getString ( " inactivity_warnings_end " , " 22:00 " ) ) ;
deviceSharedPrefsEdit . putBoolean ( " inactivity_warnings_dnd " , prefs . getBoolean ( " inactivity_warnings_dnd " , false ) ) ;
deviceSharedPrefsEdit . putString ( " inactivity_warnings_dnd_start " , prefs . getString ( " inactivity_warnings_dnd_start " , " 12:00 " ) ) ;
deviceSharedPrefsEdit . putString ( " inactivity_warnings_dnd_end " , prefs . getString ( " inactivity_warnings_dnd_end " , " 14:00 " ) ) ;
deviceSharedPrefsEdit . putBoolean ( " fitness_goal_notification " , prefs . getBoolean ( " mi2_goal_notification " , false ) ) ;
}
// Not removing the first 4 preferences since they're still used by some devices (ZeTime)
editor . remove ( " inactivity_warnings_dnd " ) ;
editor . remove ( " inactivity_warnings_dnd_start " ) ;
editor . remove ( " inactivity_warnings_dnd_end " ) ;
editor . remove ( " mi2_goal_notification " ) ;
deviceSharedPrefsEdit . apply ( ) ;
}
} catch ( Exception e ) {
Log . w ( TAG , " error acquiring DB lock " ) ;
}
}
2022-05-15 23:15:10 +02:00
if ( oldVersion < 14 ) {
try ( DBHandler db = acquireDB ( ) ) {
final DaoSession daoSession = db . getDaoSession ( ) ;
final List < Device > activeDevices = DBHelper . getActiveDevices ( daoSession ) ;
for ( Device dbDevice : activeDevices ) {
final SharedPreferences deviceSharedPrefs = GBApplication . getDeviceSpecificSharedPrefs ( dbDevice . getIdentifier ( ) ) ;
final SharedPreferences . Editor deviceSharedPrefsEdit = deviceSharedPrefs . edit ( ) ;
if ( DeviceType . MIBAND . equals ( dbDevice . getType ( ) ) | | dbDevice . getManufacturer ( ) . equals ( " Huami " ) ) {
deviceSharedPrefsEdit . putBoolean ( " heartrate_sleep_detection " , prefs . getBoolean ( " mi_hr_sleep_detection " , false ) ) ;
deviceSharedPrefsEdit . putString ( " heartrate_measurement_interval " , prefs . getString ( " heartrate_measurement_interval " , " 0 " ) ) ;
}
// Not removing heartrate_measurement_interval since it's still used by some devices (ZeTime)
editor . remove ( " mi_hr_sleep_detection " ) ;
deviceSharedPrefsEdit . apply ( ) ;
}
} catch ( Exception e ) {
Log . w ( TAG , " error acquiring DB lock " ) ;
}
}
2022-05-29 21:18:53 +02:00
if ( oldVersion < 15 ) {
try ( DBHandler db = acquireDB ( ) ) {
final DaoSession daoSession = db . getDaoSession ( ) ;
final List < Device > activeDevices = DBHelper . getActiveDevices ( daoSession ) ;
for ( Device dbDevice : activeDevices ) {
final SharedPreferences deviceSharedPrefs = GBApplication . getDeviceSpecificSharedPrefs ( dbDevice . getIdentifier ( ) ) ;
final SharedPreferences . Editor deviceSharedPrefsEdit = deviceSharedPrefs . edit ( ) ;
if ( DeviceType . FITPRO . equals ( dbDevice . getType ( ) ) ) {
editor . remove ( " inactivity_warnings_threshold " ) ;
deviceSharedPrefsEdit . apply ( ) ;
}
}
} catch ( Exception e ) {
Log . w ( TAG , " error acquiring DB lock " ) ;
}
}
2022-06-15 01:54:44 +02:00
if ( oldVersion < 16 ) {
// If transliteration was enabled for a device, migrate it to the per-language setting
2023-10-31 07:33:06 +01:00
final String defaultLanguagesIfEnabled = " extended_ascii,common_symbols,scandinavian,german,russian,hebrew,greek,ukranian,arabic,persian,latvian,lithuanian,polish,estonian,icelandic,czech,turkish,bengali,korean,hungarian " ;
2022-06-15 01:54:44 +02:00
try ( DBHandler db = acquireDB ( ) ) {
final DaoSession daoSession = db . getDaoSession ( ) ;
final List < Device > activeDevices = DBHelper . getActiveDevices ( daoSession ) ;
for ( Device dbDevice : activeDevices ) {
final SharedPreferences deviceSharedPrefs = GBApplication . getDeviceSpecificSharedPrefs ( dbDevice . getIdentifier ( ) ) ;
final SharedPreferences . Editor deviceSharedPrefsEdit = deviceSharedPrefs . edit ( ) ;
if ( deviceSharedPrefs . getBoolean ( " pref_transliteration_enabled " , false ) ) {
deviceSharedPrefsEdit . putString ( " pref_transliteration_languages " , defaultLanguagesIfEnabled ) ;
}
deviceSharedPrefsEdit . remove ( " pref_transliteration_enabled " ) ;
deviceSharedPrefsEdit . apply ( ) ;
}
} catch ( Exception e ) {
Log . w ( TAG , " error acquiring DB lock " ) ;
}
}
2022-06-16 23:28:17 +02:00
if ( oldVersion < 17 ) {
final HashSet < String > calendarBlacklist = ( HashSet < String > ) prefs . getStringSet ( GBPrefs . CALENDAR_BLACKLIST , null ) ;
try ( DBHandler db = acquireDB ( ) ) {
final DaoSession daoSession = db . getDaoSession ( ) ;
final List < Device > activeDevices = DBHelper . getActiveDevices ( daoSession ) ;
for ( Device dbDevice : activeDevices ) {
final SharedPreferences deviceSharedPrefs = GBApplication . getDeviceSpecificSharedPrefs ( dbDevice . getIdentifier ( ) ) ;
final SharedPreferences . Editor deviceSharedPrefsEdit = deviceSharedPrefs . edit ( ) ;
deviceSharedPrefsEdit . putBoolean ( " sync_calendar " , prefs . getBoolean ( " enable_calendar_sync " , true ) ) ;
if ( calendarBlacklist ! = null ) {
Prefs . putStringSet ( deviceSharedPrefsEdit , GBPrefs . CALENDAR_BLACKLIST , calendarBlacklist ) ;
}
deviceSharedPrefsEdit . apply ( ) ;
}
} catch ( Exception e ) {
Log . w ( TAG , " error acquiring DB lock " ) ;
}
editor . remove ( GBPrefs . CALENDAR_BLACKLIST ) ;
}
2022-09-26 19:20:12 +02:00
if ( oldVersion < 18 ) {
// Migrate the default value for Huami find band vibration pattern
try ( DBHandler db = acquireDB ( ) ) {
final DaoSession daoSession = db . getDaoSession ( ) ;
final List < Device > activeDevices = DBHelper . getActiveDevices ( daoSession ) ;
for ( Device dbDevice : activeDevices ) {
if ( ! dbDevice . getManufacturer ( ) . equals ( " Huami " ) ) {
continue ;
}
final SharedPreferences deviceSharedPrefs = GBApplication . getDeviceSpecificSharedPrefs ( dbDevice . getIdentifier ( ) ) ;
final SharedPreferences . Editor deviceSharedPrefsEdit = deviceSharedPrefs . edit ( ) ;
deviceSharedPrefsEdit . putString ( " huami_vibration_profile_find_band " , " long " ) ;
deviceSharedPrefsEdit . putString ( " huami_vibration_count_find_band " , " 1 " ) ;
deviceSharedPrefsEdit . apply ( ) ;
}
} catch ( Exception e ) {
Log . w ( TAG , " error acquiring DB lock " ) ;
}
2023-06-17 18:28:11 +02:00
}
if ( oldVersion < 19 ) {
//remove old ble scanning prefences, now unsupported
editor . remove ( " disable_new_ble_scanning " ) ;
}
if ( oldVersion < 20 ) {
// Add the new stress tab to all devices
try ( DBHandler db = acquireDB ( ) ) {
final DaoSession daoSession = db . getDaoSession ( ) ;
final List < Device > activeDevices = DBHelper . getActiveDevices ( daoSession ) ;
for ( final Device dbDevice : activeDevices ) {
final SharedPreferences deviceSharedPrefs = GBApplication . getDeviceSpecificSharedPrefs ( dbDevice . getIdentifier ( ) ) ;
2022-11-27 18:56:10 +01:00
2023-06-17 18:28:11 +02:00
final String chartsTabsValue = deviceSharedPrefs . getString ( " charts_tabs " , null ) ;
if ( chartsTabsValue = = null ) {
continue ;
}
final String newPrefValue ;
if ( ! StringUtils . isBlank ( chartsTabsValue ) ) {
newPrefValue = chartsTabsValue + " ,stress " ;
} else {
newPrefValue = " stress " ;
}
final SharedPreferences . Editor deviceSharedPrefsEdit = deviceSharedPrefs . edit ( ) ;
deviceSharedPrefsEdit . putString ( " charts_tabs " , newPrefValue ) ;
deviceSharedPrefsEdit . apply ( ) ;
}
} catch ( Exception e ) {
Log . w ( TAG , " error acquiring DB lock " ) ;
2022-11-27 18:56:10 +01:00
}
2023-06-17 18:28:11 +02:00
}
2022-09-26 19:20:12 +02:00
2023-07-02 17:05:10 +02:00
if ( oldVersion < 21 ) {
// Add the new PAI tab to all devices
try ( DBHandler db = acquireDB ( ) ) {
final DaoSession daoSession = db . getDaoSession ( ) ;
final List < Device > activeDevices = DBHelper . getActiveDevices ( daoSession ) ;
for ( final Device dbDevice : activeDevices ) {
final SharedPreferences deviceSharedPrefs = GBApplication . getDeviceSpecificSharedPrefs ( dbDevice . getIdentifier ( ) ) ;
final String chartsTabsValue = deviceSharedPrefs . getString ( " charts_tabs " , null ) ;
if ( chartsTabsValue = = null ) {
continue ;
}
final String newPrefValue ;
if ( ! StringUtils . isBlank ( chartsTabsValue ) ) {
newPrefValue = chartsTabsValue + " ,pai " ;
} else {
newPrefValue = " pai " ;
}
final SharedPreferences . Editor deviceSharedPrefsEdit = deviceSharedPrefs . edit ( ) ;
deviceSharedPrefsEdit . putString ( " charts_tabs " , newPrefValue ) ;
deviceSharedPrefsEdit . apply ( ) ;
}
} catch ( Exception e ) {
Log . w ( TAG , " error acquiring DB lock " ) ;
}
}
2023-08-27 23:47:19 +02:00
if ( oldVersion < 22 ) {
try ( DBHandler db = acquireDB ( ) ) {
final DaoSession daoSession = db . getDaoSession ( ) ;
final List < Device > activeDevices = DBHelper . getActiveDevices ( daoSession ) ;
for ( Device dbDevice : activeDevices ) {
2023-10-22 01:33:08 +02:00
final DeviceType deviceType = DeviceType . fromName ( dbDevice . getTypeName ( ) ) ;
2023-08-27 23:47:19 +02:00
if ( deviceType = = MIBAND2 ) {
final String name = dbDevice . getName ( ) ;
if ( " Mi Band HRX " . equalsIgnoreCase ( name ) | | " Mi Band 2i " . equalsIgnoreCase ( name ) ) {
2023-10-22 01:33:08 +02:00
dbDevice . setTypeName ( DeviceType . MIBAND2_HRX . name ( ) ) ;
2023-08-27 23:47:19 +02:00
daoSession . getDeviceDao ( ) . update ( dbDevice ) ;
}
}
}
} catch ( Exception e ) {
Log . w ( TAG , " error acquiring DB lock " ) ;
}
}
2023-10-28 20:32:44 +02:00
if ( oldVersion < 26 ) {
try ( DBHandler db = acquireDB ( ) ) {
final DaoSession daoSession = db . getDaoSession ( ) ;
final List < Device > activeDevices = DBHelper . getActiveDevices ( daoSession ) ;
for ( final Device dbDevice : activeDevices ) {
final SharedPreferences deviceSharedPrefs = GBApplication . getDeviceSpecificSharedPrefs ( dbDevice . getIdentifier ( ) ) ;
final String chartsTabsValue = deviceSharedPrefs . getString ( " charts_tabs " , null ) ;
if ( chartsTabsValue = = null ) {
continue ;
}
final String newPrefValue ;
if ( ! StringUtils . isBlank ( chartsTabsValue ) ) {
newPrefValue = chartsTabsValue + " ,spo2 " ;
} else {
newPrefValue = " spo2 " ;
}
final SharedPreferences . Editor deviceSharedPrefsEdit = deviceSharedPrefs . edit ( ) ;
deviceSharedPrefsEdit . putString ( " charts_tabs " , newPrefValue ) ;
deviceSharedPrefsEdit . apply ( ) ;
}
} catch ( Exception e ) {
Log . w ( TAG , " error acquiring DB lock " ) ;
}
}
2023-12-05 12:45:32 +01:00
if ( oldVersion < 27 ) {
try ( DBHandler db = acquireDB ( ) ) {
final DaoSession daoSession = db . getDaoSession ( ) ;
final List < Device > activeDevices = DBHelper . getActiveDevices ( daoSession ) ;
for ( final Device dbDevice : activeDevices ) {
final SharedPreferences deviceSharedPrefs = GBApplication . getDeviceSpecificSharedPrefs ( dbDevice . getIdentifier ( ) ) ;
final SharedPreferences . Editor deviceSharedPrefsEdit = deviceSharedPrefs . edit ( ) ;
for ( final Map . Entry < String , ? > entry : deviceSharedPrefs . getAll ( ) . entrySet ( ) ) {
final String key = entry . getKey ( ) ;
if ( key . startsWith ( " huami_2021_known_config_ " ) ) {
deviceSharedPrefsEdit . putString (
key . replace ( " huami_2021_known_config_ " , " " ) + " _is_known " ,
entry . getValue ( ) . toString ( )
) ;
} else if ( key . endsWith ( " _huami_2021_possible_values " ) ) {
deviceSharedPrefsEdit . putString (
key . replace ( " _huami_2021_possible_values " , " " ) + " _possible_values " ,
entry . getValue ( ) . toString ( )
) ;
}
}
deviceSharedPrefsEdit . apply ( ) ;
}
} catch ( Exception e ) {
Log . w ( TAG , " error acquiring DB lock " ) ;
}
}
2023-12-09 12:40:28 +01:00
if ( oldVersion < 28 ) {
try ( DBHandler db = acquireDB ( ) ) {
final DaoSession daoSession = db . getDaoSession ( ) ;
final List < Device > activeDevices = DBHelper . getActiveDevices ( daoSession ) ;
for ( final Device dbDevice : activeDevices ) {
final SharedPreferences deviceSharedPrefs = GBApplication . getDeviceSpecificSharedPrefs ( dbDevice . getIdentifier ( ) ) ;
final SharedPreferences . Editor deviceSharedPrefsEdit = deviceSharedPrefs . edit ( ) ;
boolean shouldApply = false ;
if ( ! " UNKNOWN " . equals ( deviceSharedPrefs . getString ( " events_forwarding_fellsleep_action_selection " , " UNKNOWN " ) ) ) {
shouldApply = true ;
deviceSharedPrefsEdit . putStringSet (
" events_forwarding_fellsleep_action_selections " ,
Collections . singleton ( deviceSharedPrefs . getString ( " events_forwarding_fellsleep_action_selection " , " UNKNOWN " ) )
) ;
}
if ( ! " UNKNOWN " . equals ( deviceSharedPrefs . getString ( " events_forwarding_wokeup_action_selection " , " UNKNOWN " ) ) ) {
shouldApply = true ;
deviceSharedPrefsEdit . putStringSet (
" events_forwarding_wokeup_action_selections " ,
Collections . singleton ( deviceSharedPrefs . getString ( " events_forwarding_wokeup_action_selection " , " UNKNOWN " ) )
) ;
}
if ( ! " UNKNOWN " . equals ( deviceSharedPrefs . getString ( " events_forwarding_startnonwear_action_selection " , " UNKNOWN " ) ) ) {
shouldApply = true ;
deviceSharedPrefsEdit . putStringSet (
" events_forwarding_startnonwear_action_selections " ,
Collections . singleton ( deviceSharedPrefs . getString ( " events_forwarding_startnonwear_action_selection " , " UNKNOWN " ) )
) ;
}
if ( shouldApply ) {
deviceSharedPrefsEdit . apply ( ) ;
}
}
} catch ( Exception e ) {
Log . w ( TAG , " error acquiring DB lock " ) ;
}
}
2019-05-20 16:36:06 +02:00
editor . putString ( PREFS_VERSION , Integer . toString ( CURRENT_PREFS_VERSION ) ) ;
2016-04-14 16:15:58 +02:00
editor . apply ( ) ;
2016-02-02 17:33:24 +01:00
}
2019-05-09 14:55:37 +02:00
public static SharedPreferences getDeviceSpecificSharedPrefs ( String deviceIdentifier ) {
if ( deviceIdentifier = = null | | deviceIdentifier . isEmpty ( ) ) {
return null ;
}
return context . getSharedPreferences ( " devicesettings_ " + deviceIdentifier , Context . MODE_PRIVATE ) ;
}
2021-09-18 22:27:15 +02:00
public static void deleteDeviceSpecificSharedPrefs ( String deviceIdentifier ) {
if ( deviceIdentifier = = null | | deviceIdentifier . isEmpty ( ) ) {
return ;
}
context . getSharedPreferences ( " devicesettings_ " + deviceIdentifier , Context . MODE_PRIVATE ) . edit ( ) . clear ( ) . apply ( ) ;
}
2017-07-31 23:00:02 +02:00
public static void setLanguage ( String lang ) {
2017-07-31 22:49:05 +02:00
if ( lang . equals ( " default " ) ) {
2017-07-31 23:00:02 +02:00
language = Resources . getSystem ( ) . getConfiguration ( ) . locale ;
2017-07-31 22:49:05 +02:00
} else {
language = new Locale ( lang ) ;
}
2017-09-03 01:02:31 +02:00
updateLanguage ( language ) ;
}
2017-07-31 22:49:05 +02:00
2017-09-03 01:02:31 +02:00
public static void updateLanguage ( Locale locale ) {
AndroidUtils . setLanguage ( context , locale ) ;
2017-07-31 23:00:02 +02:00
2017-07-31 22:49:05 +02:00
Intent intent = new Intent ( ) ;
intent . setAction ( ACTION_LANGUAGE_CHANGE ) ;
LocalBroadcastManager . getInstance ( context ) . sendBroadcast ( intent ) ;
}
2023-12-10 11:30:27 +01:00
public static LimitedQueue < Integer , String > getIDSenderLookup ( ) {
2015-12-14 23:31:31 +01:00
return mIDSenderLookup ;
}
2016-04-14 15:21:25 +02:00
public static boolean isDarkThemeEnabled ( ) {
2020-05-05 14:46:09 +02:00
String selectedTheme = prefs . getString ( " pref_key_theme " , context . getString ( R . string . pref_theme_value_system ) ) ;
2021-07-19 16:07:20 +02:00
Resources resources = context . getResources ( ) ;
2020-04-13 17:51:36 +02:00
2021-07-19 16:07:20 +02:00
if ( Build . VERSION . SDK_INT > = Build . VERSION_CODES . Q & &
2023-07-22 21:23:27 +02:00
( selectedTheme . equals ( context . getString ( R . string . pref_theme_value_system ) ) | | selectedTheme . equals ( context . getString ( R . string . pref_theme_value_dynamic ) ) ) ) {
2021-07-19 16:07:20 +02:00
return ( resources . getConfiguration ( ) . uiMode & Configuration . UI_MODE_NIGHT_MASK ) = = Configuration . UI_MODE_NIGHT_YES ;
} else {
return selectedTheme . equals ( context . getString ( R . string . pref_theme_value_dark ) ) ;
}
2016-04-14 15:21:25 +02:00
}
2016-04-23 23:24:56 +02:00
2021-07-06 08:31:22 +02:00
public static boolean isAmoledBlackEnabled ( ) {
return prefs . getBoolean ( " pref_key_theme_amoled_black " , false ) ;
}
2023-07-22 21:23:27 +02:00
public static boolean areDynamicColorsEnabled ( ) {
String selectedTheme = prefs . getString ( " pref_key_theme " , context . getString ( R . string . pref_theme_value_system ) ) ;
return selectedTheme . equals ( context . getString ( R . string . pref_theme_value_dynamic ) ) ;
}
2016-04-23 23:24:56 +02:00
public static int getTextColor ( Context context ) {
2023-08-05 22:57:09 +02:00
if ( GBApplication . isDarkThemeEnabled ( ) ) {
return context . getResources ( ) . getColor ( R . color . primarytext_dark ) ;
} else {
return context . getResources ( ) . getColor ( R . color . primarytext_light ) ;
}
}
public static int getSecondaryTextColor ( Context context ) {
return context . getResources ( ) . getColor ( R . color . secondarytext ) ;
2016-04-23 23:24:56 +02:00
}
2016-12-07 23:13:51 +01:00
2017-09-03 01:02:31 +02:00
@Override
public void onConfigurationChanged ( Configuration newConfig ) {
super . onConfigurationChanged ( newConfig ) ;
updateLanguage ( getLanguage ( ) ) ;
}
2016-04-23 23:24:56 +02:00
public static int getBackgroundColor ( Context context ) {
TypedValue typedValue = new TypedValue ( ) ;
Resources . Theme theme = context . getTheme ( ) ;
theme . resolveAttribute ( android . R . attr . background , typedValue , true ) ;
return typedValue . data ;
}
2020-10-22 06:32:21 +02:00
public static int getWindowBackgroundColor ( Context context ) {
TypedValue typedValue = new TypedValue ( ) ;
Resources . Theme theme = context . getTheme ( ) ;
theme . resolveAttribute ( android . R . attr . windowBackground , typedValue , true ) ;
return typedValue . data ;
}
2016-04-25 23:18:55 +02:00
public static Prefs getPrefs ( ) {
return prefs ;
}
2016-04-28 23:17:13 +02:00
public static GBPrefs getGBPrefs ( ) {
return gbPrefs ;
2016-04-14 15:21:25 +02:00
}
2016-06-18 23:35:34 +02:00
2016-12-07 23:13:51 +01:00
public DeviceManager getDeviceManager ( ) {
2016-06-18 23:35:34 +02:00
return deviceManager ;
}
2017-04-27 07:52:11 +02:00
public static GBApplication app ( ) {
return app ;
}
2017-07-31 22:49:05 +02:00
public static Locale getLanguage ( ) {
return language ;
}
2017-10-19 21:52:38 +02:00
2023-08-09 20:52:03 +02:00
public static boolean isNightly ( ) {
return BuildConfig . APPLICATION_ID . contains ( " nightly " ) ;
}
2023-12-20 20:09:49 +01:00
public static boolean isDebug ( ) {
return BuildConfig . DEBUG ;
}
2017-10-19 21:52:38 +02:00
public String getVersion ( ) {
try {
return getPackageManager ( ) . getPackageInfo ( getPackageName ( ) , PackageManager . GET_META_DATA ) . versionName ;
} catch ( PackageManager . NameNotFoundException e ) {
GB . log ( " Unable to determine Gadgetbridge's version " , GB . WARN , e ) ;
return " 0.0.0 " ;
}
}
public String getNameAndVersion ( ) {
try {
ApplicationInfo appInfo = getPackageManager ( ) . getApplicationInfo ( getContext ( ) . getPackageName ( ) , PackageManager . GET_META_DATA ) ;
PackageInfo packageInfo = getPackageManager ( ) . getPackageInfo ( getPackageName ( ) , PackageManager . GET_META_DATA ) ;
return String . format ( " %s %s " , appInfo . name , packageInfo . versionName ) ;
} catch ( PackageManager . NameNotFoundException e ) {
GB . log ( " Unable to determine Gadgetbridge's name/version " , GB . WARN , e ) ;
return " Gadgetbridge " ;
}
}
2022-02-08 13:38:23 +01:00
public void setOpenTracksObserver ( OpenTracksContentObserver openTracksObserver ) {
this . openTracksObserver = openTracksObserver ;
}
public OpenTracksContentObserver getOpenTracksObserver ( ) {
return openTracksObserver ;
}
2022-02-27 18:43:53 +01:00
public long getLastAutoExportTimestamp ( ) {
return lastAutoExportTimestamp ;
}
public void setLastAutoExportTimestamp ( long lastAutoExportTimestamp ) {
this . lastAutoExportTimestamp = lastAutoExportTimestamp ;
}
public long getAutoExportScheduledTimestamp ( ) {
return autoExportScheduledTimestamp ;
}
public void setAutoExportScheduledTimestamp ( long autoExportScheduledTimestamp ) {
this . autoExportScheduledTimestamp = autoExportScheduledTimestamp ;
}
2015-05-01 09:36:10 +02:00
}