| /* |
| * 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.wallpaper.picker; |
| |
| import android.app.Activity; |
| import android.app.ProgressDialog; |
| import android.content.Context; |
| import android.content.Intent; |
| import android.graphics.Color; |
| import android.graphics.Insets; |
| import android.graphics.PorterDuff.Mode; |
| import android.graphics.drawable.Drawable; |
| import android.net.Uri; |
| import android.os.AsyncTask; |
| import android.os.Build.VERSION; |
| import android.os.Build.VERSION_CODES; |
| import android.os.Bundle; |
| import android.provider.Settings; |
| import android.util.Log; |
| import android.view.View; |
| import android.view.View.OnClickListener; |
| import android.view.WindowInsets; |
| import android.widget.Button; |
| import android.widget.FrameLayout; |
| import android.widget.ImageView; |
| import android.widget.LinearLayout; |
| import android.widget.TextView; |
| |
| import androidx.annotation.NonNull; |
| import androidx.annotation.Nullable; |
| import androidx.appcompat.app.AlertDialog; |
| import androidx.appcompat.widget.Toolbar; |
| import androidx.fragment.app.Fragment; |
| import androidx.fragment.app.FragmentManager; |
| |
| import com.android.wallpaper.R; |
| import com.android.wallpaper.asset.Asset; |
| import com.android.wallpaper.compat.BuildCompat; |
| import com.android.wallpaper.compat.ButtonDrawableSetterCompat; |
| import com.android.wallpaper.config.Flags; |
| import com.android.wallpaper.model.Category; |
| import com.android.wallpaper.model.ImageWallpaperInfo; |
| import com.android.wallpaper.model.WallpaperInfo; |
| import com.android.wallpaper.module.CurrentWallpaperInfoFactory; |
| import com.android.wallpaper.module.CurrentWallpaperInfoFactory.WallpaperInfoCallback; |
| import com.android.wallpaper.module.DailyLoggingAlarmScheduler; |
| import com.android.wallpaper.module.ExploreIntentChecker; |
| import com.android.wallpaper.module.FormFactorChecker; |
| import com.android.wallpaper.module.Injector; |
| import com.android.wallpaper.module.InjectorProvider; |
| import com.android.wallpaper.module.NetworkStatusNotifier; |
| import com.android.wallpaper.module.NetworkStatusNotifier.NetworkStatus; |
| import com.android.wallpaper.module.UserEventLogger; |
| import com.android.wallpaper.module.UserEventLogger.WallpaperSetFailureReason; |
| import com.android.wallpaper.module.WallpaperPersister; |
| import com.android.wallpaper.module.WallpaperPersister.Destination; |
| import com.android.wallpaper.module.WallpaperPersister.SetWallpaperCallback; |
| import com.android.wallpaper.module.WallpaperPersister.WallpaperPosition; |
| import com.android.wallpaper.module.WallpaperPreferences; |
| import com.android.wallpaper.module.WallpaperPreferences.PresentationMode; |
| import com.android.wallpaper.module.WallpaperRotationRefresher; |
| import com.android.wallpaper.module.WallpaperRotationRefresher.Listener; |
| import com.android.wallpaper.picker.CategoryFragment.CategoryFragmentHost; |
| import com.android.wallpaper.picker.WallpaperDisabledFragment.WallpaperSupportLevel; |
| import com.android.wallpaper.picker.individual.IndividualPickerFragment; |
| import com.android.wallpaper.util.ScreenSizeCalculator; |
| import com.android.wallpaper.util.ThrowableAnalyzer; |
| import com.android.wallpaper.widget.BottomActionBar; |
| import com.android.wallpaper.widget.BottomActionBar.BottomActionBarHost; |
| |
| import com.google.android.material.bottomsheet.BottomSheetBehavior; |
| import com.google.android.material.bottomsheet.BottomSheetBehavior.BottomSheetCallback; |
| import com.google.android.material.tabs.TabLayout; |
| import com.google.android.material.tabs.TabLayout.OnTabSelectedListener; |
| import com.google.android.material.tabs.TabLayout.Tab; |
| |
| import java.util.List; |
| |
| /** |
| * Activity allowing users to select a category of wallpapers to choose from. |
| */ |
| public class TopLevelPickerActivity extends BaseActivity implements WallpapersUiContainer, |
| CurrentWallpaperBottomSheetPresenter, SetWallpaperErrorDialogFragment.Listener, |
| MyPhotosStarter, CategoryFragmentHost, BottomActionBarHost { |
| |
| private static final String TAG_SET_WALLPAPER_ERROR_DIALOG_FRAGMENT = |
| "toplevel_set_wallpaper_error_dialog"; |
| |
| private static final String TAG = "TopLevelPicker"; |
| private static final String KEY_SELECTED_CATEGORY_TAB = "selected_category_tab"; |
| |
| private WallpaperPickerDelegate mDelegate; |
| private int mLastSelectedCategoryTabIndex; |
| private UserEventLogger mUserEventLogger; |
| private NetworkStatusNotifier mNetworkStatusNotifier; |
| private NetworkStatusNotifier.Listener mNetworkStatusListener; |
| private WallpaperPersister mWallpaperPersister; |
| private WallpaperPreferences mWallpaperPreferences; |
| private boolean mWasCustomPhotoWallpaperSet; |
| @WallpaperPosition |
| private int mCustomPhotoWallpaperPosition; |
| |
| /** |
| * Progress dialogs for "refresh daily wallpaper" and "set wallpaper" operations. |
| */ |
| private ProgressDialog mRefreshWallpaperProgressDialog; |
| private ProgressDialog mSetWallpaperProgressDialog; |
| |
| /** |
| * Designates a test mode of operation -- in which certain UI features are disabled to allow for |
| * UI tests to run correctly. |
| */ |
| private boolean mTestingMode; |
| |
| /** |
| * UI for the "currently set wallpaper" BottomSheet. |
| */ |
| private LinearLayout mBottomSheet; |
| private ImageView mCurrentWallpaperImage; |
| private TextView mCurrentWallpaperPresentationMode; |
| private TextView mCurrentWallpaperTitle; |
| private TextView mCurrentWallpaperSubtitle; |
| private Button mCurrentWallpaperExploreButton; |
| private Button mCurrentWallpaperSkipWallpaperButton; |
| private FrameLayout mFragmentContainer; |
| private FrameLayout mLoadingIndicatorContainer; |
| private LinearLayout mWallpaperPositionOptions; |
| |
| /** |
| * Staged error dialog fragments that were unable to be shown when the activity didn't allow |
| * committing fragment transactions. |
| */ |
| private SetWallpaperErrorDialogFragment mStagedSetWallpaperErrorDialogFragment; |
| |
| /** |
| * A wallpaper pending set to the device--we retain a reference to this in order to facilitate |
| * retry or re-crop operations. |
| */ |
| private WallpaperInfo mPendingSetWallpaperInfo; |
| |
| private static int getTextColorIdForWallpaperPositionButton(boolean isSelected) { |
| return isSelected ? R.color.accent_color : R.color.material_grey500; |
| } |
| |
| @Override |
| protected void onCreate(Bundle savedInstanceState) { |
| super.onCreate(savedInstanceState); |
| |
| mLastSelectedCategoryTabIndex = -1; |
| |
| Injector injector = InjectorProvider.getInjector(); |
| mDelegate = new WallpaperPickerDelegate(this, this, injector); |
| mUserEventLogger = injector.getUserEventLogger(this); |
| mNetworkStatusNotifier = injector.getNetworkStatusNotifier(this); |
| mWallpaperPersister = injector.getWallpaperPersister(this); |
| mWallpaperPreferences = injector.getPreferences(this); |
| mWasCustomPhotoWallpaperSet = false; |
| |
| @WallpaperSupportLevel int wallpaperSupportLevel = mDelegate.getWallpaperSupportLevel(); |
| if (wallpaperSupportLevel != WallpaperDisabledFragment.SUPPORTED_CAN_SET) { |
| setContentView(R.layout.activity_top_level_picker); |
| |
| FragmentManager fm = getSupportFragmentManager(); |
| WallpaperDisabledFragment wallpaperDisabledFragment = |
| WallpaperDisabledFragment.newInstance(wallpaperSupportLevel); |
| fm.beginTransaction() |
| .add(R.id.fragment_container, wallpaperDisabledFragment) |
| .commit(); |
| return; |
| } |
| |
| if (mDelegate.getFormFactor() == FormFactorChecker.FORM_FACTOR_MOBILE) { |
| initializeMobile(true /* shouldForceRefresh */); |
| } else { // DESKTOP |
| initializeDesktop(savedInstanceState); |
| } |
| } |
| |
| @Override |
| protected void onResume() { |
| super.onResume(); |
| boolean provisioned = Settings.Global.getInt(getContentResolver(), |
| Settings.Global.DEVICE_PROVISIONED, 0) != 0; |
| |
| mUserEventLogger.logResumed(provisioned, true); |
| // Show the staged 'load wallpaper' or 'set wallpaper' error dialog fragments if there is one |
| // that was unable to be shown earlier when this fragment's hosting activity didn't allow |
| // committing fragment transactions. |
| if (mStagedSetWallpaperErrorDialogFragment != null) { |
| mStagedSetWallpaperErrorDialogFragment.show( |
| getSupportFragmentManager(), TAG_SET_WALLPAPER_ERROR_DIALOG_FRAGMENT); |
| mStagedSetWallpaperErrorDialogFragment = null; |
| } |
| } |
| |
| @Override |
| protected void onStop() { |
| mUserEventLogger.logStopped(); |
| super.onStop(); |
| } |
| |
| @Override |
| protected void onDestroy() { |
| super.onDestroy(); |
| mDelegate.cleanUp(); |
| if (mNetworkStatusListener != null) { |
| mNetworkStatusNotifier.unregisterListener(mNetworkStatusListener); |
| } |
| |
| if (mRefreshWallpaperProgressDialog != null) { |
| mRefreshWallpaperProgressDialog.dismiss(); |
| } |
| if (mSetWallpaperProgressDialog != null) { |
| mSetWallpaperProgressDialog.dismiss(); |
| } |
| } |
| |
| @Override |
| public void onBackPressed() { |
| Fragment fragment = getSupportFragmentManager().findFragmentById(R.id.fragment_container); |
| if (fragment instanceof BottomActionBarFragment |
| && ((BottomActionBarFragment) fragment).onBackPressed()) { |
| return; |
| } |
| |
| CategoryFragment categoryFragment = getCategoryFragment(); |
| if (categoryFragment != null && categoryFragment.popChildFragment()) { |
| return; |
| } |
| super.onBackPressed(); |
| } |
| |
| @Override |
| public void requestCustomPhotoPicker(PermissionChangedListener listener) { |
| mDelegate.requestCustomPhotoPicker(listener); |
| } |
| |
| @Override |
| public void requestExternalStoragePermission(PermissionChangedListener listener) { |
| mDelegate.requestExternalStoragePermission(listener); |
| } |
| |
| /** |
| * Returns whether READ_EXTERNAL_STORAGE has been granted for the application. |
| */ |
| public boolean isReadExternalStoragePermissionGranted() { |
| return mDelegate.isReadExternalStoragePermissionGranted(); |
| } |
| |
| private void initializeMobile(boolean shouldForceRefresh) { |
| setContentView(R.layout.activity_top_level_picker); |
| getWindow().getDecorView().setSystemUiVisibility( |
| getWindow().getDecorView().getSystemUiVisibility() |
| | View.SYSTEM_UI_FLAG_LAYOUT_HIDE_NAVIGATION |
| | View.SYSTEM_UI_FLAG_LAYOUT_STABLE); |
| View fragmentContainer = findViewById(R.id.fragment_container); |
| fragmentContainer.setOnApplyWindowInsetsListener((view, windowInsets) -> { |
| view.setPadding(view.getPaddingLeft(), windowInsets.getSystemWindowInsetTop(), |
| view.getPaddingRight(), view.getPaddingBottom()); |
| // Consume only the top inset (status bar), to let other content in the Activity consume |
| // the nav bar (ie, by using "fitSystemWindows") |
| if (BuildCompat.isAtLeastQ()) { |
| WindowInsets.Builder builder = new WindowInsets.Builder(windowInsets); |
| builder.setSystemWindowInsets(Insets.of(windowInsets.getSystemWindowInsetLeft(), |
| 0, windowInsets.getStableInsetRight(), |
| windowInsets.getSystemWindowInsetBottom())); |
| return builder.build(); |
| } else { |
| return windowInsets.replaceSystemWindowInsets( |
| windowInsets.getSystemWindowInsetLeft(), |
| 0, windowInsets.getStableInsetRight(), |
| windowInsets.getSystemWindowInsetBottom()); |
| } |
| }); |
| |
| FrameLayout.LayoutParams layoutParams = |
| (FrameLayout.LayoutParams) fragmentContainer.getLayoutParams(); |
| int bottomActionBarHeight = getResources() |
| .getDimensionPixelSize(R.dimen.bottom_navbar_height); |
| BottomActionBar bottomActionBar = findViewById(R.id.bottom_actionbar); |
| bottomActionBar.addVisibilityChangeListener(isVisible -> { |
| if (layoutParams != null) { |
| layoutParams.bottomMargin = isVisible ? bottomActionBarHeight : 0; |
| } |
| }); |
| |
| // Set toolbar as the action bar. |
| Toolbar toolbar = (Toolbar) findViewById(R.id.toolbar); |
| setSupportActionBar(toolbar); |
| |
| FragmentManager fm = getSupportFragmentManager(); |
| Fragment fragment = fm.findFragmentById(R.id.fragment_container); |
| |
| if (fragment == null) { |
| // App launch specific logic: log the "app launched" event and set up daily logging. |
| mUserEventLogger.logAppLaunched(); |
| mWallpaperPreferences.incrementAppLaunched(); |
| DailyLoggingAlarmScheduler.setAlarm(getApplicationContext()); |
| |
| CategoryFragment newFragment = CategoryFragment.newInstance( |
| getString(R.string.wallpaper_app_name)); |
| fm.beginTransaction() |
| .add(R.id.fragment_container, newFragment) |
| .commit(); |
| } |
| |
| mDelegate.initialize(shouldForceRefresh); |
| } |
| |
| private void initializeDesktop(Bundle savedInstanceState) { |
| setContentView(R.layout.activity_top_level_desktop); |
| |
| mBottomSheet = (LinearLayout) findViewById(R.id.bottom_sheet); |
| mCurrentWallpaperImage = (ImageView) mBottomSheet.findViewById(R.id.current_wallpaper_image); |
| mCurrentWallpaperImage.getLayoutParams().width = getSingleWallpaperImageWidthPx(); |
| |
| mCurrentWallpaperPresentationMode = |
| (TextView) mBottomSheet.findViewById(R.id.current_wallpaper_presentation_mode); |
| mCurrentWallpaperTitle = (TextView) findViewById(R.id.current_wallpaper_title); |
| mCurrentWallpaperSubtitle = (TextView) findViewById(R.id.current_wallpaper_subtitle); |
| mCurrentWallpaperExploreButton = (Button) findViewById( |
| R.id.current_wallpaper_explore_button); |
| mCurrentWallpaperSkipWallpaperButton = (Button) findViewById( |
| R.id.current_wallpaper_skip_wallpaper_button); |
| mFragmentContainer = (FrameLayout) findViewById(R.id.fragment_container); |
| mLoadingIndicatorContainer = (FrameLayout) findViewById(R.id.loading_indicator_container); |
| mWallpaperPositionOptions = (LinearLayout) findViewById( |
| R.id.desktop_wallpaper_position_options); |
| |
| final TabLayout tabLayout = (TabLayout) findViewById(R.id.tab_layout); |
| tabLayout.addOnTabSelectedListener(new OnTabSelectedListener() { |
| @Override |
| public void onTabSelected(Tab tab) { |
| Category category = (Category) tab.getTag(); |
| showCategoryDesktop(category.getCollectionId()); |
| mLastSelectedCategoryTabIndex = tabLayout.getSelectedTabPosition(); |
| } |
| |
| @Override |
| public void onTabUnselected(Tab tab) { |
| } |
| |
| @Override |
| public void onTabReselected(Tab tab) { |
| Category category = (Category) tab.getTag(); |
| // If offline, "My photos" may be the only visible category. In this case we want to allow |
| // re-selection so user can still select a photo as wallpaper while offline. |
| if (!category.isEnumerable()) { |
| onTabSelected(tab); |
| } |
| } |
| }); |
| |
| FragmentManager fm = getSupportFragmentManager(); |
| Fragment fragment = fm.findFragmentById(R.id.fragment_container); |
| |
| if (fragment == null) { |
| // App launch specific logic: log the "app launched" event and set up daily logging. |
| mUserEventLogger.logAppLaunched(); |
| mWallpaperPreferences.incrementAppLaunched(); |
| DailyLoggingAlarmScheduler.setAlarm(getApplicationContext()); |
| } |
| |
| mNetworkStatusListener = new NetworkStatusNotifier.Listener() { |
| @Override |
| public void onNetworkChanged(@NetworkStatus int networkStatus) { |
| initializeDesktopBasedOnNetwork(networkStatus, savedInstanceState); |
| } |
| }; |
| // Upon registering a listener, the onNetworkChanged method is immediately called with the |
| // initial network status. |
| mNetworkStatusNotifier.registerListener(mNetworkStatusListener); |
| } |
| |
| private void initializeDesktopBasedOnNetwork(@NetworkStatus int networkStatus, |
| Bundle savedInstanceState) { |
| if (networkStatus == NetworkStatusNotifier.NETWORK_CONNECTED) { |
| initializeDesktopOnline(savedInstanceState); |
| } else { |
| initializeDesktopOffline(); |
| } |
| } |
| |
| private void initializeDesktopOnline(Bundle savedInstanceState) { |
| FragmentManager fm = getSupportFragmentManager(); |
| Fragment fragment = fm.findFragmentById(R.id.fragment_container); |
| |
| // Require a category refresh if this is the first load of the app or if the app is now |
| // returning online after having been offline. |
| boolean forceCategoryRefresh = fragment == null || fragment instanceof OfflineDesktopFragment; |
| |
| if (fragment != null) { |
| fm.beginTransaction() |
| .remove(fragment) |
| .commit(); |
| } |
| |
| mLastSelectedCategoryTabIndex = savedInstanceState != null |
| ? savedInstanceState.getInt(KEY_SELECTED_CATEGORY_TAB) : -1; |
| mDelegate.populateCategories(forceCategoryRefresh); |
| |
| setDesktopLoading(true); |
| setUpBottomSheet(); |
| refreshCurrentWallpapers(null /* refreshListener */); |
| } |
| |
| private void initializeDesktopOffline() { |
| FragmentManager fm = getSupportFragmentManager(); |
| Fragment fragment = fm.findFragmentById(R.id.fragment_container); |
| |
| if (fragment != null) { |
| fm.beginTransaction() |
| .remove(fragment) |
| .commit(); |
| } |
| OfflineDesktopFragment newFragment = new OfflineDesktopFragment(); |
| fm.beginTransaction() |
| .add(R.id.fragment_container, newFragment) |
| .commit(); |
| |
| // Reset the last selected category tab index to ensure the app doesn't try to reselect a |
| // tab for a category not yet repopulated. |
| mLastSelectedCategoryTabIndex = -1; |
| |
| mDelegate.populateCategories(true /* forceCategoryRefresh */); |
| |
| setDesktopLoading(false); |
| setCurrentWallpapersExpanded(false); |
| } |
| |
| /** |
| * Sets the status of the loading indicator overlay in desktop mode. |
| * |
| * @param loading Whether an indeterminate loading indicator is displayed in place of the main |
| * fragment. |
| */ |
| private void setDesktopLoading(boolean loading) { |
| if (loading) { |
| mLoadingIndicatorContainer.setVisibility(View.VISIBLE); |
| mFragmentContainer.setVisibility(View.GONE); |
| } else { |
| mLoadingIndicatorContainer.setVisibility(View.GONE); |
| mFragmentContainer.setVisibility(View.VISIBLE); |
| } |
| } |
| |
| /** |
| * Returns the width (in physical px) to use for the "currently set wallpaper" thumbnail. |
| */ |
| private int getSingleWallpaperImageWidthPx() { |
| final float screenAspectRatio = |
| ScreenSizeCalculator.getInstance().getScreenAspectRatio(this); |
| |
| int height = getResources().getDimensionPixelSize( |
| R.dimen.current_wallpaper_bottom_sheet_thumb_height); |
| return (int) (height / screenAspectRatio); |
| } |
| |
| /** |
| * Enables and populates the "Currently set" wallpaper BottomSheet. |
| */ |
| private void setUpBottomSheet() { |
| mBottomSheet.setVisibility(View.VISIBLE); |
| |
| if (Flags.skipDailyWallpaperButtonEnabled) { |
| // Add "next" icon to the Next Wallpaper button |
| Drawable nextWallpaperButtonDrawable = getResources().getDrawable( |
| R.drawable.ic_refresh_18px); |
| |
| // This Drawable's state is shared across the app, so make a copy of it before applying a |
| // color tint as not to affect other clients elsewhere in the app. |
| nextWallpaperButtonDrawable = |
| nextWallpaperButtonDrawable.getConstantState().newDrawable().mutate(); |
| // Color the "compass" icon with the accent color. |
| nextWallpaperButtonDrawable.setColorFilter( |
| getResources().getColor(R.color.accent_color), Mode.SRC_IN); |
| ButtonDrawableSetterCompat.setDrawableToButtonStart( |
| mCurrentWallpaperSkipWallpaperButton, nextWallpaperButtonDrawable); |
| } |
| |
| final BottomSheetBehavior<LinearLayout> bottomSheetBehavior = |
| BottomSheetBehavior.from(mBottomSheet); |
| bottomSheetBehavior.setBottomSheetCallback(new BottomSheetCallback() { |
| @Override |
| public void onStateChanged(@NonNull View view, int i) { |
| } |
| |
| @Override |
| public void onSlide(@NonNull View view, float slideOffset) { |
| float alpha; |
| if (slideOffset >= 0) { |
| alpha = slideOffset; |
| } else { |
| alpha = 1f - slideOffset; |
| } |
| LinearLayout bottomSheetContents = findViewById(R.id.bottom_sheet_contents); |
| bottomSheetContents.setAlpha(alpha); |
| } |
| }); |
| } |
| |
| /** |
| * Enables a test mode of operation -- in which certain UI features are disabled to allow for |
| * UI tests to run correctly. Works around issue in ProgressDialog currently where the dialog |
| * constantly keeps the UI thread alive and blocks a test forever. |
| */ |
| void setTestingMode(boolean testingMode) { |
| mTestingMode = testingMode; |
| } |
| |
| /** |
| * Obtains the {@link WallpaperInfo} object(s) representing the wallpaper(s) currently set to the |
| * device from the {@link CurrentWallpaperInfoFactory} and displays them in the BottomSheet. |
| */ |
| @Override |
| public void refreshCurrentWallpapers(@Nullable RefreshListener refreshListener) { |
| final Injector injector = InjectorProvider.getInjector(); |
| final Context appContext = getApplicationContext(); |
| |
| CurrentWallpaperInfoFactory factory = injector.getCurrentWallpaperFactory(this); |
| factory.createCurrentWallpaperInfos(new WallpaperInfoCallback() { |
| @Override |
| public void onWallpaperInfoCreated( |
| final WallpaperInfo homeWallpaper, |
| @Nullable final WallpaperInfo lockWallpaper, |
| @PresentationMode final int presentationMode) { |
| |
| if (isDestroyed()) { |
| return; |
| } |
| |
| // Fetch the home wallpaper's thumbnail asset asynchronously to work around expensive |
| // method call to WallpaperManager#getWallpaperFile made from the CurrentWallpaperInfoVN |
| // getAsset() method. |
| AssetReceiver assetReceiver = (Asset thumbAsset) -> { |
| if (isDestroyed()) { |
| return; |
| } |
| |
| homeWallpaper.getThumbAsset(appContext).loadDrawableWithTransition( |
| TopLevelPickerActivity.this, |
| mCurrentWallpaperImage, |
| 200 /* transitionDurationMillis */, |
| () -> { |
| if (refreshListener != null) { |
| refreshListener.onCurrentWallpaperRefreshed(); |
| } |
| }, |
| Color.TRANSPARENT); |
| }; |
| new FetchThumbAssetTask(appContext, homeWallpaper, assetReceiver).executeOnExecutor( |
| AsyncTask.THREAD_POOL_EXECUTOR); |
| |
| mCurrentWallpaperPresentationMode.setText( |
| AttributionFormatter.getHumanReadableWallpaperPresentationMode( |
| TopLevelPickerActivity.this, presentationMode)); |
| |
| List<String> attributions = homeWallpaper.getAttributions(appContext); |
| if (attributions.size() > 0 && attributions.get(0) != null) { |
| mCurrentWallpaperTitle.setText(attributions.get(0)); |
| } |
| |
| mCurrentWallpaperSubtitle.setText( |
| AttributionFormatter.formatWallpaperSubtitle(appContext, homeWallpaper)); |
| |
| final String actionUrl = homeWallpaper.getActionUrl(appContext); |
| if (actionUrl != null && !actionUrl.isEmpty()) { |
| Uri exploreUri = Uri.parse(actionUrl); |
| |
| ExploreIntentChecker intentChecker = injector.getExploreIntentChecker(appContext); |
| intentChecker.fetchValidActionViewIntent(exploreUri, (@Nullable Intent exploreIntent) -> { |
| if (exploreIntent != null && !isDestroyed()) { |
| // Set the icon for the button |
| Drawable exploreButtonDrawable = getResources().getDrawable( |
| homeWallpaper.getActionIconRes(appContext)); |
| |
| // This Drawable's state is shared across the app, so make a copy of it |
| // before applying a color tint as not to affect other clients elsewhere |
| // in the app. |
| exploreButtonDrawable = exploreButtonDrawable.getConstantState() |
| .newDrawable().mutate(); |
| // Color the "compass" icon with the accent color. |
| exploreButtonDrawable.setColorFilter( |
| getResources().getColor(R.color.accent_color), Mode.SRC_IN); |
| |
| ButtonDrawableSetterCompat.setDrawableToButtonStart( |
| mCurrentWallpaperExploreButton, exploreButtonDrawable); |
| mCurrentWallpaperExploreButton.setText(getString( |
| homeWallpaper.getActionLabelRes(appContext))); |
| mCurrentWallpaperExploreButton.setVisibility(View.VISIBLE); |
| mCurrentWallpaperExploreButton.setOnClickListener(new OnClickListener() { |
| @Override |
| public void onClick(View v) { |
| mUserEventLogger.logActionClicked( |
| homeWallpaper.getCollectionId(appContext), |
| homeWallpaper.getActionLabelRes(appContext)); |
| startActivity(exploreIntent); |
| } |
| }); |
| } |
| }); |
| } else { |
| mCurrentWallpaperExploreButton.setVisibility(View.GONE); |
| } |
| |
| // Hide the wallpaper position options UI if the current home wallpaper is not from |
| // "my photos". |
| String homeCollectionId = homeWallpaper.getCollectionId(TopLevelPickerActivity.this); |
| if (mWallpaperPositionOptions != null |
| && homeCollectionId != null // May be null if app is being used for the first time. |
| && !homeCollectionId.equals(getString(R.string.image_wallpaper_collection_id))) { |
| mWallpaperPositionOptions.setVisibility(View.GONE); |
| } |
| |
| boolean showSkipWallpaperButton = Flags.skipDailyWallpaperButtonEnabled |
| && presentationMode == WallpaperPreferences.PRESENTATION_MODE_ROTATING; |
| if (showSkipWallpaperButton) { |
| mCurrentWallpaperSkipWallpaperButton.setVisibility(View.VISIBLE); |
| mCurrentWallpaperSkipWallpaperButton.setOnClickListener( |
| v -> refreshDailyWallpaper()); |
| } else { |
| mCurrentWallpaperSkipWallpaperButton.setVisibility(View.GONE); |
| } |
| |
| if (refreshListener != null) { |
| refreshListener.onCurrentWallpaperRefreshed(); |
| } |
| } |
| }, true /* forceRefresh */); |
| } |
| |
| @Override |
| public void onSaveInstanceState(Bundle savedInstanceState) { |
| FormFactorChecker formFactorChecker = InjectorProvider.getInjector().getFormFactorChecker(this); |
| if (formFactorChecker.getFormFactor() == FormFactorChecker.FORM_FACTOR_DESKTOP) { |
| TabLayout tabLayout = (TabLayout) findViewById(R.id.tab_layout); |
| |
| // tabLayout is only present when the main IndividualPickerFragment is present (as |
| // opposed to |
| // the WallpaperDisabledFragment), so need this null check. |
| if (tabLayout != null) { |
| savedInstanceState.putInt(KEY_SELECTED_CATEGORY_TAB, tabLayout.getSelectedTabPosition()); |
| } |
| } |
| |
| super.onSaveInstanceState(savedInstanceState); |
| } |
| |
| @Override |
| @Nullable |
| public CategoryFragment getCategoryFragment() { |
| if (mDelegate.getFormFactor() != FormFactorChecker.FORM_FACTOR_MOBILE) { |
| return null; |
| } |
| FragmentManager fm = getSupportFragmentManager(); |
| return (CategoryFragment) fm.findFragmentById(R.id.fragment_container); |
| } |
| |
| /** |
| * Populates the category tabs on DESKTOP form factor. |
| * |
| * @param selectedTabPosition The position of the tab to show as selected, or -1 if no particular |
| * tab should be selected (in which case: the tab of the category for the currently set |
| * wallpaper will be selected if enumerable; if not, the first enumerable category's tab will |
| * be selected). |
| */ |
| private void populateCategoryTabs(int selectedTabPosition) { |
| final TabLayout tabLayout = (TabLayout) findViewById(R.id.tab_layout); |
| tabLayout.removeAllTabs(); |
| |
| String currentlySetCollectionId = mDelegate.getPreferences().getHomeWallpaperCollectionId(); |
| |
| Tab tabToSelect = null; |
| Tab firstEnumerableCategoryTab = null; |
| for (int i = 0; i < mDelegate.getCategoryProvider().getSize(); i++) { |
| Category category = mDelegate.getCategoryProvider().getCategory(i); |
| |
| Tab tab = tabLayout.newTab(); |
| tab.setText(category.getTitle()); |
| tab.setTag(category); |
| tabLayout.addTab(tab, false /* setSelected */); |
| |
| if (firstEnumerableCategoryTab == null && category.isEnumerable()) { |
| firstEnumerableCategoryTab = tab; |
| } |
| |
| boolean shouldSelectTab = (i == selectedTabPosition) |
| || (selectedTabPosition == -1 |
| && tabToSelect == null |
| && category.isEnumerable() |
| && currentlySetCollectionId != null |
| && currentlySetCollectionId.equals(category.getCollectionId())); |
| |
| if (shouldSelectTab) { |
| tabToSelect = tab; |
| } |
| } |
| |
| // If the above loop did not identify a specific tab to select, then just select the tab for |
| // the first enumerable category. |
| if (tabToSelect == null) { |
| tabToSelect = firstEnumerableCategoryTab; |
| } |
| |
| // There may be no enumerable tabs (e.g., offline case), so we need to null-check again. |
| if (tabToSelect != null) { |
| tabToSelect.select(); |
| } |
| } |
| |
| /** |
| * Refreshes the current wallpaper in a daily wallpaper rotation. |
| */ |
| private void refreshDailyWallpaper() { |
| // ProgressDialog endlessly updates the UI thread, keeping it from going idle which therefore |
| // causes Espresso to hang once the dialog is shown. |
| if (!mTestingMode) { |
| int themeResId; |
| if (VERSION.SDK_INT < VERSION_CODES.LOLLIPOP) { |
| themeResId = R.style.ProgressDialogThemePreL; |
| } else { |
| themeResId = R.style.LightDialogTheme; |
| } |
| mRefreshWallpaperProgressDialog = new ProgressDialog(this, themeResId); |
| mRefreshWallpaperProgressDialog.setTitle(null); |
| mRefreshWallpaperProgressDialog.setMessage( |
| getResources().getString(R.string.refreshing_daily_wallpaper_dialog_message)); |
| mRefreshWallpaperProgressDialog.setIndeterminate(true); |
| mRefreshWallpaperProgressDialog.show(); |
| } |
| |
| WallpaperRotationRefresher wallpaperRotationRefresher = |
| InjectorProvider.getInjector().getWallpaperRotationRefresher(); |
| wallpaperRotationRefresher.refreshWallpaper(this, new Listener() { |
| @Override |
| public void onRefreshed() { |
| if (isDestroyed()) { |
| return; |
| } |
| |
| if (mRefreshWallpaperProgressDialog != null) { |
| mRefreshWallpaperProgressDialog.dismiss(); |
| } |
| |
| refreshCurrentWallpapers(null /* refreshListener */); |
| } |
| |
| @Override |
| public void onError() { |
| if (mRefreshWallpaperProgressDialog != null) { |
| mRefreshWallpaperProgressDialog.dismiss(); |
| } |
| |
| AlertDialog errorDialog = new AlertDialog.Builder( |
| TopLevelPickerActivity.this, R.style.LightDialogTheme) |
| .setMessage(R.string.refresh_daily_wallpaper_failed_message) |
| .setPositiveButton(android.R.string.ok, null /* onClickListener */) |
| .create(); |
| errorDialog.show(); |
| } |
| }); |
| } |
| |
| @Override |
| protected void onActivityResult(int requestCode, int resultCode, Intent data) { |
| super.onActivityResult(requestCode, resultCode, data); |
| |
| if (requestCode == WallpaperPickerDelegate.SHOW_CATEGORY_REQUEST_CODE |
| && resultCode == Activity.RESULT_OK) { |
| if (mDelegate.getFormFactor() == FormFactorChecker.FORM_FACTOR_DESKTOP) { |
| Uri imageUri = (data == null) ? null : data.getData(); |
| if (imageUri != null) { |
| // User selected an image from the system picker, so launch the preview for that |
| // image. |
| ImageWallpaperInfo imageWallpaper = new ImageWallpaperInfo(imageUri); |
| setCustomPhotoWallpaper(imageWallpaper); |
| return; |
| } |
| } |
| } |
| if (mDelegate.handleActivityResult(requestCode, resultCode, data)) { |
| finishActivityWithResultOk(); |
| } |
| } |
| |
| /** |
| * Shows the view-only preview activity for the given wallpaper. |
| */ |
| public void showViewOnlyPreview(WallpaperInfo wallpaperInfo, boolean isViewAsHome) { |
| mDelegate.showViewOnlyPreview(wallpaperInfo, isViewAsHome); |
| } |
| |
| @Override |
| public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, |
| @NonNull int[] grantResults) { |
| mDelegate.onRequestPermissionsResult(requestCode, permissions, grantResults); |
| } |
| |
| private void reselectLastTab() { |
| TabLayout tabLayout = (TabLayout) findViewById(R.id.tab_layout); |
| |
| // In the offline case, "My photos" could be the only category. Thus we need this check -- |
| // to ensure that we don't try to select the "previously selected" category which was -1. |
| if (mLastSelectedCategoryTabIndex > -1) { |
| Tab tabToSelect = tabLayout.getTabAt( |
| mLastSelectedCategoryTabIndex); |
| if (((Category) tabToSelect.getTag()).isEnumerable()) { |
| tabToSelect.select(); |
| } |
| } |
| } |
| |
| private void showCategoryDesktop(String collectionId) { |
| Category category = mDelegate.findCategoryForCollectionId(collectionId); |
| if (category == null) { |
| return; |
| } |
| |
| if (category.isEnumerable()) { |
| // Replace contained IndividualPickerFragment with a new instance for the given category. |
| final FragmentManager fm = getSupportFragmentManager(); |
| Fragment fragment = fm.findFragmentById(R.id.fragment_container); |
| if (fragment != null) { |
| fm.beginTransaction() |
| .remove(fragment) |
| .commit(); |
| } |
| Injector injector = InjectorProvider.getInjector(); |
| IndividualPickerFragment newFragment = injector.getIndividualPickerFragment( |
| collectionId); |
| fm.beginTransaction() |
| .add(R.id.fragment_container, newFragment) |
| .commit(); |
| newFragment.setCurrentWallpaperBottomSheetPresenter(this); |
| newFragment.setWallpapersUiContainer(this); |
| } else { |
| category.show(this, mDelegate.getPickerIntentFactory(), |
| WallpaperPickerDelegate.SHOW_CATEGORY_REQUEST_CODE); |
| |
| // Need to select the tab here in case we are coming back from a "My photos" in which case |
| // the tab would have been set to "My photos" while viewing a regular image category. |
| reselectLastTab(); |
| } |
| } |
| |
| private void finishActivityWithResultOk() { |
| overridePendingTransition(R.anim.fade_in, R.anim.fade_out); |
| setResult(Activity.RESULT_OK); |
| finish(); |
| } |
| |
| @Override |
| public void setCurrentWallpapersExpanded(boolean expanded) { |
| final BottomSheetBehavior<LinearLayout> bottomSheetBehavior = |
| BottomSheetBehavior.from(mBottomSheet); |
| bottomSheetBehavior.setState( |
| expanded ? BottomSheetBehavior.STATE_EXPANDED |
| : BottomSheetBehavior.STATE_COLLAPSED); |
| } |
| |
| @Override |
| public void doneFetchingCategories() { |
| populateCategoryTabs(mLastSelectedCategoryTabIndex); |
| } |
| |
| @Override |
| public void onWallpapersReady() { |
| setDesktopLoading(false); |
| setCurrentWallpapersExpanded(true); |
| } |
| |
| @Override |
| public MyPhotosStarter getMyPhotosStarter() { |
| return this; |
| } |
| |
| @Override |
| public void onClickTryAgain(@Destination int unused) { |
| // Retry the set wallpaper operation with the default center-crop setting. |
| if (mPendingSetWallpaperInfo != null) { |
| setCustomPhotoWallpaper(mPendingSetWallpaperInfo); |
| } |
| } |
| |
| /** |
| * Sets the provides wallpaper to the device with center-cropped and scaled to fit the device's |
| * default display. |
| */ |
| private void setCustomPhotoWallpaper(final WallpaperInfo wallpaper) { |
| // Save this WallpaperInfo so we can retry this operation later if it fails. |
| mPendingSetWallpaperInfo = wallpaper; |
| |
| showSettingWallpaperProgressDialog(); |
| |
| mWallpaperPersister.setIndividualWallpaperWithPosition(this, wallpaper, |
| WallpaperPersister.WALLPAPER_POSITION_CENTER_CROP, new SetWallpaperCallback() { |
| @Override |
| public void onSuccess(WallpaperInfo wallpaperInfo) { |
| dismissSettingWallpaperProgressDialog(); |
| refreshCurrentWallpapers(null /* refreshListener */); |
| |
| mDelegate.getPreferences().setPendingWallpaperSetStatus( |
| WallpaperPreferences.WALLPAPER_SET_NOT_PENDING); |
| mUserEventLogger.logWallpaperSet( |
| wallpaper.getCollectionId(getApplicationContext()), |
| wallpaper.getWallpaperId()); |
| mUserEventLogger.logWallpaperSetResult(UserEventLogger.WALLPAPER_SET_RESULT_SUCCESS); |
| |
| // The user may have closed the activity before the set wallpaper operation completed. |
| if (isDestroyed()) { |
| return; |
| } |
| |
| // Show the wallpaper crop option selector and bind click event handlers. |
| mWallpaperPositionOptions.setVisibility(View.VISIBLE); |
| |
| mWasCustomPhotoWallpaperSet = true; |
| mCustomPhotoWallpaperPosition = WallpaperPersister.WALLPAPER_POSITION_CENTER_CROP; |
| |
| initializeWallpaperPositionOptionClickHandlers(wallpaper); |
| } |
| |
| @Override |
| public void onError(Throwable throwable) { |
| dismissSettingWallpaperProgressDialog(); |
| showSetWallpaperErrorDialog(); |
| |
| mDelegate.getPreferences().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); |
| Log.e(TAG, "Unable to set wallpaper from 'my photos'."); |
| } |
| }); |
| } |
| |
| /** |
| * Initializes the wallpaper position button click handlers to change the way the provided |
| * wallpaper is set to the device. |
| */ |
| private void initializeWallpaperPositionOptionClickHandlers(final WallpaperInfo wallpaperInfo) { |
| Button centerCropOptionBtn = (Button) findViewById(R.id.wallpaper_position_option_center_crop); |
| Button stretchOptionBtn = (Button) findViewById(R.id.wallpaper_position_option_stretched); |
| Button centerOptionBtn = (Button) findViewById(R.id.wallpaper_position_option_center); |
| |
| // The "center crop" wallpaper position button is selected by default. |
| setCenterCropWallpaperPositionButtonSelected(centerCropOptionBtn, true /* isSelected */); |
| centerCropOptionBtn.setOnClickListener(new OnClickListener() { |
| @Override |
| public void onClick(View view) { |
| mWallpaperPersister.setIndividualWallpaperWithPosition(TopLevelPickerActivity.this, |
| wallpaperInfo, WallpaperPersister.WALLPAPER_POSITION_CENTER_CROP, |
| new SetWallpaperCallback() { |
| @Override |
| public void onSuccess(WallpaperInfo wallpaperInfo) { |
| // The user may have closed the activity before the set wallpaper operation |
| // completed. |
| if (isDestroyed()) { |
| return; |
| } |
| |
| refreshCurrentWallpapers(null /* refreshListener */); |
| |
| setCenterCropWallpaperPositionButtonSelected( |
| centerCropOptionBtn, true /* isSelected */); |
| setCenterWallpaperPositionButtonSelected(centerOptionBtn, false /* isSelected */); |
| setStretchWallpaperPositionButtonSelected(stretchOptionBtn, false /* isSelected */); |
| |
| mCustomPhotoWallpaperPosition = WallpaperPersister.WALLPAPER_POSITION_CENTER_CROP; |
| } |
| |
| @Override |
| public void onError(@Nullable Throwable throwable) { |
| // no-op |
| } |
| }); |
| } |
| }); |
| |
| // "Stretch" is not selected by default. |
| setStretchWallpaperPositionButtonSelected(stretchOptionBtn, false /* isSelected */); |
| stretchOptionBtn.setOnClickListener(new OnClickListener() { |
| @Override |
| public void onClick(View view) { |
| mWallpaperPersister.setIndividualWallpaperWithPosition(TopLevelPickerActivity.this, |
| wallpaperInfo, WallpaperPersister.WALLPAPER_POSITION_STRETCH, |
| new SetWallpaperCallback() { |
| @Override |
| public void onSuccess(WallpaperInfo wallpaperInfo) { |
| // The user may have closed the activity before the set wallpaper operation |
| // completed. |
| if (isDestroyed()) { |
| return; |
| } |
| |
| refreshCurrentWallpapers(null /* refreshListener */); |
| |
| setStretchWallpaperPositionButtonSelected(stretchOptionBtn, true /* isSelected */); |
| setCenterCropWallpaperPositionButtonSelected( |
| centerCropOptionBtn, false /* isSelected */); |
| setCenterWallpaperPositionButtonSelected(centerOptionBtn, false /* isSelected */); |
| |
| mCustomPhotoWallpaperPosition = WallpaperPersister.WALLPAPER_POSITION_STRETCH; |
| } |
| |
| @Override |
| public void onError(@Nullable Throwable throwable) { |
| // no-op |
| } |
| }); |
| } |
| }); |
| |
| // "Center" is not selected by default. |
| setCenterWallpaperPositionButtonSelected(centerOptionBtn, false /* isSelected */); |
| centerOptionBtn.setOnClickListener(new OnClickListener() { |
| @Override |
| public void onClick(View view) { |
| mWallpaperPersister.setIndividualWallpaperWithPosition(TopLevelPickerActivity.this, |
| wallpaperInfo, WallpaperPersister.WALLPAPER_POSITION_CENTER, |
| new SetWallpaperCallback() { |
| @Override |
| public void onSuccess(WallpaperInfo wallpaperInfo) { |
| // The user may have closed the activity before the set wallpaper operation |
| // completed. |
| if (isDestroyed()) { |
| return; |
| } |
| |
| refreshCurrentWallpapers(null /* refreshListener */); |
| |
| setCenterWallpaperPositionButtonSelected(centerOptionBtn, true /* isSelected */); |
| setCenterCropWallpaperPositionButtonSelected( |
| centerCropOptionBtn, false /* isSelected */); |
| setStretchWallpaperPositionButtonSelected(stretchOptionBtn, false /* isSelected */); |
| |
| mCustomPhotoWallpaperPosition = WallpaperPersister.WALLPAPER_POSITION_CENTER; |
| } |
| |
| @Override |
| public void onError(@Nullable Throwable throwable) { |
| // no-op |
| } |
| }); |
| } |
| }); |
| } |
| |
| private void setCenterWallpaperPositionButtonSelected(Button button, boolean isSelected) { |
| int drawableId = isSelected ? R.drawable.center_blue : R.drawable.center_grey; |
| ButtonDrawableSetterCompat.setDrawableToButtonStart(button, getDrawable(drawableId)); |
| button.setTextColor(getColor(getTextColorIdForWallpaperPositionButton(isSelected))); |
| } |
| |
| private void setCenterCropWallpaperPositionButtonSelected(Button button, boolean isSelected) { |
| int drawableId = isSelected ? R.drawable.center_crop_blue : R.drawable.center_crop_grey; |
| ButtonDrawableSetterCompat.setDrawableToButtonStart(button, getDrawable(drawableId)); |
| button.setTextColor(getColor(getTextColorIdForWallpaperPositionButton(isSelected))); |
| } |
| |
| private void setStretchWallpaperPositionButtonSelected(Button button, boolean isSelected) { |
| int drawableId = isSelected ? R.drawable.stretch_blue : R.drawable.stretch_grey; |
| ButtonDrawableSetterCompat.setDrawableToButtonStart(button, getDrawable(drawableId)); |
| button.setTextColor(getColor(getTextColorIdForWallpaperPositionButton(isSelected))); |
| } |
| |
| private void showSettingWallpaperProgressDialog() { |
| // ProgressDialog endlessly updates the UI thread, keeping it from going idle which |
| // therefore causes Espresso to hang once the dialog is shown. |
| if (!mTestingMode) { |
| int themeResId; |
| if (VERSION.SDK_INT < VERSION_CODES.LOLLIPOP) { |
| themeResId = R.style.ProgressDialogThemePreL; |
| } else { |
| themeResId = R.style.LightDialogTheme; |
| } |
| mSetWallpaperProgressDialog = new ProgressDialog(this, themeResId); |
| mSetWallpaperProgressDialog.setTitle(null); |
| mSetWallpaperProgressDialog.setMessage( |
| getResources().getString(R.string.set_wallpaper_progress_message)); |
| mSetWallpaperProgressDialog.setIndeterminate(true); |
| mSetWallpaperProgressDialog.show(); |
| } |
| } |
| |
| private void dismissSettingWallpaperProgressDialog() { |
| if (mSetWallpaperProgressDialog != null) { |
| mSetWallpaperProgressDialog.dismiss(); |
| } |
| } |
| |
| private void showSetWallpaperErrorDialog() { |
| SetWallpaperErrorDialogFragment dialogFragment = SetWallpaperErrorDialogFragment.newInstance( |
| R.string.set_wallpaper_error_message, WallpaperPersister.DEST_BOTH); |
| |
| if (isSafeToCommitFragmentTransaction()) { |
| dialogFragment.show(getSupportFragmentManager(), TAG_SET_WALLPAPER_ERROR_DIALOG_FRAGMENT); |
| } else { |
| mStagedSetWallpaperErrorDialogFragment = dialogFragment; |
| } |
| } |
| |
| @Override |
| public BottomActionBar getBottomActionBar() { |
| return findViewById(R.id.bottom_actionbar); |
| } |
| |
| private interface AssetReceiver { |
| void onAssetReceived(Asset asset); |
| } |
| |
| /** |
| * An AsyncTask for asynchronously fetching the thumbnail asset for a given WallpaperInfo. |
| * Used to work around expensive method call to WallpaperManager#getWallpaperFile made from the |
| * CurrentWallpaperInfoVN getAsset() method. |
| */ |
| private static class FetchThumbAssetTask extends AsyncTask<Void, Void, Asset> { |
| private Context mAppContext; |
| private WallpaperInfo mWallpaperInfo; |
| private AssetReceiver mReceiver; |
| |
| public FetchThumbAssetTask(Context appContext, WallpaperInfo wallpaperInfo, |
| AssetReceiver receiver) { |
| mAppContext = appContext; |
| mWallpaperInfo = wallpaperInfo; |
| mReceiver = receiver; |
| } |
| |
| @Override |
| protected Asset doInBackground(Void... params) { |
| return mWallpaperInfo.getThumbAsset(mAppContext); |
| } |
| |
| @Override |
| protected void onPostExecute(Asset thumbAsset) { |
| mReceiver.onAssetReceived(thumbAsset); |
| } |
| } |
| } |