@josuelmm/cordova-background-geolocation 3.1.0 → 3.2.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (38) hide show
  1. package/CHANGELOG.md +55 -0
  2. package/HISTORY.md +6 -0
  3. package/README.md +62 -9
  4. package/RELEASE.MD +15 -4
  5. package/android/CDVBackgroundGeolocation/src/main/java/com/marianhello/bgloc/cordova/ConfigMapper.java +8 -0
  6. package/android/CDVBackgroundGeolocation/src/main/java/com/tenforwardconsulting/bgloc/cordova/BackgroundGeolocationPlugin.java +57 -1
  7. package/android/common/src/main/java/com/marianhello/bgloc/BackgroundGeolocationFacade.java +26 -0
  8. package/android/common/src/main/java/com/marianhello/bgloc/Config.java +44 -0
  9. package/android/common/src/main/java/com/marianhello/bgloc/PostLocationTask.java +13 -0
  10. package/android/common/src/main/java/com/marianhello/bgloc/data/SessionLocationDAO.java +18 -0
  11. package/android/common/src/main/java/com/marianhello/bgloc/data/sqlite/SQLiteConfigurationContract.java +5 -1
  12. package/android/common/src/main/java/com/marianhello/bgloc/data/sqlite/SQLiteConfigurationDAO.java +13 -1
  13. package/android/common/src/main/java/com/marianhello/bgloc/data/sqlite/SQLiteOpenHelper.java +13 -1
  14. package/android/common/src/main/java/com/marianhello/bgloc/data/sqlite/SQLiteSessionContract.java +74 -0
  15. package/android/common/src/main/java/com/marianhello/bgloc/data/sqlite/SQLiteSessionLocationDAO.java +169 -0
  16. package/android/common/src/main/java/com/marianhello/bgloc/service/LocationServiceImpl.java +132 -4
  17. package/angular/background-geolocation.service.ts +28 -0
  18. package/angular/dist/background-geolocation.service.d.ts +4 -0
  19. package/angular/dist/esm2022/background-geolocation.service.mjs +13 -1
  20. package/angular/dist/fesm2022/josuelmm-cordova-background-geolocation.mjs +12 -0
  21. package/angular/dist/fesm2022/josuelmm-cordova-background-geolocation.mjs.map +1 -1
  22. package/angular/dist/public-api.d.ts +1 -1
  23. package/ios/CDVBackgroundGeolocation/CDVBackgroundGeolocation.h +4 -0
  24. package/ios/CDVBackgroundGeolocation/CDVBackgroundGeolocation.m +41 -1
  25. package/ios/common/BackgroundGeolocation/MAURBackgroundGeolocationFacade.h +4 -0
  26. package/ios/common/BackgroundGeolocation/MAURBackgroundGeolocationFacade.m +21 -0
  27. package/ios/common/BackgroundGeolocation/MAURGeolocationOpenHelper.m +12 -3
  28. package/ios/common/BackgroundGeolocation/MAURPostLocationTask.m +5 -0
  29. package/ios/common/BackgroundGeolocation/MAURSessionLocationContract.h +29 -0
  30. package/ios/common/BackgroundGeolocation/MAURSessionLocationContract.m +31 -0
  31. package/ios/common/BackgroundGeolocation/MAURSessionLocationDAO.h +25 -0
  32. package/ios/common/BackgroundGeolocation/MAURSessionLocationDAO.m +153 -0
  33. package/package.json +7 -3
  34. package/plugin.xml +8 -1
  35. package/www/BackgroundGeolocation.d.ts +60 -0
  36. package/www/BackgroundGeolocation.js +24 -0
  37. package/www/cordova-channel-stub.js +27 -0
  38. package/www/cordova-exec-stub.js +15 -0
