blob: e07ba19af1f7be55536974295113171138a5a376 [file] [log] [blame]
/*
* Copyright (C) 2022 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.wallpaper.module;
import static android.app.WallpaperManager.FLAG_LOCK;
import android.app.Activity;
import android.app.ProgressDialog;
import android.app.WallpaperColors;
import android.app.WallpaperManager;
import android.content.Context;
import android.content.pm.ActivityInfo;
import android.graphics.Point;
import android.graphics.Rect;
import android.os.Build.VERSION;
import android.os.Build.VERSION_CODES;
import android.util.Log;
import android.view.Display;
import androidx.annotation.NonNull;
import androidx.annotation.Nullable;
import androidx.annotation.StringRes;
import androidx.fragment.app.FragmentManager;
import androidx.lifecycle.Lifecycle.Event;
import androidx.lifecycle.LifecycleEventObserver;
import androidx.lifecycle.LifecycleOwner;
import com.android.wallpaper.R;
import com.android.wallpaper.asset.Asset;
import com.android.wallpaper.model.LiveWallpaperInfo;
import com.android.wallpaper.model.WallpaperInfo;
import com.android.wallpaper.module.UserEventLogger.WallpaperSetFailureReason;
import com.android.wallpaper.module.WallpaperPersister.Destination;
import com.android.wallpaper.module.WallpaperPersister.SetWallpaperCallback;
import com.android.wallpaper.picker.SetWallpaperDialogFragment;
import com.android.wallpaper.picker.SetWallpaperDialogFragment.Listener;
import com.android.wallpaper.util.ScreenSizeCalculator;
import com.android.wallpaper.util.ThrowableAnalyzer;
import com.android.wallpaper.util.WallpaperCropUtils;
import com.bumptech.glide.Glide;
import java.io.IOException;
import java.util.Optional;
/**
* Helper class used to set the current wallpaper. It handles showing the destination request dialog
* and actually setting the wallpaper on a given destination.
* It is expected to be instantiated within a Fragment or Activity, and {@link #cleanUp()} should
* be called from its owner's onDestroy method (or equivalent).
* TODO b/256819974 This is a duplicate of WallpaperSetter and should be removed after
* WallpaperSetter in the branch of tm-qpr-dev is synced with master.
*/
public class WallpaperSetter2 {
private static final String TAG = "WallpaperSetter2";
private static final String PROGRESS_DIALOG_NO_TITLE = null;
private static final boolean PROGRESS_DIALOG_INDETERMINATE = true;
private static final int UNUSED_REQUEST_CODE = 1;
private static final String TAG_SET_WALLPAPER_DIALOG_FRAGMENT = "set_wallpaper_dialog";
private final WallpaperPersister mWallpaperPersister;
private final WallpaperPreferences mPreferences;
private final boolean mTestingModeEnabled;
private final UserEventLogger mUserEventLogger;
private ProgressDialog mProgressDialog;
private Optional<Integer> mCurrentScreenOrientation = Optional.empty();
public WallpaperSetter2(WallpaperPersister wallpaperPersister,
WallpaperPreferences preferences, UserEventLogger userEventLogger,
boolean isTestingModeEnabled) {
mTestingModeEnabled = isTestingModeEnabled;
mWallpaperPersister = wallpaperPersister;
mPreferences = preferences;
mUserEventLogger = userEventLogger;
}
/**
* Sets current wallpaper to the device with the minimum scale to fit the screen size.
*
* @param containerActivity main Activity that owns the current fragment
* @param wallpaper info for the actual wallpaper to set
* @param destination the wallpaper destination i.e. home vs. lockscreen vs. both.
* @param callback optional callback to be notified when the wallpaper is set.
*/
public void setCurrentWallpaper(Activity containerActivity, WallpaperInfo wallpaper,
@Destination final int destination,
@Nullable SetWallpaperCallback callback) {
Asset wallpaperAsset = wallpaper.getAsset(containerActivity.getApplicationContext());
wallpaperAsset.decodeRawDimensions(containerActivity, dimensions -> {
if (dimensions == null) {
Log.e(TAG, "Raw wallpaper's dimensions are null");
return;
}
Display defaultDisplay = containerActivity.getWindowManager().getDefaultDisplay();
Point screenSize = ScreenSizeCalculator.getInstance().getScreenSize(defaultDisplay);
Rect visibleRawWallpaperRect =
WallpaperCropUtils.calculateVisibleRect(dimensions, screenSize);
float wallpaperScale = WallpaperCropUtils.calculateMinZoom(dimensions, screenSize);
Rect cropRect = WallpaperCropUtils.calculateCropRect(
containerActivity.getApplicationContext(), defaultDisplay,
dimensions, visibleRawWallpaperRect, wallpaperScale);
setCurrentWallpaper(containerActivity, wallpaper, wallpaperAsset, destination,
wallpaperScale, cropRect, null, callback);
});
}
/**
* Sets current wallpaper to the device based on current zoom and scroll state.
*
* @param containerActivity main Activity that owns the current fragment
* @param wallpaper info for the actual wallpaper to set
* @param wallpaperAsset Wallpaper asset from which to retrieve image data.
* @param destination The wallpaper destination i.e. home vs. lockscreen vs. both.
* @param wallpaperScale Scaling factor applied to the source image before setting the
* wallpaper to the device.
* @param cropRect Desired crop area of the wallpaper in post-scale units. If null,
* then the
* wallpaper image will be set without any scaling or cropping.
* @param callback optional callback to be notified when the wallpaper is set.
*/
public void setCurrentWallpaper(Activity containerActivity, WallpaperInfo wallpaper,
@Nullable Asset wallpaperAsset, @Destination final int destination,
float wallpaperScale, @Nullable Rect cropRect, WallpaperColors wallpaperColors,
@Nullable SetWallpaperCallback callback) {
if (wallpaper instanceof LiveWallpaperInfo) {
setCurrentLiveWallpaper(containerActivity, (LiveWallpaperInfo) wallpaper, destination,
wallpaperColors, callback);
return;
}
mPreferences.setPendingWallpaperSetStatus(
WallpaperPreferences.WALLPAPER_SET_PENDING);
// Save current screen rotation so we can temporarily disable rotation while setting the
// wallpaper and restore after setting the wallpaper finishes.
saveAndLockScreenOrientationIfNeeded(containerActivity);
// Clear MosaicView tiles and Glide's cache and pools to reclaim memory for final cropped
// bitmap.
Glide.get(containerActivity).clearMemory();
// ProgressDialog endlessly updates the UI thread, keeping it from going idle which
// therefore
// causes Espresso to hang once the dialog is shown.
if (!mTestingModeEnabled && !containerActivity.isFinishing()) {
int themeResId = (VERSION.SDK_INT < VERSION_CODES.LOLLIPOP)
? R.style.ProgressDialogThemePreL : R.style.LightDialogTheme;
mProgressDialog = new ProgressDialog(containerActivity, themeResId);
mProgressDialog.setTitle(PROGRESS_DIALOG_NO_TITLE);
mProgressDialog.setMessage(containerActivity.getString(
R.string.set_wallpaper_progress_message));
mProgressDialog.setIndeterminate(PROGRESS_DIALOG_INDETERMINATE);
if (containerActivity instanceof LifecycleOwner) {
((LifecycleOwner) containerActivity).getLifecycle().addObserver(
new LifecycleEventObserver() {
@Override
public void onStateChanged(@NonNull LifecycleOwner source,
@NonNull Event event) {
if (event == Event.ON_DESTROY) {
if (mProgressDialog != null) {
mProgressDialog.dismiss();
mProgressDialog = null;
}
}
}
});
}
mProgressDialog.show();
}
mWallpaperPersister.setIndividualWallpaper(
wallpaper, wallpaperAsset, cropRect,
wallpaperScale, destination, new SetWallpaperCallback() {
@Override
public void onSuccess(WallpaperInfo wallpaperInfo) {
onWallpaperApplied(wallpaper, containerActivity);
if (callback != null) {
callback.onSuccess(wallpaper);
}
}
@Override
public void onError(Throwable throwable) {
onWallpaperApplyError(throwable, containerActivity);
if (callback != null) {
callback.onError(throwable);
}
}
});
}
private void setCurrentLiveWallpaper(Activity activity, LiveWallpaperInfo wallpaper,
@Destination final int destination, WallpaperColors colors,
@Nullable SetWallpaperCallback callback) {
try {
// Save current screen rotation so we can temporarily disable rotation while setting the
// wallpaper and restore after setting the wallpaper finishes.
saveAndLockScreenOrientationIfNeeded(activity);
if (destination == WallpaperPersister.DEST_LOCK_SCREEN) {
throw new IllegalArgumentException(
"Live wallpaper cannot be applied on lock screen only");
}
WallpaperManager wallpaperManager = WallpaperManager.getInstance(activity);
wallpaperManager.setWallpaperComponentWithFlags(
wallpaper.getWallpaperComponent().getComponent(),
WallpaperPersister.destinationToFlags(destination));
wallpaperManager.setWallpaperOffsetSteps(0.5f /* xStep */, 0.0f /* yStep */);
wallpaperManager.setWallpaperOffsets(
activity.getWindow().getDecorView().getRootView().getWindowToken(),
0.5f /* xOffset */, 0.0f /* yOffset */);
if (destination == WallpaperPersister.DEST_BOTH) {
wallpaperManager.clear(FLAG_LOCK);
}
mPreferences.storeLatestHomeWallpaper(wallpaper.getWallpaperId(), wallpaper, colors);
onWallpaperApplied(wallpaper, activity);
if (callback != null) {
callback.onSuccess(wallpaper);
}
} catch (RuntimeException | IOException e) {
onWallpaperApplyError(e, activity);
if (callback != null) {
callback.onError(e);
}
}
}
/**
* Sets current live wallpaper to the device (restore case)
*
* @param context The context for initiating wallpaper manager
* @param wallpaper Information for the actual wallpaper to set
* @param destination The wallpaper destination i.e. home vs. lockscreen vs. both
* @param colors The {@link WallpaperColors} for placeholder of quickswitching
* @param callback Optional callback to be notified when the wallpaper is set.
*/
public void setCurrentLiveWallpaper(Context context, LiveWallpaperInfo wallpaper,
@Destination final int destination, @Nullable WallpaperColors colors,
@Nullable SetWallpaperCallback callback) {
try {
if (destination == WallpaperPersister.DEST_LOCK_SCREEN) {
throw new IllegalArgumentException(
"Live wallpaper cannot be applied on lock screen only");
}
WallpaperManager wallpaperManager = WallpaperManager.getInstance(context);
wallpaperManager.setWallpaperComponentWithFlags(
wallpaper.getWallpaperComponent().getComponent(),
WallpaperPersister.destinationToFlags(destination));
if (destination == WallpaperPersister.DEST_BOTH) {
wallpaperManager.clear(FLAG_LOCK);
}
mPreferences.storeLatestHomeWallpaper(wallpaper.getWallpaperId(), wallpaper,
colors != null ? colors :
WallpaperColors.fromBitmap(wallpaper.getThumbAsset(context)
.getLowResBitmap(context)));
// Not call onWallpaperApplied() as no UI is presented.
if (callback != null) {
callback.onSuccess(wallpaper);
}
} catch (RuntimeException | IOException e) {
// Not call onWallpaperApplyError() as no UI is presented.
if (callback != null) {
callback.onError(e);
}
}
}
private void onWallpaperApplied(WallpaperInfo wallpaper, Activity containerActivity) {
mUserEventLogger.logWallpaperSet(
wallpaper.getCollectionId(containerActivity),
wallpaper.getWallpaperId(), wallpaper.getEffectNames());
mPreferences.setPendingWallpaperSetStatus(
WallpaperPreferences.WALLPAPER_SET_NOT_PENDING);
mUserEventLogger.logWallpaperSetResult(
UserEventLogger.WALLPAPER_SET_RESULT_SUCCESS);
cleanUp();
restoreScreenOrientationIfNeeded(containerActivity);
}
private void onWallpaperApplyError(Throwable throwable, Activity containerActivity) {
mPreferences.setPendingWallpaperSetStatus(
WallpaperPreferences.WALLPAPER_SET_NOT_PENDING);
mUserEventLogger.logWallpaperSetResult(
UserEventLogger.WALLPAPER_SET_RESULT_FAILURE);
@WallpaperSetFailureReason int failureReason = ThrowableAnalyzer.isOOM(
throwable)
? UserEventLogger.WALLPAPER_SET_FAILURE_REASON_OOM
: UserEventLogger.WALLPAPER_SET_FAILURE_REASON_OTHER;
mUserEventLogger.logWallpaperSetFailureReason(failureReason);
cleanUp();
restoreScreenOrientationIfNeeded(containerActivity);
}
/**
* Call this method to clean up this instance's state.
*/
public void cleanUp() {
if (mProgressDialog != null) {
mProgressDialog.dismiss();
mProgressDialog = null;
}
}
/**
* Show a dialog asking the user for the Wallpaper's destination
* (eg, "Home screen", "Lock Screen")
*
* @param isLiveWallpaper whether the wallpaper that we want to set is a live wallpaper.
* @param listener {@link Listener} that will receive the response.
* @see Destination
*/
public void requestDestination(Activity activity, FragmentManager fragmentManager,
Listener listener, boolean isLiveWallpaper) {
requestDestination(activity, fragmentManager, R.string.set_wallpaper_dialog_message,
listener, isLiveWallpaper);
}
/**
* Show a dialog asking the user for the Wallpaper's destination
* (eg, "Home screen", "Lock Screen")
*
* @param isLiveWallpaper whether the wallpaper that we want to set is a live wallpaper.
* @param listener {@link Listener} that will receive the response.
* @param titleResId title for the dialog
* @see Destination
*/
public void requestDestination(Activity activity, FragmentManager fragmentManager,
@StringRes int titleResId, Listener listener, boolean isLiveWallpaper) {
saveAndLockScreenOrientationIfNeeded(activity);
Listener listenerWrapper = new Listener() {
@Override
public void onSet(int destination) {
if (listener != null) {
listener.onSet(destination);
}
}
@Override
public void onDialogDismissed(boolean withItemSelected) {
if (!withItemSelected) {
restoreScreenOrientationIfNeeded(activity);
}
if (listener != null) {
listener.onDialogDismissed(withItemSelected);
}
}
};
WallpaperStatusChecker wallpaperStatusChecker =
InjectorProvider.getInjector().getWallpaperStatusChecker();
boolean isLiveWallpaperSet =
WallpaperManager.getInstance(activity).getWallpaperInfo() != null;
// Alternative of ag/15567276
boolean isBuiltIn = !isLiveWallpaperSet
&& !wallpaperStatusChecker.isHomeStaticWallpaperSet(activity);
SetWallpaperDialogFragment setWallpaperDialog = new SetWallpaperDialogFragment();
setWallpaperDialog.setTitleResId(titleResId);
setWallpaperDialog.setListener(listenerWrapper);
if ((isLiveWallpaperSet || isBuiltIn)
&& !wallpaperStatusChecker.isLockWallpaperSet(activity)) {
if (isLiveWallpaper) {
// If lock wallpaper is live and we're setting a live wallpaper, we can only
// set it to both, so bypass the dialog.
listener.onSet(WallpaperPersister.DEST_BOTH);
restoreScreenOrientationIfNeeded(activity);
return;
}
// if the lock wallpaper is a live wallpaper, we cannot set a home-only static one
setWallpaperDialog.setHomeOptionAvailable(false);
}
if (isLiveWallpaper) {
setWallpaperDialog.setLockOptionAvailable(false);
}
setWallpaperDialog.show(fragmentManager, TAG_SET_WALLPAPER_DIALOG_FRAGMENT);
}
private void saveAndLockScreenOrientationIfNeeded(Activity activity) {
if (!mCurrentScreenOrientation.isPresent()) {
mCurrentScreenOrientation = Optional.of(activity.getRequestedOrientation());
activity.setRequestedOrientation(ActivityInfo.SCREEN_ORIENTATION_LOCKED);
}
}
private void restoreScreenOrientationIfNeeded(Activity activity) {
mCurrentScreenOrientation.ifPresent(orientation -> {
if (activity.getRequestedOrientation() != orientation) {
activity.setRequestedOrientation(orientation);
}
mCurrentScreenOrientation = Optional.empty();
});
}
}