-
-
Save leopic/baccf21403a8fa8fdb6b36709f5bfc29 to your computer and use it in GitHub Desktop.
Singleton Util that share the status of the App when it goes to background or foreground
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
import android.app.Activity; | |
import android.app.Application; | |
import android.content.Context; | |
import android.os.Bundle; | |
import android.os.Handler; | |
import android.util.Log; | |
import java.util.List; | |
import java.util.concurrent.CopyOnWriteArrayList; | |
/** | |
* Usage: | |
* <p/> | |
* 1. Get the Foreground Singleton, passing a Context or Application object unless you | |
* are sure that the Singleton has definitely already been initialised elsewhere. | |
* <p/> | |
* 2.a) Perform a direct, synchronous check: Foreground.isForeground() / .isBackground() | |
* <p/> | |
* or | |
* <p/> | |
* 2.b) Register to be notified (useful in Service or other non-UI components): | |
* <p/> | |
* Foreground.Listener myListener = new Foreground.Listener(){ | |
* public void onBecameForeground(){ | |
* // ... whatever you want to do | |
* } | |
* public void onBecameBackground(){ | |
* // ... whatever you want to do | |
* } | |
* } | |
* <p/> | |
* public void onCreate(){ | |
* super.onCreate(); | |
* Foreground.get(this).addListener(listener); | |
* } | |
* <p/> | |
* public void onDestroy(){ | |
* super.onCreate(); | |
* Foreground.get(this).removeListener(listener); | |
* } | |
*/ | |
public class ForegroundUtils implements Application.ActivityLifecycleCallbacks { | |
public static final long CHECK_DELAY = 500; | |
public static final String TAG = ForegroundUtils.class.getName(); | |
public interface Listener { | |
public void onBecameForeground(); | |
public void onBecameBackground(); | |
} | |
private static ForegroundUtils instance; | |
private boolean foreground = false, paused = true; | |
private Handler handler = new Handler(); | |
private List<Listener> listeners = new CopyOnWriteArrayList<Listener>(); | |
private Runnable check; | |
/** | |
* Its not strictly necessary to use this method - _usually_ invoking | |
* get with a Context gives us a path to retrieve the Application and | |
* initialise, but sometimes (e.g. in test harness) the ApplicationContext | |
* is != the Application, and the docs make no guarantees. | |
* | |
* @param application | |
* @return an initialised Foreground instance | |
*/ | |
public static ForegroundUtils init(Application application) { | |
if (instance == null) { | |
instance = new ForegroundUtils(); | |
application.registerActivityLifecycleCallbacks(instance); | |
} | |
return instance; | |
} | |
public static ForegroundUtils get(Application application) { | |
if (instance == null) { | |
init(application); | |
} | |
return instance; | |
} | |
public static ForegroundUtils get(Context ctx) { | |
if (instance == null) { | |
Context appCtx = ctx.getApplicationContext(); | |
if (appCtx instanceof Application) { | |
init((Application) appCtx); | |
} | |
throw new IllegalStateException( | |
"Foreground is not initialised and " + | |
"cannot obtain the Application object"); | |
} | |
return instance; | |
} | |
public static ForegroundUtils get() { | |
if (instance == null) { | |
throw new IllegalStateException( | |
"Foreground is not initialised - invoke " + | |
"at least once with parameterised init/get"); | |
} | |
return instance; | |
} | |
public boolean isForeground() { | |
return foreground; | |
} | |
public boolean isBackground() { | |
return !foreground; | |
} | |
public void addListener(Listener listener) { | |
listeners.add(listener); | |
} | |
public void removeListener(Listener listener) { | |
listeners.remove(listener); | |
} | |
@Override | |
public void onActivityResumed(Activity activity) { | |
paused = false; | |
boolean wasBackground = !foreground; | |
foreground = true; | |
if (check != null) | |
handler.removeCallbacks(check); | |
if (wasBackground) { | |
Log.i(TAG, "went foreground"); | |
for (Listener l : listeners) { | |
try { | |
l.onBecameForeground(); | |
} catch (Exception exc) { | |
Log.e(TAG, "Listener threw exception!", exc); | |
} | |
} | |
} else { | |
Log.i(TAG, "still foreground"); | |
} | |
} | |
@Override | |
public void onActivityPaused(Activity activity) { | |
paused = true; | |
if (check != null) | |
handler.removeCallbacks(check); | |
handler.postDelayed(check = new Runnable() { | |
@Override | |
public void run() { | |
if (foreground && paused) { | |
foreground = false; | |
Log.i(TAG, "went background"); | |
for (Listener l : listeners) { | |
try { | |
l.onBecameBackground(); | |
} catch (Exception exc) { | |
Log.e(TAG, "Listener threw exception!", exc); | |
} | |
} | |
} else { | |
Log.i(TAG, "still foreground"); | |
} | |
} | |
}, CHECK_DELAY); | |
} | |
@Override | |
public void onActivityCreated(Activity activity, Bundle savedInstanceState) { | |
} | |
@Override | |
public void onActivityStarted(Activity activity) { | |
} | |
@Override | |
public void onActivityStopped(Activity activity) { | |
} | |
@Override | |
public void onActivitySaveInstanceState(Activity activity, Bundle outState) { | |
} | |
@Override | |
public void onActivityDestroyed(Activity activity) { | |
} | |
} |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment