AppClone: Implement clone backend flow
- Add onClick listeners of add/trash icons on Cloned Apps page - New layout with ImageView(Add icon) and ProgressBar - Creation of clone user and install package in clone user - Uninstallation of cloned app - Summary when app is being cloned and after clone completion - Action metrics Bug: 259022623 Test: make RunSettingsRoboTests -j64 Change-Id: Idc76fb8d88ba8987084beef2a0ce4c57d6c45b9e
This commit is contained in:
@@ -16,28 +16,37 @@
|
||||
|
||||
package com.android.settings.applications.manageapplications;
|
||||
|
||||
import static com.android.settings.applications.manageapplications.ManageApplications.ApplicationsAdapter;
|
||||
import static com.android.settings.applications.manageapplications.ManageApplications.LIST_TYPE_CLONED_APPS;
|
||||
import static com.android.settings.applications.manageapplications.ManageApplications.LIST_TYPE_NONE;
|
||||
|
||||
import android.app.settings.SettingsEnums;
|
||||
import android.content.Context;
|
||||
import android.content.pm.ApplicationInfo;
|
||||
import android.content.pm.PackageManager;
|
||||
import android.graphics.drawable.Drawable;
|
||||
import android.os.AsyncTask;
|
||||
import android.text.TextUtils;
|
||||
import android.util.Log;
|
||||
import android.view.LayoutInflater;
|
||||
import android.view.View;
|
||||
import android.view.ViewGroup;
|
||||
import android.widget.ImageView;
|
||||
import android.widget.ProgressBar;
|
||||
import android.widget.Switch;
|
||||
import android.widget.TextView;
|
||||
|
||||
import androidx.annotation.StringRes;
|
||||
import androidx.annotation.VisibleForTesting;
|
||||
import androidx.fragment.app.FragmentActivity;
|
||||
import androidx.recyclerview.widget.RecyclerView;
|
||||
|
||||
import com.android.settings.R;
|
||||
import com.android.settings.overlay.FeatureFactory;
|
||||
import com.android.settingslib.applications.ApplicationsState;
|
||||
import com.android.settingslib.applications.ApplicationsState.AppEntry;
|
||||
import com.android.settingslib.core.instrumentation.MetricsFeatureProvider;
|
||||
|
||||
|
||||
public class ApplicationViewHolder extends RecyclerView.ViewHolder {
|
||||
|
||||
@@ -51,8 +60,8 @@ public class ApplicationViewHolder extends RecyclerView.ViewHolder {
|
||||
final ViewGroup mWidgetContainer;
|
||||
@VisibleForTesting
|
||||
final Switch mSwitch;
|
||||
|
||||
private static int sListType;
|
||||
final ImageView mAddIcon;
|
||||
final ProgressBar mProgressBar;
|
||||
|
||||
private final ImageView mAppIcon;
|
||||
|
||||
@@ -64,33 +73,23 @@ public class ApplicationViewHolder extends RecyclerView.ViewHolder {
|
||||
mDisabled = itemView.findViewById(R.id.appendix);
|
||||
mSwitch = itemView.findViewById(R.id.switchWidget);
|
||||
mWidgetContainer = itemView.findViewById(android.R.id.widget_frame);
|
||||
mAddIcon = itemView.findViewById(R.id.add_preference_widget);
|
||||
mProgressBar = itemView.findViewById(R.id.progressBar_cyclic);
|
||||
}
|
||||
|
||||
static View newView(ViewGroup parent) {
|
||||
return newView(parent, false /* twoTarget */);
|
||||
return newView(parent, false /* twoTarget */, LIST_TYPE_NONE /* listType */);
|
||||
}
|
||||
|
||||
static View newView(ViewGroup parent , boolean twoTarget, int listType, Context context) {
|
||||
sListType = listType;
|
||||
return newView(parent, twoTarget);
|
||||
}
|
||||
|
||||
static View newView(ViewGroup parent, boolean twoTarget) {
|
||||
static View newView(ViewGroup parent, boolean twoTarget, int listType) {
|
||||
ViewGroup view = (ViewGroup) LayoutInflater.from(parent.getContext())
|
||||
.inflate(R.layout.preference_app, parent, false);
|
||||
final ViewGroup widgetFrame = view.findViewById(android.R.id.widget_frame);
|
||||
ViewGroup widgetFrame = view.findViewById(android.R.id.widget_frame);
|
||||
if (twoTarget) {
|
||||
if (widgetFrame != null) {
|
||||
if (sListType == LIST_TYPE_CLONED_APPS) {
|
||||
if (listType == LIST_TYPE_CLONED_APPS) {
|
||||
LayoutInflater.from(parent.getContext())
|
||||
.inflate(R.layout.preference_widget_add, widgetFrame, true);
|
||||
//todo(b/259022623): Invoke the clone backend flow i.e.
|
||||
// i) upon onclick of add icon, create new clone profile the first time
|
||||
// and clone an app.
|
||||
// ii) Show progress bar while app is being cloned
|
||||
// iii) And upon onClick of trash icon, delete the cloned app instance
|
||||
// from clone profile.
|
||||
// iv) Log metrics
|
||||
.inflate(R.layout.preference_widget_add_progressbar, widgetFrame, true);
|
||||
} else {
|
||||
LayoutInflater.from(parent.getContext())
|
||||
.inflate(R.layout.preference_widget_primary_switch, widgetFrame, true);
|
||||
@@ -202,4 +201,72 @@ public class ApplicationViewHolder extends RecyclerView.ViewHolder {
|
||||
mSwitch.setEnabled(enabled);
|
||||
}
|
||||
}
|
||||
|
||||
void updateAppCloneWidget(Context context, View.OnClickListener onClickListener,
|
||||
AppEntry entry) {
|
||||
if (mAddIcon != null) {
|
||||
if (!entry.isCloned) {
|
||||
mAddIcon.setBackground(context.getDrawable(R.drawable.ic_add_24dp));
|
||||
} else {
|
||||
mAddIcon.setBackground(context.getDrawable(R.drawable.ic_trash_can));
|
||||
setSummary(R.string.cloned_app_created_summary);
|
||||
}
|
||||
mAddIcon.setOnClickListener(onClickListener);
|
||||
}
|
||||
}
|
||||
|
||||
View.OnClickListener appCloneOnClickListener(AppEntry entry,
|
||||
ApplicationsAdapter adapter, FragmentActivity manageApplicationsActivity) {
|
||||
Context context = manageApplicationsActivity.getApplicationContext();
|
||||
return new View.OnClickListener() {
|
||||
@Override
|
||||
public void onClick(View v) {
|
||||
CloneBackend cloneBackend = CloneBackend.getInstance(context);
|
||||
final MetricsFeatureProvider metricsFeatureProvider =
|
||||
FeatureFactory.getFactory(context).getMetricsFeatureProvider();
|
||||
|
||||
String packageName = entry.info.packageName;
|
||||
|
||||
if (mWidgetContainer != null) {
|
||||
if (!entry.isCloned) {
|
||||
metricsFeatureProvider.action(context,
|
||||
SettingsEnums.ACTION_CREATE_CLONE_APP);
|
||||
mAddIcon.setVisibility(View.INVISIBLE);
|
||||
mProgressBar.setVisibility(View.VISIBLE);
|
||||
setSummary(R.string.cloned_app_creation_summary);
|
||||
|
||||
// todo(b/262352524): To figure out a way to prevent memory leak
|
||||
// without making this static.
|
||||
new AsyncTask<Void, Void, Integer>(){
|
||||
|
||||
@Override
|
||||
protected Integer doInBackground(Void... unused) {
|
||||
return cloneBackend.installCloneApp(packageName);
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onPostExecute(Integer res) {
|
||||
mProgressBar.setVisibility(View.INVISIBLE);
|
||||
mAddIcon.setVisibility(View.VISIBLE);
|
||||
|
||||
if (res != CloneBackend.SUCCESS) {
|
||||
setSummary(null);
|
||||
return;
|
||||
}
|
||||
|
||||
// Refresh the page to reflect newly created cloned app.
|
||||
adapter.rebuild();
|
||||
}
|
||||
}.execute();
|
||||
|
||||
} else if (entry.isCloned) {
|
||||
metricsFeatureProvider.action(context,
|
||||
SettingsEnums.ACTION_DELETE_CLONE_APP);
|
||||
cloneBackend.uninstallClonedApp(packageName, /*allUsers*/ false,
|
||||
manageApplicationsActivity);
|
||||
}
|
||||
}
|
||||
}
|
||||
};
|
||||
}
|
||||
}
|
||||
|
Reference in New Issue
Block a user