@@ -0,0 +1,74 @@
1
+ package com.marianhello.bgloc.data.sqlite;
2
+
3
+ import android.provider.BaseColumns;
4
+
5
+ import static com.marianhello.bgloc.data.sqlite.SQLiteOpenHelper.COMMA_SEP;
6
+ import static com.marianhello.bgloc.data.sqlite.SQLiteOpenHelper.INTEGER_TYPE;
7
+ import static com.marianhello.bgloc.data.sqlite.SQLiteOpenHelper.REAL_TYPE;
8
+ import static com.marianhello.bgloc.data.sqlite.SQLiteOpenHelper.TEXT_TYPE;
9
+
10
+ /**
11
+ * Contract for the "session" location table.
12
+ * Stores all locations for the current recording session (route).
13
+ * Cleared on startSession() and clearSession(); not cleared when sync succeeds.
14
+ */
15
+ public final class SQLiteSessionContract {
16
+
17
+ public SQLiteSessionContract() {}
18
+
19
+ public static abstract class SessionEntry implements BaseColumns {
20
+ public static final String TABLE_NAME = "location_session";
21
+ public static final String COLUMN_NAME_NULLABLE = "NULLHACK";
22
+ public static final String COLUMN_NAME_TIME = "time";
23
+ public static final String COLUMN_NAME_ACCURACY = "accuracy";
24
+ public static final String COLUMN_NAME_VERTICAL_ACCURACY = "vertical_accuracy";
25
+ public static final String COLUMN_NAME_SPEED = "speed";
26
+ public static final String COLUMN_NAME_BEARING = "bearing";
27
+ public static final String COLUMN_NAME_ALTITUDE = "altitude";
28
+ public static final String COLUMN_NAME_LATITUDE = "latitude";
29
+ public static final String COLUMN_NAME_LONGITUDE = "longitude";
30
+ public static final String COLUMN_NAME_RADIUS = "radius";
31
+ public static final String COLUMN_NAME_HAS_ACCURACY = "has_accuracy";
32
+ public static final String COLUMN_NAME_HAS_VERTICAL_ACCURACY = "has_vertical_accuracy";
33
+ public static final String COLUMN_NAME_HAS_SPEED = "has_speed";
34
+ public static final String COLUMN_NAME_HAS_BEARING = "has_bearing";
35
+ public static final String COLUMN_NAME_HAS_ALTITUDE = "has_altitude";
36
+ public static final String COLUMN_NAME_HAS_RADIUS = "has_radius";
37
+ public static final String COLUMN_NAME_PROVIDER = "provider";
38
+ public static final String COLUMN_NAME_LOCATION_PROVIDER = "service_provider";
39
+ public static final String COLUMN_NAME_STATUS = "valid";
40
+ public static final String COLUMN_NAME_BATCH_START_MILLIS = "batch_start";
41
+ public static final String COLUMN_NAME_MOCK_FLAGS = "mock_flags";
42
+
43
+ public static final String SQL_CREATE_SESSION_TABLE =
44
+ "CREATE TABLE " + SessionEntry.TABLE_NAME + " (" +
45
+ SessionEntry._ID + " INTEGER PRIMARY KEY AUTOINCREMENT," +
46
+ SessionEntry.COLUMN_NAME_TIME + INTEGER_TYPE + COMMA_SEP +
47
+ SessionEntry.COLUMN_NAME_ACCURACY + REAL_TYPE + COMMA_SEP +
48
+ SessionEntry.COLUMN_NAME_VERTICAL_ACCURACY + REAL_TYPE + COMMA_SEP +
49
+ SessionEntry.COLUMN_NAME_SPEED + REAL_TYPE + COMMA_SEP +
50
+ SessionEntry.COLUMN_NAME_BEARING + REAL_TYPE + COMMA_SEP +
51
+ SessionEntry.COLUMN_NAME_ALTITUDE + REAL_TYPE + COMMA_SEP +
52
+ SessionEntry.COLUMN_NAME_LATITUDE + REAL_TYPE + COMMA_SEP +
53
+ SessionEntry.COLUMN_NAME_LONGITUDE + REAL_TYPE + COMMA_SEP +
54
+ SessionEntry.COLUMN_NAME_RADIUS + REAL_TYPE + COMMA_SEP +
55
+ SessionEntry.COLUMN_NAME_HAS_ACCURACY + INTEGER_TYPE + COMMA_SEP +
56
+ SessionEntry.COLUMN_NAME_HAS_VERTICAL_ACCURACY + INTEGER_TYPE + COMMA_SEP +
57
+ SessionEntry.COLUMN_NAME_HAS_SPEED + INTEGER_TYPE + COMMA_SEP +
58
+ SessionEntry.COLUMN_NAME_HAS_BEARING + INTEGER_TYPE + COMMA_SEP +
59
+ SessionEntry.COLUMN_NAME_HAS_ALTITUDE + INTEGER_TYPE + COMMA_SEP +
60
+ SessionEntry.COLUMN_NAME_HAS_RADIUS + INTEGER_TYPE + COMMA_SEP +
61
+ SessionEntry.COLUMN_NAME_PROVIDER + TEXT_TYPE + COMMA_SEP +
62
+ SessionEntry.COLUMN_NAME_LOCATION_PROVIDER + INTEGER_TYPE + COMMA_SEP +
63
+ SessionEntry.COLUMN_NAME_STATUS + INTEGER_TYPE + COMMA_SEP +
64
+ SessionEntry.COLUMN_NAME_BATCH_START_MILLIS + INTEGER_TYPE + COMMA_SEP +
65
+ SessionEntry.COLUMN_NAME_MOCK_FLAGS + INTEGER_TYPE +
66
+ " )";
67
+
68
+ public static final String SQL_DROP_SESSION_TABLE =
69
+ "DROP TABLE IF EXISTS " + SessionEntry.TABLE_NAME;
70
+
71
+ public static final String SQL_CREATE_SESSION_TABLE_TIME_IDX =
72
+ "CREATE INDEX session_time_idx ON " + SessionEntry.TABLE_NAME + " (" + SessionEntry.COLUMN_NAME_TIME + ")";
73
+ }
74
+ }
@@ -0,0 +1,169 @@
1
+ package com.marianhello.bgloc.data.sqlite;
2
+
3
+ import android.content.ContentValues;
4
+ import android.content.Context;
5
+ import android.database.Cursor;
6
+ import android.database.sqlite.SQLiteDatabase;
7
+ import com.marianhello.bgloc.data.BackgroundLocation;
8
+ import com.marianhello.bgloc.data.SessionLocationDAO;
9
+
10
+ import java.util.ArrayList;
11
+ import java.util.Collection;
12
+
13
+ public class SQLiteSessionLocationDAO implements SessionLocationDAO {
14
+
15
+ private static final String PREFS_NAME = "bgloc_session";
16
+ private static final String KEY_SESSION_ACTIVE = "session_active";
17
+
18
+ private final SQLiteDatabase db;
19
+ private final Context context;
20
+
21
+ public SQLiteSessionLocationDAO(Context context) {
22
+ this.context = context.getApplicationContext();
23
+ SQLiteOpenHelper helper = SQLiteOpenHelper.getHelper(this.context);
24
+ this.db = helper.getWritableDatabase();
25
+ }
26
+
27
+ @Override
28
+ public void startSession() {
29
+ db.delete(SQLiteSessionContract.SessionEntry.TABLE_NAME, null, null);
30
+ setSessionActive(true);
31
+ }
32
+
33
+ @Override
34
+ public void clearSession() {
35
+ db.delete(SQLiteSessionContract.SessionEntry.TABLE_NAME, null, null);
36
+ setSessionActive(false);
37
+ }
38
+
39
+ @Override
40
+ public boolean isSessionActive() {
41
+ return context.getSharedPreferences(PREFS_NAME, Context.MODE_PRIVATE)
42
+ .getBoolean(KEY_SESSION_ACTIVE, false);
43
+ }
44
+
45
+ private void setSessionActive(boolean active) {
46
+ context.getSharedPreferences(PREFS_NAME, Context.MODE_PRIVATE)
47
+ .edit()
48
+ .putBoolean(KEY_SESSION_ACTIVE, active)
49
+ .apply();
50
+ }
51
+
52
+ @Override
53
+ public void persistSessionLocation(BackgroundLocation location) {
54
+ if (!isSessionActive() || location == null) return;
55
+ ContentValues values = getContentValues(location);
56
+ db.insertOrThrow(SQLiteSessionContract.SessionEntry.TABLE_NAME,
57
+ SQLiteSessionContract.SessionEntry.COLUMN_NAME_NULLABLE, values);
58
+ }
59
+
60
+ @Override
61
+ public Collection<BackgroundLocation> getSessionLocations() {
62
+ Collection<BackgroundLocation> locations = new ArrayList<>();
63
+ String orderBy = SQLiteSessionContract.SessionEntry.COLUMN_NAME_TIME + " ASC";
64
+ Cursor cursor = null;
65
+ try {
66
+ cursor = db.query(
67
+ SQLiteSessionContract.SessionEntry.TABLE_NAME,
68
+ queryColumns(),
69
+ null, null, null, null, orderBy);
70
+ while (cursor.moveToNext()) {
71
+ locations.add(hydrate(cursor));
72
+ }
73
+ } finally {
74
+ if (cursor != null) cursor.close();
75
+ }
76
+ return locations;
77
+ }
78
+
79
+ @Override
80
+ public int getSessionLocationsCount() {
81
+ Cursor cursor = null;
82
+ try {
83
+ cursor = db.rawQuery("SELECT COUNT(*) FROM " + SQLiteSessionContract.SessionEntry.TABLE_NAME, null);
84
+ return cursor.moveToFirst() ? cursor.getInt(0) : 0;
85
+ } finally {
86
+ if (cursor != null) cursor.close();
87
+ }
88
+ }
89
+
90
+ private BackgroundLocation hydrate(Cursor c) {
91
+ BackgroundLocation l = new BackgroundLocation(c.getString(c.getColumnIndex(SQLiteSessionContract.SessionEntry.COLUMN_NAME_PROVIDER)));
92
+ l.setTime(c.getLong(c.getColumnIndex(SQLiteSessionContract.SessionEntry.COLUMN_NAME_TIME)));
93
+ if (c.getInt(c.getColumnIndex(SQLiteSessionContract.SessionEntry.COLUMN_NAME_HAS_ACCURACY)) == 1) {
94
+ l.setAccuracy(c.getFloat(c.getColumnIndex(SQLiteSessionContract.SessionEntry.COLUMN_NAME_ACCURACY)));
95
+ }
96
+ if (c.getInt(c.getColumnIndex(SQLiteSessionContract.SessionEntry.COLUMN_NAME_HAS_VERTICAL_ACCURACY)) == 1) {
97
+ l.setVerticalAccuracy(c.getFloat(c.getColumnIndex(SQLiteSessionContract.SessionEntry.COLUMN_NAME_VERTICAL_ACCURACY)));
98
+ }
99
+ if (c.getInt(c.getColumnIndex(SQLiteSessionContract.SessionEntry.COLUMN_NAME_HAS_SPEED)) == 1) {
100
+ l.setSpeed(c.getFloat(c.getColumnIndex(SQLiteSessionContract.SessionEntry.COLUMN_NAME_SPEED)));
101
+ }
102
+ if (c.getInt(c.getColumnIndex(SQLiteSessionContract.SessionEntry.COLUMN_NAME_HAS_BEARING)) == 1) {
103
+ l.setBearing(c.getFloat(c.getColumnIndex(SQLiteSessionContract.SessionEntry.COLUMN_NAME_BEARING)));
104
+ }
105
+ if (c.getInt(c.getColumnIndex(SQLiteSessionContract.SessionEntry.COLUMN_NAME_HAS_ALTITUDE)) == 1) {
106
+ l.setAltitude(c.getDouble(c.getColumnIndex(SQLiteSessionContract.SessionEntry.COLUMN_NAME_ALTITUDE)));
107
+ }
108
+ if (c.getInt(c.getColumnIndex(SQLiteSessionContract.SessionEntry.COLUMN_NAME_HAS_RADIUS)) == 1) {
109
+ l.setRadius(c.getFloat(c.getColumnIndex(SQLiteSessionContract.SessionEntry.COLUMN_NAME_RADIUS)));
110
+ }
111
+ l.setLatitude(c.getDouble(c.getColumnIndex(SQLiteSessionContract.SessionEntry.COLUMN_NAME_LATITUDE)));
112
+ l.setLongitude(c.getDouble(c.getColumnIndex(SQLiteSessionContract.SessionEntry.COLUMN_NAME_LONGITUDE)));
113
+ l.setLocationProvider(c.getInt(c.getColumnIndex(SQLiteSessionContract.SessionEntry.COLUMN_NAME_LOCATION_PROVIDER)));
114
+ l.setLocationId(c.getLong(c.getColumnIndex(SQLiteSessionContract.SessionEntry._ID)));
115
+ l.setMockFlags(c.getInt(c.getColumnIndex(SQLiteSessionContract.SessionEntry.COLUMN_NAME_MOCK_FLAGS)));
116
+ return l;
117
+ }
118
+
119
+ private ContentValues getContentValues(BackgroundLocation l) {
120
+ ContentValues values = new ContentValues();
121
+ values.put(SQLiteSessionContract.SessionEntry.COLUMN_NAME_PROVIDER, l.getProvider());
122
+ values.put(SQLiteSessionContract.SessionEntry.COLUMN_NAME_TIME, l.getTime());
123
+ values.put(SQLiteSessionContract.SessionEntry.COLUMN_NAME_ACCURACY, l.getAccuracy());
124
+ values.put(SQLiteSessionContract.SessionEntry.COLUMN_NAME_VERTICAL_ACCURACY, l.getVerticalAccuracy());
125
+ values.put(SQLiteSessionContract.SessionEntry.COLUMN_NAME_SPEED, l.getSpeed());
126
+ values.put(SQLiteSessionContract.SessionEntry.COLUMN_NAME_BEARING, l.getBearing());
127
+ values.put(SQLiteSessionContract.SessionEntry.COLUMN_NAME_ALTITUDE, l.getAltitude());
128
+ values.put(SQLiteSessionContract.SessionEntry.COLUMN_NAME_RADIUS, l.getRadius());
129
+ values.put(SQLiteSessionContract.SessionEntry.COLUMN_NAME_LATITUDE, l.getLatitude());
130
+ values.put(SQLiteSessionContract.SessionEntry.COLUMN_NAME_LONGITUDE, l.getLongitude());
131
+ values.put(SQLiteSessionContract.SessionEntry.COLUMN_NAME_HAS_ACCURACY, l.hasAccuracy() ? 1 : 0);
132
+ values.put(SQLiteSessionContract.SessionEntry.COLUMN_NAME_HAS_VERTICAL_ACCURACY, l.hasVerticalAccuracy() ? 1 : 0);
133
+ values.put(SQLiteSessionContract.SessionEntry.COLUMN_NAME_HAS_SPEED, l.hasSpeed() ? 1 : 0);
134
+ values.put(SQLiteSessionContract.SessionEntry.COLUMN_NAME_HAS_BEARING, l.hasBearing() ? 1 : 0);
135
+ values.put(SQLiteSessionContract.SessionEntry.COLUMN_NAME_HAS_ALTITUDE, l.hasAltitude() ? 1 : 0);
136
+ values.put(SQLiteSessionContract.SessionEntry.COLUMN_NAME_HAS_RADIUS, l.hasRadius() ? 1 : 0);
137
+ values.put(SQLiteSessionContract.SessionEntry.COLUMN_NAME_LOCATION_PROVIDER, l.getLocationProvider());
138
+ values.put(SQLiteSessionContract.SessionEntry.COLUMN_NAME_STATUS, 0);
139
+ values.put(SQLiteSessionContract.SessionEntry.COLUMN_NAME_BATCH_START_MILLIS, 0L);
140
+ values.put(SQLiteSessionContract.SessionEntry.COLUMN_NAME_MOCK_FLAGS, l.getMockFlags());
141
+ return values;
142
+ }
143
+
144
+ private String[] queryColumns() {
145
+ return new String[]{
146
+ SQLiteSessionContract.SessionEntry._ID,
147
+ SQLiteSessionContract.SessionEntry.COLUMN_NAME_PROVIDER,
148
+ SQLiteSessionContract.SessionEntry.COLUMN_NAME_TIME,
149
+ SQLiteSessionContract.SessionEntry.COLUMN_NAME_ACCURACY,
150
+ SQLiteSessionContract.SessionEntry.COLUMN_NAME_VERTICAL_ACCURACY,
151
+ SQLiteSessionContract.SessionEntry.COLUMN_NAME_SPEED,
152
+ SQLiteSessionContract.SessionEntry.COLUMN_NAME_BEARING,
153
+ SQLiteSessionContract.SessionEntry.COLUMN_NAME_ALTITUDE,
154
+ SQLiteSessionContract.SessionEntry.COLUMN_NAME_RADIUS,
155
+ SQLiteSessionContract.SessionEntry.COLUMN_NAME_LATITUDE,
156
+ SQLiteSessionContract.SessionEntry.COLUMN_NAME_LONGITUDE,
157
+ SQLiteSessionContract.SessionEntry.COLUMN_NAME_HAS_ACCURACY,
158
+ SQLiteSessionContract.SessionEntry.COLUMN_NAME_HAS_VERTICAL_ACCURACY,
159
+ SQLiteSessionContract.SessionEntry.COLUMN_NAME_HAS_SPEED,
160
+ SQLiteSessionContract.SessionEntry.COLUMN_NAME_HAS_BEARING,
161
+ SQLiteSessionContract.SessionEntry.COLUMN_NAME_HAS_ALTITUDE,
162
+ SQLiteSessionContract.SessionEntry.COLUMN_NAME_HAS_RADIUS,
163
+ SQLiteSessionContract.SessionEntry.COLUMN_NAME_LOCATION_PROVIDER,
164
+ SQLiteSessionContract.SessionEntry.COLUMN_NAME_STATUS,
165
+ SQLiteSessionContract.SessionEntry.COLUMN_NAME_BATCH_START_MILLIS,
166
+ SQLiteSessionContract.SessionEntry.COLUMN_NAME_MOCK_FLAGS
167
+ };
168
+ }
169
+ }
@@ -22,6 +22,7 @@ import android.content.ComponentName;
22
22
  import android.content.pm.PackageManager;
23
23
  import android.content.pm.ServiceInfo;
24
24
  import android.Manifest;
25
+ import android.location.Location;
25
26
  import android.net.ConnectivityManager;
26
27
  import android.net.NetworkInfo;
27
28
  import android.os.Binder;
@@ -48,7 +49,9 @@ import com.marianhello.bgloc.data.BackgroundLocation;
48
49
  import com.marianhello.bgloc.data.ConfigurationDAO;
49
50
  import com.marianhello.bgloc.data.DAOFactory;
50
51
  import com.marianhello.bgloc.data.LocationDAO;
52
+ import com.marianhello.bgloc.data.SessionLocationDAO;
51
53
  import com.marianhello.bgloc.data.LocationTransform;
54
+ import com.marianhello.bgloc.data.sqlite.SQLiteSessionLocationDAO;
52
55
  import com.marianhello.bgloc.headless.AbstractTaskRunner;
53
56
  import com.marianhello.bgloc.headless.ActivityTask;
54
57
  import com.marianhello.bgloc.headless.LocationTask;
@@ -67,6 +70,8 @@ import com.marianhello.logging.UncaughtExceptionLogger;
67
70
  import org.chromium.content.browser.ThreadUtils;
68
71
  import org.json.JSONException;
69
72
 
73
+ import java.util.Locale;
74
+
70
75
  import static com.marianhello.bgloc.service.LocationServiceIntentBuilder.containsCommand;
71
76
  import static com.marianhello.bgloc.service.LocationServiceIntentBuilder.containsMessage;
72
77
  import static com.marianhello.bgloc.service.LocationServiceIntentBuilder.getCommand;
@@ -122,6 +127,7 @@ public class LocationServiceImpl extends Service implements ProviderDelegate, Lo
122
127
  private HandlerThread mHandlerThread;
123
128
  private ServiceHandler mServiceHandler;
124
129
  private LocationDAO mLocationDAO;
130
+ private SessionLocationDAO mSessionDAO;
125
131
  private PostLocationTask mPostLocationTask;
126
132
  private String mHeadlessTaskRunnerClass;
127
133
  private TaskRunner mHeadlessTaskRunner;
@@ -156,6 +162,30 @@ public class LocationServiceImpl extends Service implements ProviderDelegate, Lo
156
162
  }
157
163
  };
158
164
 
165
+ /** Session start time for notification elapsed time (showTime). */
166
+ private volatile long mSessionStartTime = 0L;
167
+ /** Accumulated distance in meters for notification (showDistance). */
168
+ private volatile double mSessionDistanceMeters = 0.0;
169
+ private volatile double mLastLat = 0.0;
170
+ private volatile double mLastLon = 0.0;
171
+ private volatile boolean mHasLastLocation = false;
172
+ private static final long NOTIFICATION_UPDATE_INTERVAL_MS = 1000L;
173
+ private final Runnable mNotificationUpdateRunnable = new Runnable() {
174
+ @Override
175
+ public void run() {
176
+ if (!sIsRunning || !mIsInForeground || mConfig == null) {
177
+ return;
178
+ }
179
+ boolean showTime = Boolean.TRUE.equals(mConfig.getShowTime());
180
+ boolean showDistance = Boolean.TRUE.equals(mConfig.getShowDistance());
181
+ if (!showTime && !showDistance) {
182
+ return;
183
+ }
184
+ updateForegroundNotification();
185
+ mMainHandler.postDelayed(this, NOTIFICATION_UPDATE_INTERVAL_MS);
186
+ }
187
+ };
188
+
159
189
  private static LocationTransform sLocationTransform;
