mirror of
https://github.com/mozilla/gecko-dev.git
synced 2024-11-08 04:27:37 +00:00
b37f6afd24
--HG-- rename : mobile/android/base/util/UiAsyncTask.java => mobile/android/base/util/UIAsyncTask.java
122 lines
3.9 KiB
Java
122 lines
3.9 KiB
Java
/* -*- Mode: Java; c-basic-offset: 4; tab-width: 4; indent-tabs-mode: nil; -*-
|
|
* This Source Code Form is subject to the terms of the Mozilla Public
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
|
|
|
package org.mozilla.gecko.util;
|
|
|
|
import android.os.Handler;
|
|
import android.os.Looper;
|
|
|
|
/**
|
|
* Executes a background task and publishes the result on the UI thread.
|
|
*
|
|
* The standard {@link android.os.AsyncTask} only runs onPostExecute on the
|
|
* thread it is constructed on, so this is a convenience class for creating
|
|
* tasks off the UI thread.
|
|
*
|
|
* We use generics differently to Android's AsyncTask.
|
|
* Android uses a "Params" type parameter to represent the type of all the parameters to this task.
|
|
* It then uses arguments of type Params... to permit arbitrarily-many of these to be passed
|
|
* fluently.
|
|
*
|
|
* Unfortunately, since Java does not support generic array types (and since varargs desugars to a
|
|
* single array parameter) that behaviour exposes a hole in the type system. See:
|
|
* http://docs.oracle.com/javase/tutorial/java/generics/nonReifiableVarargsType.html#vulnerabilities
|
|
*
|
|
* Instead, we equivalently have a single type parameter "Param". A UiAsyncTask may take exactly one
|
|
* parameter of type Param. Since Param can be an array type, this no more restrictive than the
|
|
* other approach, it just provides additional type safety.
|
|
*/
|
|
public abstract class UIAsyncTask<Param, Result> {
|
|
/**
|
|
* Provide a convenient API for parameter-free UiAsyncTasks by wrapping parameter-taking methods
|
|
* from UiAsyncTask in parameterless equivalents.
|
|
*/
|
|
public static abstract class WithoutParams<InnerResult> extends UIAsyncTask<Void, InnerResult> {
|
|
public WithoutParams(Handler backgroundThreadHandler) {
|
|
super(backgroundThreadHandler);
|
|
}
|
|
|
|
public void execute() {
|
|
execute(null);
|
|
}
|
|
|
|
@Override
|
|
protected InnerResult doInBackground(Void unused) {
|
|
return doInBackground();
|
|
}
|
|
|
|
protected abstract InnerResult doInBackground();
|
|
}
|
|
|
|
/* inner-access */ final Handler mBackgroundThreadHandler;
|
|
private volatile boolean mCancelled;
|
|
private static Handler sHandler;
|
|
|
|
/**
|
|
* Creates a new asynchronous task.
|
|
*
|
|
* @param backgroundThreadHandler the handler to execute the background task on
|
|
*/
|
|
public UIAsyncTask(Handler backgroundThreadHandler) {
|
|
mBackgroundThreadHandler = backgroundThreadHandler;
|
|
}
|
|
|
|
private static synchronized Handler getUiHandler() {
|
|
if (sHandler == null) {
|
|
sHandler = new Handler(Looper.getMainLooper());
|
|
}
|
|
|
|
return sHandler;
|
|
}
|
|
|
|
private final class BackgroundTaskRunnable implements Runnable {
|
|
private Param mParam;
|
|
|
|
public BackgroundTaskRunnable(Param param) {
|
|
mParam = param;
|
|
}
|
|
|
|
@Override
|
|
public void run() {
|
|
final Result result = doInBackground(mParam);
|
|
|
|
getUiHandler().post(new Runnable() {
|
|
@Override
|
|
public void run() {
|
|
if (mCancelled) {
|
|
onCancelled();
|
|
} else {
|
|
onPostExecute(result);
|
|
}
|
|
}
|
|
});
|
|
}
|
|
}
|
|
|
|
protected void execute(final Param param) {
|
|
getUiHandler().post(new Runnable() {
|
|
@Override
|
|
public void run() {
|
|
onPreExecute();
|
|
mBackgroundThreadHandler.post(new BackgroundTaskRunnable(param));
|
|
}
|
|
});
|
|
}
|
|
|
|
public final boolean cancel() {
|
|
mCancelled = true;
|
|
return mCancelled;
|
|
}
|
|
|
|
public final boolean isCancelled() {
|
|
return mCancelled;
|
|
}
|
|
|
|
protected void onPreExecute() { }
|
|
protected void onPostExecute(Result result) { }
|
|
protected void onCancelled() { }
|
|
protected abstract Result doInBackground(Param param);
|
|
}
|