org.robolectric.shadows.ShadowPausedSystemClock Maven / Gradle / Ivy
Show all versions of shadows-framework Show documentation
package org.robolectric.shadows;
import static android.os.Build.VERSION_CODES.JELLY_BEAN_MR1;
import static android.os.Build.VERSION_CODES.P;
import android.os.SystemClock;
import java.time.DateTimeException;
import java.util.List;
import java.util.concurrent.CopyOnWriteArrayList;
import org.robolectric.annotation.HiddenApi;
import org.robolectric.annotation.Implementation;
import org.robolectric.annotation.Implements;
import org.robolectric.annotation.Resetter;
/**
* A shadow SystemClock used when {@link LooperMode.Mode.PAUSED} is active.
*
* In this variant, there is just one global system time controlled by this class. The current
* time is fixed in place, and manually advanced by calling {@link
* SystemClock#setCurrentTimeMillis(long)}
*
*
{@link SystemClock#uptimeMillis()} and {@link SystemClock#currentThreadTimeMillis()} are
* identical.
*
*
This class should not be referenced directly. Use ShadowSystemClock instead.
*/
@Implements(
value = SystemClock.class,
isInAndroidSdk = false,
shadowPicker = ShadowSystemClock.Picker.class)
public class ShadowPausedSystemClock extends ShadowSystemClock {
private static final long INITIAL_TIME = 100;
private static final int MILLIS_PER_NANO = 1000000;;
private static long currentTimeMillis = INITIAL_TIME;
private static List listeners = new CopyOnWriteArrayList<>();
/**
* Callback for clock updates
*/
interface Listener {
void clockUpdated(long newCurrentTimeMillis);
}
static void addListener(Listener listener) {
listeners.add(listener);
}
static void removeListener(Listener listener) {
listeners.remove(listener);
}
/** Advances the current time by given millis, without sleeping the current thread/ */
@Implementation
protected static void sleep(long millis) {
currentTimeMillis += millis;
}
/**
* Sets the current wall time.
*
* Currently does not perform any permission checks.
*
* @return false if specified time is less than current time.
*/
@Implementation
protected static boolean setCurrentTimeMillis(long millis) {
if (currentTimeMillis > millis) {
return false;
}
currentTimeMillis = millis;
for (Listener listener : listeners) {
listener.clockUpdated(currentTimeMillis);
}
return true;
}
@Implementation
protected static long uptimeMillis() {
return currentTimeMillis;
}
@Implementation
protected static long elapsedRealtime() {
return uptimeMillis();
}
@Implementation(minSdk = JELLY_BEAN_MR1)
protected static long elapsedRealtimeNanos() {
return elapsedRealtime() * MILLIS_PER_NANO;
}
@Implementation
protected static long currentThreadTimeMillis() {
return uptimeMillis();
}
@HiddenApi
@Implementation
protected static long currentThreadTimeMicro() {
return uptimeMillis() * 1000;
}
@HiddenApi
@Implementation
protected static long currentTimeMicro() {
return currentThreadTimeMicro();
}
@Implementation(minSdk = P)
@HiddenApi
protected static long currentNetworkTimeMillis() {
if (networkTimeAvailable) {
return currentTimeMillis;
} else {
throw new DateTimeException("Network time not available");
}
}
@Resetter
public static void reset() {
currentTimeMillis = INITIAL_TIME;
ShadowSystemClock.reset();
}
}