160
190
  private static LocationProviderFactory sLocationProviderFactory;
161
191
 
@@ -228,8 +258,9 @@ public class LocationServiceImpl extends Service implements ProviderDelegate, Lo
228
258
  ContentResolver.setSyncAutomatically(mSyncAccount, authority, true);
229
259
 
230
260
  mLocationDAO = DAOFactory.createLocationDAO(this);
261
+ mSessionDAO = new SQLiteSessionLocationDAO(this);
231
262
 
232
- mPostLocationTask = new PostLocationTask(mLocationDAO,
263
+ mPostLocationTask = new PostLocationTask(mLocationDAO, mSessionDAO,
233
264
  new PostLocationTask.PostLocationTaskListener() {
234
265
  @Override
235
266
  public void onRequestedAbortUpdates() {
@@ -418,6 +449,12 @@ public class LocationServiceImpl extends Service implements ProviderDelegate, Lo
418
449
  mMainHandler.postDelayed(mWatchdogRunnable, WATCHDOG_INTERVAL_MS);
419
450
  }
420
451
 
452
+ mSessionStartTime = System.currentTimeMillis();
453
+ mSessionDistanceMeters = 0.0;
454
+ mLastLat = 0.0;
455
+ mLastLon = 0.0;
456
+ mHasLastLocation = false;
457
+
421
458
  ThreadUtils.runOnUiThreadBlocking(new Runnable() {
422
459
  @Override
423
460
  public void run() {
@@ -447,6 +484,7 @@ public class LocationServiceImpl extends Service implements ProviderDelegate, Lo
447
484
  }
448
485
 
449
486
  mMainHandler.removeCallbacks(mWatchdogRunnable);
487
+ mMainHandler.removeCallbacks(mNotificationUpdateRunnable);
450
488
 
451
489
  if (mWakeLock != null && mWakeLock.isHeld()) {
452
490
  try {
@@ -461,6 +499,7 @@ public class LocationServiceImpl extends Service implements ProviderDelegate, Lo
461
499
  mProvider.onStop();
462
500
  }
463
501
 
502
+ mIsInForeground = false;
464
503
  stopForeground(true);
465
504
  stopSelf();
466
505
 
@@ -530,9 +569,10 @@ public class LocationServiceImpl extends Service implements ProviderDelegate, Lo
530
569
  return;
531
570
  }
532
571
  Config config = getConfig();
572
+ String contentText = buildNotificationContentText(config);
533
573
  Notification notification = new NotificationHelper.NotificationFactory(this).getNotification(
534
574
  config.getNotificationTitle(),
535
- config.getNotificationText(),
575
+ contentText,
536
576
  config.getLargeNotificationIcon(),
537
577
  config.getSmallNotificationIcon(),
538
578
  config.getNotificationIconColor());
@@ -548,12 +588,14 @@ public class LocationServiceImpl extends Service implements ProviderDelegate, Lo
548
588
  super.startForeground(NOTIFICATION_ID, notification);
549
589
  }
550
590
  mIsInForeground = true;
591
+ scheduleNotificationUpdater();
551
592
  }
552
593
  }
553
594
 
554
595
  @Override
555
596
  public synchronized void stopForeground() {
556
597
  if (sIsRunning && mIsInForeground) {
598
+ mMainHandler.removeCallbacks(mNotificationUpdateRunnable);
557
599
  stopForeground(true);
558
600
  if (mProvider != null) {
559
601
  mProvider.onCommand(LocationProvider.CMD_SWITCH_MODE,
@@ -563,6 +605,70 @@ public class LocationServiceImpl extends Service implements ProviderDelegate, Lo
563
605
  }
564
606
  }
565
607
 
608
+ /** Resource names for optional app-localized notification labels (showTime / showDistance). */
609
+ private static final String RES_NOTIFICATION_TIME_LABEL = "plugin_bgloc_notification_time_label";
610
+ private static final String RES_NOTIFICATION_DISTANCE_LABEL = "plugin_bgloc_notification_distance_label";
611
+
612
+ private String getNotificationLabel(String resourceName, String defaultValue) {
613
+ Context app = getApplicationContext();
614
+ int id = app.getResources().getIdentifier(resourceName, "string", app.getPackageName());
615
+ return (id != 0) ? app.getString(id) : defaultValue;
616
+ }
617
+
618
+ private String buildNotificationContentText(Config config) {
619
+ String base = config.getNotificationText() != null ? config.getNotificationText() : "ENABLED";
620
+ if (Boolean.TRUE.equals(config.getShowTime())) {
621
+ String timeLabel = getNotificationLabel(RES_NOTIFICATION_TIME_LABEL, "Time");
622
+ base += "\n" + timeLabel + ": " + formatElapsed(mSessionStartTime);
623
+ }
624
+ if (Boolean.TRUE.equals(config.getShowDistance())) {
625
+ String distanceLabel = getNotificationLabel(RES_NOTIFICATION_DISTANCE_LABEL, "Distance");
626
+ base += "\n" + distanceLabel + ": " + formatDistance(mSessionDistanceMeters);
627
+ }
628
+ return base;
629
+ }
630
+
631
+ private static String formatElapsed(long startTimeMs) {
632
+ long elapsed = Math.max(0L, System.currentTimeMillis() - startTimeMs);
633
+ long s = (elapsed / 1000L) % 60L;
634
+ long m = (elapsed / 60000L) % 60L;
635
+ long h = elapsed / 3600000L;
636
+ return String.format(Locale.US, "%02d:%02d:%02d", h, m, s);
637
+ }
638
+
639
+ private static String formatDistance(double meters) {
640
+ return String.format(Locale.US, "%.2f km", meters / 1000.0);
641
+ }
642
+
643
+ private void updateForegroundNotification() {
644
+ if (!sIsRunning || !mIsInForeground || mConfig == null) {
645
+ return;
646
+ }
647
+ String contentText = buildNotificationContentText(mConfig);
648
+ Notification notification = new NotificationHelper.NotificationFactory(this).getNotification(
649
+ mConfig.getNotificationTitle(),
650
+ contentText,
651
+ mConfig.getLargeNotificationIcon(),
652
+ mConfig.getSmallNotificationIcon(),
653
+ mConfig.getNotificationIconColor());
654
+ NotificationManager nm = (NotificationManager) getSystemService(Context.NOTIFICATION_SERVICE);
655
+ if (nm != null) {
656
+ nm.notify(NOTIFICATION_ID, notification);
657
+ }
658
+ }
659
+
660
+ private void scheduleNotificationUpdater() {
661
+ mMainHandler.removeCallbacks(mNotificationUpdateRunnable);
662
+ if (mConfig == null) {
663
+ return;
664
+ }
665
+ boolean showTime = Boolean.TRUE.equals(mConfig.getShowTime());
666
+ boolean showDistance = Boolean.TRUE.equals(mConfig.getShowDistance());
667
+ if ((showTime || showDistance) && sIsRunning && mIsInForeground) {
668
+ mMainHandler.postDelayed(mNotificationUpdateRunnable, NOTIFICATION_UPDATE_INTERVAL_MS);
669
+ }
670
+ }
671
+
566
672
  @Override
567
673
  public synchronized void configure(Config config) {
568
674
  if (mConfig == null) {
@@ -580,7 +686,7 @@ public class LocationServiceImpl extends Service implements ProviderDelegate, Lo
580
686
  public void run() {
581
687
  if (sIsRunning) {
582
688
  if (currentConfig.getStartForeground() == true && mConfig.getStartForeground() == false) {
583
- stopForeground(true);
689
+ stopForeground();
584
690
  }
585
691
 
586
692
  if (mConfig.getStartForeground() == true) {
@@ -589,15 +695,17 @@ public class LocationServiceImpl extends Service implements ProviderDelegate, Lo
589
695
  startForeground();
590
696
  } else {
591
697
  // was running in foreground, so just update existing notification
698
+ String contentText = buildNotificationContentText(mConfig);
592
699
  Notification notification = new NotificationHelper.NotificationFactory(LocationServiceImpl.this).getNotification(
593
700
  mConfig.getNotificationTitle(),
594
- mConfig.getNotificationText(),
701
+ contentText,
595
702
  mConfig.getLargeNotificationIcon(),
596
703
  mConfig.getSmallNotificationIcon(),
597
704
  mConfig.getNotificationIconColor());
598
705
 
599
706
  NotificationManager notificationManager = (NotificationManager) getSystemService(Context.NOTIFICATION_SERVICE);
600
707
  notificationManager.notify(NOTIFICATION_ID, notification);
708
+ scheduleNotificationUpdater();
601
709
  }
602
710
  }
603
711
  }
@@ -661,6 +769,26 @@ public class LocationServiceImpl extends Service implements ProviderDelegate, Lo
661
769
  @Override
662
770
  public void onLocation(BackgroundLocation location) {
663
771
  mLastLocationTime = System.currentTimeMillis();
772
+ if (Boolean.TRUE.equals(mConfig != null ? mConfig.getShowDistance() : null)) {
773
+ double lat = location.getLatitude();
774
+ double lon = location.getLongitude();
775
+ if (mHasLastLocation) {
776
+ float[] dist = new float[1];
777
+ Location.distanceBetween(mLastLat, mLastLon, lat, lon, dist);
778
+ mSessionDistanceMeters += (double) dist[0];
779
+ }
780
+ mLastLat = lat;
781
+ mLastLon = lon;
782
+ mHasLastLocation = true;
783
+ if (mIsInForeground && mConfig != null && (Boolean.TRUE.equals(mConfig.getShowTime()) || Boolean.TRUE.equals(mConfig.getShowDistance()))) {
784
+ mMainHandler.post(new Runnable() {
785
+ @Override
786
+ public void run() {
787
+ updateForegroundNotification();
788
+ }
789
+ });
790
+ }
791
+ }
664
792
  logger.debug("New location {}", location.toString());
665
793
 
666
794
  location = transformLocation(location);
@@ -168,6 +168,34 @@ export class BackgroundGeolocationService {
168
168
  return this.ensurePlugin().getPendingSyncCount(success, fail);
169
169
  }
170
170
 
171
+ startSession(
172
+ success?: () => void,
173
+ fail?: (error: any) => void
174
+ ): Promise<void> {
175
+ return this.ensurePlugin().startSession(success, fail);
176
+ }
177
+
178
+ getSessionLocations(
179
+ success?: (locations: any[]) => void,
180
+ fail?: (error: any) => void
181
+ ): Promise<any[]> {
182
+ return this.ensurePlugin().getSessionLocations(success, fail);
183
+ }
184
+
185
+ clearSession(
186
+ success?: () => void,
187
+ fail?: (error: any) => void
188
+ ): Promise<void> {
189
+ return this.ensurePlugin().clearSession(success, fail);
190
+ }
191
+
192
+ getSessionLocationsCount(
193
+ success?: (count: number) => void,
194
+ fail?: (error: any) => void
195
+ ): Promise<number> {
196
+ return this.ensurePlugin().getSessionLocationsCount(success, fail);
197
+ }
198
+
171
199
  getConfig(
172
200
  success?: (config: any) => void,
173
201
  fail?: (error: any) => void
@@ -45,6 +45,10 @@ export declare class BackgroundGeolocationService {
45
45
  forceSync(success?: () => void, fail?: (error: any) => void): Promise<void>;
46
46
  clearSync(success?: () => void, fail?: (error: any) => void): Promise<void>;
47
47
  getPendingSyncCount(success?: (count: number) => void, fail?: (error: any) => void): Promise<number>;
48
+ startSession(success?: () => void, fail?: (error: any) => void): Promise<void>;
49
+ getSessionLocations(success?: (locations: any[]) => void, fail?: (error: any) => void): Promise<any[]>;
50
+ clearSession(success?: () => void, fail?: (error: any) => void): Promise<void>;
51
+ getSessionLocationsCount(success?: (count: number) => void, fail?: (error: any) => void): Promise<number>;
48
52
  getConfig(success?: (config: any) => void, fail?: (error: any) => void): Promise<any>;
49
53
  getLogEntries(limit: number, fromId: number, minLevel: string, success?: (entries: any[]) => void, fail?: (error: any) => void): Promise<any[]>;
50
54
  removeAllListeners(event?: string): void;
@@ -97,6 +97,18 @@ export class BackgroundGeolocationService {
97
97
  getPendingSyncCount(success, fail) {
98
98
  return this.ensurePlugin().getPendingSyncCount(success, fail);
99
99
  }
100
+ startSession(success, fail) {
101
+ return this.ensurePlugin().startSession(success, fail);
102
+ }
103
+ getSessionLocations(success, fail) {
104
+ return this.ensurePlugin().getSessionLocations(success, fail);
105
+ }
106
+ clearSession(success, fail) {
107
+ return this.ensurePlugin().clearSession(success, fail);
108
+ }
109
+ getSessionLocationsCount(success, fail) {
110
+ return this.ensurePlugin().getSessionLocationsCount(success, fail);
111
+ }
100
112
  getConfig(success, fail) {
101
113
  return this.ensurePlugin().getConfig(success, fail);
102
114
  }
@@ -151,4 +163,4 @@ export class BackgroundGeolocationService {
151
163
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: BackgroundGeolocationService, decorators: [{
152
164
  type: Injectable
153
165
  }] });
154
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"background-geolocation.service.js","sourceRoot":"","sources":["../../background-geolocation.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, InjectionToken } from '@angular/core';\n\n/**\n * Token used to provide BackgroundGeolocationService without triggering JIT.\n * The module provides this token with useFactory; the class is aliased via useExisting\n * so you still inject constructor(private bg: BackgroundGeolocationService) {}.\n */\nexport const BACKGROUND_GEOLOCATION_SERVICE = new InjectionToken<BackgroundGeolocationService>(\n  'BackgroundGeolocationService'\n);\n\n/**\n * Angular service that wraps the Cordova/Capacitor BackgroundGeolocation plugin.\n * Use dependency injection instead of the global BackgroundGeolocation object.\n *\n * The native plugin must be installed and available (e.g. after deviceready).\n * Types (ConfigureOptions, Location, etc.) can be imported from\n * '@josuelmm/cordova-background-geolocation'.\n *\n * Provided via BackgroundGeolocationModule using an InjectionToken + useFactory\n * so AOT builds never need the JIT compiler for this class.\n */\n@Injectable()\nexport class BackgroundGeolocationService {\n\n  /** Returns the global plugin instance (Cordova/Capacitor). */\n  private get plugin(): any {\n    if (typeof window === 'undefined') return null;\n    return (window as any).BackgroundGeolocation || null;\n  }\n\n  private ensurePlugin(): any {\n    const p = this.plugin;\n    if (!p) {\n      throw new Error(\n        'BackgroundGeolocation is not available. Ensure the plugin is installed and the app is running in a native context (Cordova/Capacitor).'\n      );\n    }\n    return p;\n  }\n\n  configure(options: any, success?: () => void, fail?: (err: any) => void): Promise<void> {\n    return this.ensurePlugin().configure(options, success, fail);\n  }\n\n  start(): Promise<void> {\n    return this.ensurePlugin().start();\n  }\n\n  stop(): Promise<void> {\n    return this.ensurePlugin().stop();\n  }\n\n  /** Inform the native plugin that the background task may complete (iOS). Call after handling location/stationary in the callback. */\n  finish(): Promise<void> {\n    return this.ensurePlugin().finish();\n  }\n\n  /** Force the plugin to enter \"moving\" or \"stationary\" state (iOS). */\n  changePace(isMoving: boolean): Promise<void> {\n    return this.ensurePlugin().changePace(isMoving);\n  }\n\n  getCurrentLocation(\n    success?: (location: any) => void,\n    fail?: (error: any) => void,\n    options?: any\n  ): Promise<any> {\n    return this.ensurePlugin().getCurrentLocation(success, fail, options);\n  }\n\n  getStationaryLocation(\n    success?: (location: any) => void,\n    fail?: (error: any) => void\n  ): Promise<any> {\n    return this.ensurePlugin().getStationaryLocation(success, fail);\n  }\n\n  checkStatus(\n    success?: (status: any) => void,\n    fail?: (error: any) => void\n  ): Promise<any> {\n    return this.ensurePlugin().checkStatus(success, fail);\n  }\n\n  showAppSettings(): Promise<void> {\n    return this.ensurePlugin().showAppSettings();\n  }\n\n  /** Open app settings (alias for showAppSettings). */\n  openSettings(): Promise<void> {\n    return this.ensurePlugin().openSettings();\n  }\n\n  showLocationSettings(): Promise<void> {\n    return this.ensurePlugin().showLocationSettings();\n  }\n\n  getPluginVersion(\n    success?: (version: string) => void,\n    fail?: (error: any) => void\n  ): Promise<string> {\n    return this.ensurePlugin().getPluginVersion(success, fail);\n  }\n\n  getLocations(\n    success?: (locations: any[]) => void,\n    fail?: (error: any) => void\n  ): Promise<any[]> {\n    return this.ensurePlugin().getLocations(success, fail);\n  }\n\n  getValidLocations(\n    success?: (locations: any[]) => void,\n    fail?: (error: any) => void\n  ): Promise<any[]> {\n    return this.ensurePlugin().getValidLocations(success, fail);\n  }\n\n  getValidLocationsAndDelete(\n    success?: (locations: any[]) => void,\n    fail?: (error: any) => void\n  ): Promise<any[]> {\n    return this.ensurePlugin().getValidLocationsAndDelete(success, fail);\n  }\n\n  deleteLocation(\n    locationId: number,\n    success?: () => void,\n    fail?: (error: any) => void\n  ): Promise<void> {\n    return this.ensurePlugin().deleteLocation(locationId, success, fail);\n  }\n\n  deleteAllLocations(\n    success?: () => void,\n    fail?: (error: any) => void\n  ): Promise<void> {\n    return this.ensurePlugin().deleteAllLocations(success, fail);\n  }\n\n  switchMode(\n    modeId: number,\n    success?: () => void,\n    fail?: (error: any) => void\n  ): Promise<void> {\n    return this.ensurePlugin().switchMode(modeId, success, fail);\n  }\n\n  forceSync(\n    success?: () => void,\n    fail?: (error: any) => void\n  ): Promise<void> {\n    return this.ensurePlugin().forceSync(success, fail);\n  }\n\n  clearSync(\n    success?: () => void,\n    fail?: (error: any) => void\n  ): Promise<void> {\n    return this.ensurePlugin().clearSync(success, fail);\n  }\n\n  getPendingSyncCount(\n    success?: (count: number) => void,\n    fail?: (error: any) => void\n  ): Promise<number> {\n    return this.ensurePlugin().getPendingSyncCount(success, fail);\n  }\n\n  getConfig(\n    success?: (config: any) => void,\n    fail?: (error: any) => void\n  ): Promise<any> {\n    return this.ensurePlugin().getConfig(success, fail);\n  }\n\n  getLogEntries(\n    limit: number,\n    fromId: number,\n    minLevel: string,\n    success?: (entries: any[]) => void,\n    fail?: (error: any) => void\n  ): Promise<any[]> {\n    return this.ensurePlugin().getLogEntries(limit, fromId, minLevel, success, fail);\n  }\n\n  removeAllListeners(event?: string): void {\n    this.ensurePlugin().removeAllListeners(event);\n  }\n\n  startTask(\n    success?: (taskKey: number) => void,\n    fail?: (error: any) => void\n  ): Promise<number> {\n    return this.ensurePlugin().startTask(success, fail);\n  }\n\n  endTask(\n    taskKey: number,\n    success?: () => void,\n    fail?: (error: any) => void\n  ): Promise<void> {\n    return this.ensurePlugin().endTask(taskKey, success, fail);\n  }\n\n  headlessTask(task: (event: any) => void): void {\n    this.ensurePlugin().headlessTask(task);\n  }\n\n  /**\n   * Register an event listener. Compatible with Awesome-style usage:\n   * .on(BackgroundGeolocationEvents.error).subscribe((err) => ...) — returns subscription with .unsubscribe().\n   * .on('location', (loc) => ...) — also supported; returned object has .unsubscribe().\n   */\n  on(eventName: string, callback?: (value: any) => void): { subscribe(cb: (value: any) => void): { unsubscribe(): void }; unsubscribe(): void } {\n    const plugin = this.ensurePlugin();\n    if (callback !== undefined) {\n      const sub = plugin.on(eventName, callback) as { remove?: () => void };\n      return {\n        subscribe(cb: (value: any) => void) {\n          const s = cb ? (plugin.on(eventName, cb) as { remove?: () => void }) : null;\n          return { unsubscribe() { s?.remove?.(); } };\n        },\n        unsubscribe() { sub.remove?.(); }\n      };\n    }\n    const channel = plugin.on(eventName) as { subscribe: (cb: (v: any) => void) => void; unsubscribe: (cb: (v: any) => void) => void };\n    return {\n      subscribe(cb: (value: any) => void) {\n        channel.subscribe(cb);\n        return { unsubscribe() { channel.unsubscribe(cb); } };\n      },\n      unsubscribe() { /* no-op when no callback */ }\n    };\n  }\n\n  /** Convenience: access plugin constants (e.g. ACTIVITY_PROVIDER, HIGH_ACCURACY). */\n  get native(): any {\n    return this.plugin;\n  }\n}\n"]}
166
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"background-geolocation.service.js","sourceRoot":"","sources":["../../background-geolocation.service.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,cAAc,EAAE,MAAM,eAAe,CAAC;;AAE3D;;;;GAIG;AACH,MAAM,CAAC,MAAM,8BAA8B,GAAG,IAAI,cAAc,CAC9D,8BAA8B,CAC/B,CAAC;AAEF;;;;;;;;;;GAUG;AAEH,MAAM,OAAO,4BAA4B;IAEvC,8DAA8D;IAC9D,IAAY,MAAM;QAChB,IAAI,OAAO,MAAM,KAAK,WAAW;YAAE,OAAO,IAAI,CAAC;QAC/C,OAAQ,MAAc,CAAC,qBAAqB,IAAI,IAAI,CAAC;IACvD,CAAC;IAEO,YAAY;QAClB,MAAM,CAAC,GAAG,IAAI,CAAC,MAAM,CAAC;QACtB,IAAI,CAAC,CAAC,EAAE,CAAC;YACP,MAAM,IAAI,KAAK,CACb,wIAAwI,CACzI,CAAC;QACJ,CAAC;QACD,OAAO,CAAC,CAAC;IACX,CAAC;IAED,SAAS,CAAC,OAAY,EAAE,OAAoB,EAAE,IAAyB;QACrE,OAAO,IAAI,CAAC,YAAY,EAAE,CAAC,SAAS,CAAC,OAAO,EAAE,OAAO,EAAE,IAAI,CAAC,CAAC;IAC/D,CAAC;IAED,KAAK;QACH,OAAO,IAAI,CAAC,YAAY,EAAE,CAAC,KAAK,EAAE,CAAC;IACrC,CAAC;IAED,IAAI;QACF,OAAO,IAAI,CAAC,YAAY,EAAE,CAAC,IAAI,EAAE,CAAC;IACpC,CAAC;IAED,qIAAqI;IACrI,MAAM;QACJ,OAAO,IAAI,CAAC,YAAY,EAAE,CAAC,MAAM,EAAE,CAAC;IACtC,CAAC;IAED,sEAAsE;IACtE,UAAU,CAAC,QAAiB;QAC1B,OAAO,IAAI,CAAC,YAAY,EAAE,CAAC,UAAU,CAAC,QAAQ,CAAC,CAAC;IAClD,CAAC;IAED,kBAAkB,CAChB,OAAiC,EACjC,IAA2B,EAC3B,OAAa;QAEb,OAAO,IAAI,CAAC,YAAY,EAAE,CAAC,kBAAkB,CAAC,OAAO,EAAE,IAAI,EAAE,OAAO,CAAC,CAAC;IACxE,CAAC;IAED,qBAAqB,CACnB,OAAiC,EACjC,IAA2B;QAE3B,OAAO,IAAI,CAAC,YAAY,EAAE,CAAC,qBAAqB,CAAC,OAAO,EAAE,IAAI,CAAC,CAAC;IAClE,CAAC;IAED,WAAW,CACT,OAA+B,EAC/B,IAA2B;QAE3B,OAAO,IAAI,CAAC,YAAY,EAAE,CAAC,WAAW,CAAC,OAAO,EAAE,IAAI,CAAC,CAAC;IACxD,CAAC;IAED,eAAe;QACb,OAAO,IAAI,CAAC,YAAY,EAAE,CAAC,eAAe,EAAE,CAAC;IAC/C,CAAC;IAED,qDAAqD;IACrD,YAAY;QACV,OAAO,IAAI,CAAC,YAAY,EAAE,CAAC,YAAY,EAAE,CAAC;IAC5C,CAAC;IAED,oBAAoB;QAClB,OAAO,IAAI,CAAC,YAAY,EAAE,CAAC,oBAAoB,EAAE,CAAC;IACpD,CAAC;IAED,gBAAgB,CACd,OAAmC,EACnC,IAA2B;QAE3B,OAAO,IAAI,CAAC,YAAY,EAAE,CAAC,gBAAgB,CAAC,OAAO,EAAE,IAAI,CAAC,CAAC;IAC7D,CAAC;IAED,YAAY,CACV,OAAoC,EACpC,IAA2B;QAE3B,OAAO,IAAI,CAAC,YAAY,EAAE,CAAC,YAAY,CAAC,OAAO,EAAE,IAAI,CAAC,CAAC;IACzD,CAAC;IAED,iBAAiB,CACf,OAAoC,EACpC,IAA2B;QAE3B,OAAO,IAAI,CAAC,YAAY,EAAE,CAAC,iBAAiB,CAAC,OAAO,EAAE,IAAI,CAAC,CAAC;IAC9D,CAAC;IAED,0BAA0B,CACxB,OAAoC,EACpC,IAA2B;QAE3B,OAAO,IAAI,CAAC,YAAY,EAAE,CAAC,0BAA0B,CAAC,OAAO,EAAE,IAAI,CAAC,CAAC;IACvE,CAAC;IAED,cAAc,CACZ,UAAkB,EAClB,OAAoB,EACpB,IAA2B;QAE3B,OAAO,IAAI,CAAC,YAAY,EAAE,CAAC,cAAc,CAAC,UAAU,EAAE,OAAO,EAAE,IAAI,CAAC,CAAC;IACvE,CAAC;IAED,kBAAkB,CAChB,OAAoB,EACpB,IAA2B;QAE3B,OAAO,IAAI,CAAC,YAAY,EAAE,CAAC,kBAAkB,CAAC,OAAO,EAAE,IAAI,CAAC,CAAC;IAC/D,CAAC;IAED,UAAU,CACR,MAAc,EACd,OAAoB,EACpB,IAA2B;QAE3B,OAAO,IAAI,CAAC,YAAY,EAAE,CAAC,UAAU,CAAC,MAAM,EAAE,OAAO,EAAE,IAAI,CAAC,CAAC;IAC/D,CAAC;IAED,SAAS,CACP,OAAoB,EACpB,IAA2B;QAE3B,OAAO,IAAI,CAAC,YAAY,EAAE,CAAC,SAAS,CAAC,OAAO,EAAE,IAAI,CAAC,CAAC;IACtD,CAAC;IAED,SAAS,CACP,OAAoB,EACpB,IAA2B;QAE3B,OAAO,IAAI,CAAC,YAAY,EAAE,CAAC,SAAS,CAAC,OAAO,EAAE,IAAI,CAAC,CAAC;IACtD,CAAC;IAED,mBAAmB,CACjB,OAAiC,EACjC,IAA2B;QAE3B,OAAO,IAAI,CAAC,YAAY,EAAE,CAAC,mBAAmB,CAAC,OAAO,EAAE,IAAI,CAAC,CAAC;IAChE,CAAC;IAED,YAAY,CACV,OAAoB,EACpB,IAA2B;QAE3B,OAAO,IAAI,CAAC,YAAY,EAAE,CAAC,YAAY,CAAC,OAAO,EAAE,IAAI,CAAC,CAAC;IACzD,CAAC;IAED,mBAAmB,CACjB,OAAoC,EACpC,IAA2B;QAE3B,OAAO,IAAI,CAAC,YAAY,EAAE,CAAC,mBAAmB,CAAC,OAAO,EAAE,IAAI,CAAC,CAAC;IAChE,CAAC;IAED,YAAY,CACV,OAAoB,EACpB,IAA2B;QAE3B,OAAO,IAAI,CAAC,YAAY,EAAE,CAAC,YAAY,CAAC,OAAO,EAAE,IAAI,CAAC,CAAC;IACzD,CAAC;IAED,wBAAwB,CACtB,OAAiC,EACjC,IAA2B;QAE3B,OAAO,IAAI,CAAC,YAAY,EAAE,CAAC,wBAAwB,CAAC,OAAO,EAAE,IAAI,CAAC,CAAC;IACrE,CAAC;IAED,SAAS,CACP,OAA+B,EAC/B,IAA2B;QAE3B,OAAO,IAAI,CAAC,YAAY,EAAE,CAAC,SAAS,CAAC,OAAO,EAAE,IAAI,CAAC,CAAC;IACtD,CAAC;IAED,aAAa,CACX,KAAa,EACb,MAAc,EACd,QAAgB,EAChB,OAAkC,EAClC,IAA2B;QAE3B,OAAO,IAAI,CAAC,YAAY,EAAE,CAAC,aAAa,CAAC,KAAK,EAAE,MAAM,EAAE,QAAQ,EAAE,OAAO,EAAE,IAAI,CAAC,CAAC;IACnF,CAAC;IAED,kBAAkB,CAAC,KAAc;QAC/B,IAAI,CAAC,YAAY,EAAE,CAAC,kBAAkB,CAAC,KAAK,CAAC,CAAC;IAChD,CAAC;IAED,SAAS,CACP,OAAmC,EACnC,IAA2B;QAE3B,OAAO,IAAI,CAAC,YAAY,EAAE,CAAC,SAAS,CAAC,OAAO,EAAE,IAAI,CAAC,CAAC;IACtD,CAAC;IAED,OAAO,CACL,OAAe,EACf,OAAoB,EACpB,IAA2B;QAE3B,OAAO,IAAI,CAAC,YAAY,EAAE,CAAC,OAAO,CAAC,OAAO,EAAE,OAAO,EAAE,IAAI,CAAC,CAAC;IAC7D,CAAC;IAED,YAAY,CAAC,IAA0B;QACrC,IAAI,CAAC,YAAY,EAAE,CAAC,YAAY,CAAC,IAAI,CAAC,CAAC;IACzC,CAAC;IAED;;;;OAIG;IACH,EAAE,CAAC,SAAiB,EAAE,QAA+B;QACnD,MAAM,MAAM,GAAG,IAAI,CAAC,YAAY,EAAE,CAAC;QACnC,IAAI,QAAQ,KAAK,SAAS,EAAE,CAAC;YAC3B,MAAM,GAAG,GAAG,MAAM,CAAC,EAAE,CAAC,SAAS,EAAE,QAAQ,CAA4B,CAAC;YACtE,OAAO;gBACL,SAAS,CAAC,EAAwB;oBAChC,MAAM,CAAC,GAAG,EAAE,CAAC,CAAC,CAAE,MAAM,CAAC,EAAE,CAAC,SAAS,EAAE,EAAE,CAA6B,CAAC,CAAC,CAAC,IAAI,CAAC;oBAC5E,OAAO,EAAE,WAAW,KAAK,CAAC,EAAE,MAAM,EAAE,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC;gBAC9C,CAAC;gBACD,WAAW,KAAK,GAAG,CAAC,MAAM,EAAE,EAAE,CAAC,CAAC,CAAC;aAClC,CAAC;QACJ,CAAC;QACD,MAAM,OAAO,GAAG,MAAM,CAAC,EAAE,CAAC,SAAS,CAA+F,CAAC;QACnI,OAAO;YACL,SAAS,CAAC,EAAwB;gBAChC,OAAO,CAAC,SAAS,CAAC,EAAE,CAAC,CAAC;gBACtB,OAAO,EAAE,WAAW,KAAK,OAAO,CAAC,WAAW,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC;YACxD,CAAC;YACD,WAAW,KAAkC,CAAC;SAC/C,CAAC;IACJ,CAAC;IAED,oFAAoF;IACpF,IAAI,MAAM;QACR,OAAO,IAAI,CAAC,MAAM,CAAC;IACrB,CAAC;wGArPU,4BAA4B;4GAA5B,4BAA4B;;4FAA5B,4BAA4B;kBADxC,UAAU","sourcesContent":["import { Injectable, InjectionToken } from '@angular/core';\n\n/**\n * Token used to provide BackgroundGeolocationService without triggering JIT.\n * The module provides this token with useFactory; the class is aliased via useExisting\n * so you still inject constructor(private bg: BackgroundGeolocationService) {}.\n */\nexport const BACKGROUND_GEOLOCATION_SERVICE = new InjectionToken<BackgroundGeolocationService>(\n  'BackgroundGeolocationService'\n);\n\n/**\n * Angular service that wraps the Cordova/Capacitor BackgroundGeolocation plugin.\n * Use dependency injection instead of the global BackgroundGeolocation object.\n *\n * The native plugin must be installed and available (e.g. after deviceready).\n * Types (ConfigureOptions, Location, etc.) can be imported from\n * '@josuelmm/cordova-background-geolocation'.\n *\n * Provided via BackgroundGeolocationModule using an InjectionToken + useFactory\n * so AOT builds never need the JIT compiler for this class.\n */\n@Injectable()\nexport class BackgroundGeolocationService {\n\n  /** Returns the global plugin instance (Cordova/Capacitor). */\n  private get plugin(): any {\n    if (typeof window === 'undefined') return null;\n    return (window as any).BackgroundGeolocation || null;\n  }\n\n  private ensurePlugin(): any {\n    const p = this.plugin;\n    if (!p) {\n      throw new Error(\n        'BackgroundGeolocation is not available. Ensure the plugin is installed and the app is running in a native context (Cordova/Capacitor).'\n      );\n    }\n    return p;\n  }\n\n  configure(options: any, success?: () => void, fail?: (err: any) => void): Promise<void> {\n    return this.ensurePlugin().configure(options, success, fail);\n  }\n\n  start(): Promise<void> {\n    return this.ensurePlugin().start();\n  }\n\n  stop(): Promise<void> {\n    return this.ensurePlugin().stop();\n  }\n\n  /** Inform the native plugin that the background task may complete (iOS). Call after handling location/stationary in the callback. */\n  finish(): Promise<void> {\n    return this.ensurePlugin().finish();\n  }\n\n  /** Force the plugin to enter \"moving\" or \"stationary\" state (iOS). */\n  changePace(isMoving: boolean): Promise<void> {\n    return this.ensurePlugin().changePace(isMoving);\n  }\n\n  getCurrentLocation(\n    success?: (location: any) => void,\n    fail?: (error: any) => void,\n    options?: any\n  ): Promise<any> {\n    return this.ensurePlugin().getCurrentLocation(success, fail, options);\n  }\n\n  getStationaryLocation(\n    success?: (location: any) => void,\n    fail?: (error: any) => void\n  ): Promise<any> {\n    return this.ensurePlugin().getStationaryLocation(success, fail);\n  }\n\n  checkStatus(\n    success?: (status: any) => void,\n    fail?: (error: any) => void\n  ): Promise<any> {\n    return this.ensurePlugin().checkStatus(success, fail);\n  }\n\n  showAppSettings(): Promise<void> {\n    return this.ensurePlugin().showAppSettings();\n  }\n\n  /** Open app settings (alias for showAppSettings). */\n  openSettings(): Promise<void> {\n    return this.ensurePlugin().openSettings();\n  }\n\n  showLocationSettings(): Promise<void> {\n    return this.ensurePlugin().showLocationSettings();\n  }\n\n  getPluginVersion(\n    success?: (version: string) => void,\n    fail?: (error: any) => void\n  ): Promise<string> {\n    return this.ensurePlugin().getPluginVersion(success, fail);\n  }\n\n  getLocations(\n    success?: (locations: any[]) => void,\n    fail?: (error: any) => void\n  ): Promise<any[]> {\n    return this.ensurePlugin().getLocations(success, fail);\n  }\n\n  getValidLocations(\n    success?: (locations: any[]) => void,\n    fail?: (error: any) => void\n  ): Promise<any[]> {\n    return this.ensurePlugin().getValidLocations(success, fail);\n  }\n\n  getValidLocationsAndDelete(\n    success?: (locations: any[]) => void,\n    fail?: (error: any) => void\n  ): Promise<any[]> {\n    return this.ensurePlugin().getValidLocationsAndDelete(success, fail);\n  }\n\n  deleteLocation(\n    locationId: number,\n    success?: () => void,\n    fail?: (error: any) => void\n  ): Promise<void> {\n    return this.ensurePlugin().deleteLocation(locationId, success, fail);\n  }\n\n  deleteAllLocations(\n    success?: () => void,\n    fail?: (error: any) => void\n  ): Promise<void> {\n    return this.ensurePlugin().deleteAllLocations(success, fail);\n  }\n\n  switchMode(\n    modeId: number,\n    success?: () => void,\n    fail?: (error: any) => void\n  ): Promise<void> {\n    return this.ensurePlugin().switchMode(modeId, success, fail);\n  }\n\n  forceSync(\n    success?: () => void,\n    fail?: (error: any) => void\n  ): Promise<void> {\n    return this.ensurePlugin().forceSync(success, fail);\n  }\n\n  clearSync(\n    success?: () => void,\n    fail?: (error: any) => void\n  ): Promise<void> {\n    return this.ensurePlugin().clearSync(success, fail);\n  }\n\n  getPendingSyncCount(\n    success?: (count: number) => void,\n    fail?: (error: any) => void\n  ): Promise<number> {\n    return this.ensurePlugin().getPendingSyncCount(success, fail);\n  }\n\n  startSession(\n    success?: () => void,\n    fail?: (error: any) => void\n  ): Promise<void> {\n    return this.ensurePlugin().startSession(success, fail);\n  }\n\n  getSessionLocations(\n    success?: (locations: any[]) => void,\n    fail?: (error: any) => void\n  ): Promise<any[]> {\n    return this.ensurePlugin().getSessionLocations(success, fail);\n  }\n\n  clearSession(\n    success?: () => void,\n    fail?: (error: any) => void\n  ): Promise<void> {\n    return this.ensurePlugin().clearSession(success, fail);\n  }\n\n  getSessionLocationsCount(\n    success?: (count: number) => void,\n    fail?: (error: any) => void\n  ): Promise<number> {\n    return this.ensurePlugin().getSessionLocationsCount(success, fail);\n  }\n\n  getConfig(\n    success?: (config: any) => void,\n    fail?: (error: any) => void\n  ): Promise<any> {\n    return this.ensurePlugin().getConfig(success, fail);\n  }\n\n  getLogEntries(\n    limit: number,\n    fromId: number,\n    minLevel: string,\n    success?: (entries: any[]) => void,\n    fail?: (error: any) => void\n  ): Promise<any[]> {\n    return this.ensurePlugin().getLogEntries(limit, fromId, minLevel, success, fail);\n  }\n\n  removeAllListeners(event?: string): void {\n    this.ensurePlugin().removeAllListeners(event);\n  }\n\n  startTask(\n    success?: (taskKey: number) => void,\n    fail?: (error: any) => void\n  ): Promise<number> {\n    return this.ensurePlugin().startTask(success, fail);\n  }\n\n  endTask(\n    taskKey: number,\n    success?: () => void,\n    fail?: (error: any) => void\n  ): Promise<void> {\n    return this.ensurePlugin().endTask(taskKey, success, fail);\n  }\n\n  headlessTask(task: (event: any) => void): void {\n    this.ensurePlugin().headlessTask(task);\n  }\n\n  /**\n   * Register an event listener. Compatible with Awesome-style usage:\n   * .on(BackgroundGeolocationEvents.error).subscribe((err) => ...) — returns subscription with .unsubscribe().\n   * .on('location', (loc) => ...) — also supported; returned object has .unsubscribe().\n   */\n  on(eventName: string, callback?: (value: any) => void): { subscribe(cb: (value: any) => void): { unsubscribe(): void }; unsubscribe(): void } {\n    const plugin = this.ensurePlugin();\n    if (callback !== undefined) {\n      const sub = plugin.on(eventName, callback) as { remove?: () => void };\n      return {\n        subscribe(cb: (value: any) => void) {\n          const s = cb ? (plugin.on(eventName, cb) as { remove?: () => void }) : null;\n          return { unsubscribe() { s?.remove?.(); } };\n        },\n        unsubscribe() { sub.remove?.(); }\n      };\n    }\n    const channel = plugin.on(eventName) as { subscribe: (cb: (v: any) => void) => void; unsubscribe: (cb: (v: any) => void) => void };\n    return {\n      subscribe(cb: (value: any) => void) {\n        channel.subscribe(cb);\n        return { unsubscribe() { channel.unsubscribe(cb); } };\n      },\n      unsubscribe() { /* no-op when no callback */ }\n    };\n  }\n\n  /** Convenience: access plugin constants (e.g. ACTIVITY_PROVIDER, HIGH_ACCURACY). */\n  get native(): any {\n    return this.plugin;\n  }\n}\n"]}