16
16
17
17
package io .appium .android .apis .app ;
18
18
19
+ import android .annotation .SuppressLint ;
19
20
import android .app .Activity ;
20
21
import android .app .Notification ;
21
22
import android .app .NotificationManager ;
22
23
import android .app .PendingIntent ;
23
24
import android .app .Service ;
24
25
import android .content .Intent ;
26
+ import android .os .Build ;
25
27
import android .os .Bundle ;
28
+ import android .os .Handler ;
26
29
import android .os .IBinder ;
30
+ import android .os .PowerManager ;
27
31
import android .util .Log ;
28
32
import android .view .View ;
29
33
import android .view .View .OnClickListener ;
46
50
*/
47
51
public class ForegroundService extends Service {
48
52
static final String ACTION_FOREGROUND = "io.appium.android.apis.FOREGROUND" ;
53
+ static final String ACTION_FOREGROUND_WAKELOCK = "io.appium.android.apis.FOREGROUND_WAKELOCK" ;
49
54
static final String ACTION_BACKGROUND = "io.appium.android.apis.BACKGROUND" ;
50
-
55
+ static final String ACTION_BACKGROUND_WAKELOCK = "io.appium.android.apis.BACKGROUND_WAKELOCK" ;
56
+
57
+ private PowerManager .WakeLock mWakeLock ;
58
+ private Handler mHandler = new Handler ();
59
+ private Runnable mPulser = new Runnable () {
60
+ @ Override public void run () {
61
+ Log .i ("ForegroundService" , "PULSE!" );
62
+ mHandler .postDelayed (this , 5 *1000 );
63
+ }
64
+ };
51
65
52
66
private static final Class <?>[] mSetForegroundSignature = new Class [] {
53
67
boolean .class };
54
68
private static final Class <?>[] mStartForegroundSignature = new Class [] {
55
69
int .class , Notification .class };
56
70
private static final Class <?>[] mStopForegroundSignature = new Class [] {
57
71
boolean .class };
58
-
72
+
59
73
private NotificationManager mNM ;
60
74
private Method mSetForeground ;
61
75
private Method mStartForeground ;
62
76
private Method mStopForeground ;
63
77
private Object [] mSetForegroundArgs = new Object [1 ];
64
78
private Object [] mStartForegroundArgs = new Object [2 ];
65
79
private Object [] mStopForegroundArgs = new Object [1 ];
66
-
80
+
67
81
void invokeMethod (Method method , Object [] args ) {
68
82
try {
69
83
method .invoke (this , args );
@@ -75,7 +89,7 @@ void invokeMethod(Method method, Object[] args) {
75
89
Log .w ("ApiDemos" , "Unable to invoke method" , e );
76
90
}
77
91
}
78
-
92
+
79
93
/**
80
94
* This is a wrapper around the new startForeground method, using the older
81
95
* APIs if it is not available.
@@ -88,13 +102,13 @@ void startForegroundCompat(int id, Notification notification) {
88
102
invokeMethod (mStartForeground , mStartForegroundArgs );
89
103
return ;
90
104
}
91
-
105
+
92
106
// Fall back on the old API.
93
107
mSetForegroundArgs [0 ] = Boolean .TRUE ;
94
108
invokeMethod (mSetForeground , mSetForegroundArgs );
95
109
mNM .notify (id , notification );
96
110
}
97
-
111
+
98
112
/**
99
113
* This is a wrapper around the new stopForeground method, using the older
100
114
* APIs if it is not available.
@@ -106,14 +120,14 @@ void stopForegroundCompat(int id) {
106
120
invokeMethod (mStopForeground , mStopForegroundArgs );
107
121
return ;
108
122
}
109
-
123
+
110
124
// Fall back on the old API. Note to cancel BEFORE changing the
111
125
// foreground state, since we could be killed at that point.
112
126
mNM .cancel (id );
113
127
mSetForegroundArgs [0 ] = Boolean .FALSE ;
114
128
invokeMethod (mSetForeground , mSetForegroundArgs );
115
129
}
116
-
130
+
117
131
@ Override
118
132
public void onCreate () {
119
133
mNM = (NotificationManager )getSystemService (NOTIFICATION_SERVICE );
@@ -144,57 +158,65 @@ public void onDestroy() {
144
158
145
159
146
160
147
- // This is the old onStart method that will be called on the pre-2.0
148
- // platform. On 2.0 or later we override onStartCommand() so this
149
- // method will not be called.
150
- @ Override
151
- public void onStart (Intent intent , int startId ) {
152
- handleCommand (intent );
153
- }
154
-
161
+ @ SuppressLint ("InvalidWakeLockTag" )
155
162
@ Override
156
163
public int onStartCommand (Intent intent , int flags , int startId ) {
157
- handleCommand (intent );
158
- // We want this service to continue running until it is explicitly
159
- // stopped, so return sticky.
160
- return START_STICKY ;
161
- }
162
-
163
-
164
- void handleCommand (Intent intent ) {
165
- if (ACTION_FOREGROUND .equals (intent .getAction ())) {
164
+ if (ACTION_FOREGROUND .equals (intent .getAction ())
165
+ || ACTION_FOREGROUND_WAKELOCK .equals (intent .getAction ())) {
166
166
// In this sample, we'll use the same text for the ticker and the expanded notification
167
167
CharSequence text = getText (R .string .foreground_service_started );
168
-
169
- // Set the icon, scrolling text and timestamp
170
- Notification notification = new Notification (R .drawable .stat_sample , text ,
171
- System .currentTimeMillis ());
172
-
173
- // The PendingIntent to launch our activity if the user selects this notification
174
168
PendingIntent contentIntent = PendingIntent .getActivity (this , 0 ,
175
169
new Intent (this , Controller .class ), 0 );
176
-
177
170
// Set the info for the views that show in the notification panel.
178
- notification .setLatestEventInfo (this , getText (R .string .local_service_label ),
179
- text , contentIntent );
180
-
171
+ Notification notification = new Notification .Builder (this )
172
+ .setSmallIcon (R .drawable .stat_sample ) // the status icon
173
+ .setTicker (text ) // the status text
174
+ .setWhen (System .currentTimeMillis ()) // the time stamp
175
+ .setContentTitle (getText (R .string .alarm_service_label )) // the label
176
+ .setContentText (text ) // the contents of the entry
177
+ .setContentIntent (contentIntent ) // The intent to send when clicked
178
+ .build ();
181
179
startForegroundCompat (R .string .foreground_service_started , notification );
182
-
183
- } else if ( ACTION_BACKGROUND .equals (intent .getAction ())) {
180
+ } else if ( ACTION_BACKGROUND . equals ( intent . getAction ())
181
+ || ACTION_BACKGROUND_WAKELOCK .equals (intent .getAction ())) {
184
182
stopForegroundCompat (R .string .foreground_service_started );
185
183
}
184
+ if (ACTION_FOREGROUND_WAKELOCK .equals (intent .getAction ())
185
+ || ACTION_BACKGROUND_WAKELOCK .equals (intent .getAction ())) {
186
+ if (mWakeLock == null ) {
187
+ if (Build .VERSION .SDK_INT >= Build .VERSION_CODES .M ) {
188
+ mWakeLock = getSystemService (PowerManager .class ).newWakeLock (
189
+ PowerManager .PARTIAL_WAKE_LOCK , "wake-service" );
190
+ mWakeLock .acquire ();
191
+ }
192
+ } else {
193
+ releaseWakeLock ();
194
+ }
195
+ }
196
+ mHandler .removeCallbacks (mPulser );
197
+ mPulser .run ();
198
+ // We want this service to continue running until it is explicitly
199
+ // stopped, so return sticky.
200
+ return START_STICKY ;
186
201
}
187
-
202
+
203
+ void releaseWakeLock () {
204
+ if (mWakeLock != null ) {
205
+ mWakeLock .release ();
206
+ mWakeLock = null ;
207
+ }
208
+ }
209
+
188
210
@ Override
189
211
public IBinder onBind (Intent intent ) {
190
212
return null ;
191
213
}
192
-
214
+
193
215
// ----------------------------------------------------------------------
194
216
195
217
/**
196
218
* <p>Example of explicitly starting and stopping the {@link ForegroundService}.
197
- *
219
+ *
198
220
* <p>Note that this is implemented as an inner class only keep the sample
199
221
* all together; typically this code would appear in some separate class.
200
222
*/
0 commit comments