/*
|
* Copyright (C) 2017 The Android Open Source Project
|
*
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
* you may not use this file except in compliance with the License.
|
* You may obtain a copy of the License at
|
*
|
* http://www.apache.org/licenses/LICENSE-2.0
|
*
|
* Unless required by applicable law or agreed to in writing, software
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
* See the License for the specific language governing permissions and
|
* limitations under the License.
|
*/
|
package com.android.launcher3.states;
|
|
import android.content.Intent;
|
import android.os.Binder;
|
import android.os.Bundle;
|
import android.os.IBinder;
|
|
import com.android.launcher3.Launcher;
|
import com.android.launcher3.LauncherAppState;
|
import com.android.launcher3.LauncherModel.Callbacks;
|
import com.android.launcher3.MainThreadExecutor;
|
|
import java.lang.ref.WeakReference;
|
|
/**
|
* Utility class to sending state handling logic to Launcher from within the same process.
|
*
|
* Extending {@link Binder} ensures that the platform maintains a single instance of each object
|
* which allows this object to safely navigate the system process.
|
*/
|
public abstract class InternalStateHandler extends Binder {
|
|
public static final String EXTRA_STATE_HANDLER = "launcher.state_handler";
|
|
private static final Scheduler sScheduler = new Scheduler();
|
|
/**
|
* Initializes the handler when the launcher is ready.
|
* @return true if the handler wants to stay alive.
|
*/
|
protected abstract boolean init(Launcher launcher, boolean alreadyOnHome);
|
|
public final Intent addToIntent(Intent intent) {
|
Bundle extras = new Bundle();
|
extras.putBinder(EXTRA_STATE_HANDLER, this);
|
intent.putExtras(extras);
|
return intent;
|
}
|
|
public final void initWhenReady() {
|
sScheduler.schedule(this);
|
}
|
|
public boolean clearReference() {
|
return sScheduler.clearReference(this);
|
}
|
|
public static boolean hasPending() {
|
return sScheduler.hasPending();
|
}
|
|
public static boolean handleCreate(Launcher launcher, Intent intent) {
|
return handleIntent(launcher, intent, false, false);
|
}
|
|
public static boolean handleNewIntent(Launcher launcher, Intent intent, boolean alreadyOnHome) {
|
return handleIntent(launcher, intent, alreadyOnHome, true);
|
}
|
|
private static boolean handleIntent(
|
Launcher launcher, Intent intent, boolean alreadyOnHome, boolean explicitIntent) {
|
boolean result = false;
|
if (intent != null && intent.getExtras() != null) {
|
IBinder stateBinder = intent.getExtras().getBinder(EXTRA_STATE_HANDLER);
|
if (stateBinder instanceof InternalStateHandler) {
|
InternalStateHandler handler = (InternalStateHandler) stateBinder;
|
if (!handler.init(launcher, alreadyOnHome)) {
|
intent.getExtras().remove(EXTRA_STATE_HANDLER);
|
}
|
result = true;
|
}
|
}
|
if (!result && !explicitIntent) {
|
result = sScheduler.initIfPending(launcher, alreadyOnHome);
|
}
|
return result;
|
}
|
|
private static class Scheduler implements Runnable {
|
|
private WeakReference<InternalStateHandler> mPendingHandler = new WeakReference<>(null);
|
private MainThreadExecutor mMainThreadExecutor;
|
|
public void schedule(InternalStateHandler handler) {
|
synchronized (this) {
|
mPendingHandler = new WeakReference<>(handler);
|
if (mMainThreadExecutor == null) {
|
mMainThreadExecutor = new MainThreadExecutor();
|
}
|
}
|
mMainThreadExecutor.execute(this);
|
}
|
|
@Override
|
public void run() {
|
LauncherAppState app = LauncherAppState.getInstanceNoCreate();
|
if (app == null) {
|
return;
|
}
|
Callbacks cb = app.getModel().getCallback();
|
if (!(cb instanceof Launcher)) {
|
return;
|
}
|
Launcher launcher = (Launcher) cb;
|
initIfPending(launcher, launcher.isStarted());
|
}
|
|
public boolean initIfPending(Launcher launcher, boolean alreadyOnHome) {
|
InternalStateHandler pendingHandler = mPendingHandler.get();
|
if (pendingHandler != null) {
|
if (!pendingHandler.init(launcher, alreadyOnHome)) {
|
clearReference(pendingHandler);
|
}
|
return true;
|
}
|
return false;
|
}
|
|
public boolean clearReference(InternalStateHandler handler) {
|
synchronized (this) {
|
if (mPendingHandler.get() == handler) {
|
mPendingHandler.clear();
|
return true;
|
}
|
return false;
|
}
|
}
|
|
public boolean hasPending() {
|
return mPendingHandler.get() != null;
|
}
|
}
|
}
|