Merge "Improve UX of Deferred Setup"
This commit is contained in:
@@ -79,6 +79,10 @@ public class ContextualCardLookupTable {
|
||||
LegacySuggestionContextualCardRenderer.VIEW_TYPE,
|
||||
LegacySuggestionContextualCardController.class,
|
||||
LegacySuggestionContextualCardRenderer.class));
|
||||
add(new ControllerRendererMapping(CardType.SLICE,
|
||||
SliceContextualCardRenderer.VIEW_TYPE_DEFERRED_SETUP,
|
||||
SliceContextualCardController.class,
|
||||
SliceContextualCardRenderer.class));
|
||||
add(new ControllerRendererMapping(CardType.SLICE,
|
||||
SliceContextualCardRenderer.VIEW_TYPE_FULL_WIDTH,
|
||||
SliceContextualCardController.class,
|
||||
|
@@ -17,6 +17,7 @@
|
||||
package com.android.settings.homepage.contextualcards;
|
||||
|
||||
import static com.android.settings.homepage.contextualcards.ContextualCardLoader.CARD_CONTENT_LOADER_ID;
|
||||
import static com.android.settings.intelligence.ContextualCardProto.ContextualCard.Category.DEFERRED_SETUP_VALUE;
|
||||
import static com.android.settings.intelligence.ContextualCardProto.ContextualCard.Category.SUGGESTION_VALUE;
|
||||
|
||||
import static java.util.stream.Collectors.groupingBy;
|
||||
@@ -71,17 +72,15 @@ public class ContextualCardManager implements ContextualCardLoader.CardContentLo
|
||||
|
||||
@VisibleForTesting
|
||||
final List<ContextualCard> mContextualCards;
|
||||
private final Context mContext;
|
||||
private final ControllerRendererPool mControllerRendererPool;
|
||||
private final Lifecycle mLifecycle;
|
||||
private final List<LifecycleObserver> mLifecycleObservers;
|
||||
@VisibleForTesting
|
||||
long mStartTime;
|
||||
boolean mIsFirstLaunch;
|
||||
@VisibleForTesting
|
||||
List<String> mSavedCards;
|
||||
|
||||
private final Context mContext;
|
||||
private final ControllerRendererPool mControllerRendererPool;
|
||||
private final Lifecycle mLifecycle;
|
||||
private final List<LifecycleObserver> mLifecycleObservers;
|
||||
|
||||
private ContextualCardUpdateListener mListener;
|
||||
|
||||
public ContextualCardManager(Context context, Lifecycle lifecycle, Bundle savedInstanceState) {
|
||||
@@ -175,7 +174,7 @@ public class ContextualCardManager implements ContextualCardLoader.CardContentLo
|
||||
//replace with the new data
|
||||
mContextualCards.clear();
|
||||
final List<ContextualCard> sortedCards = sortCards(allCards);
|
||||
mContextualCards.addAll(assignCardWidth(sortedCards));
|
||||
mContextualCards.addAll(getCardsWithViewType(sortedCards));
|
||||
|
||||
loadCardControllers();
|
||||
|
||||
@@ -228,10 +227,19 @@ public class ContextualCardManager implements ContextualCardLoader.CardContentLo
|
||||
}
|
||||
|
||||
@VisibleForTesting
|
||||
List<ContextualCard> assignCardWidth(List<ContextualCard> cards) {
|
||||
final List<ContextualCard> result = new ArrayList<>(cards);
|
||||
List<ContextualCard> getCardsWithViewType(List<ContextualCard> cards) {
|
||||
if (cards.isEmpty()) {
|
||||
return cards;
|
||||
}
|
||||
|
||||
final List<ContextualCard> result = getCardsWithDeferredSetupViewType(cards);
|
||||
return getCardsWithSuggestionViewType(result);
|
||||
}
|
||||
|
||||
private List<ContextualCard> getCardsWithSuggestionViewType(List<ContextualCard> cards) {
|
||||
// Shows as half cards if 2 suggestion type of cards are next to each other.
|
||||
// Shows as full card if 1 suggestion type of card lives alone.
|
||||
final List<ContextualCard> result = new ArrayList<>(cards);
|
||||
for (int index = 1; index < result.size(); index++) {
|
||||
final ContextualCard previous = result.get(index - 1);
|
||||
final ContextualCard current = result.get(index);
|
||||
@@ -247,6 +255,22 @@ public class ContextualCardManager implements ContextualCardLoader.CardContentLo
|
||||
return result;
|
||||
}
|
||||
|
||||
private List<ContextualCard> getCardsWithDeferredSetupViewType(List<ContextualCard> cards) {
|
||||
// Find the deferred setup card and assign it with proper view type.
|
||||
// Reason: The returned card list will mix deferred setup card and other suggestion cards
|
||||
// after device running 1 days.
|
||||
final List<ContextualCard> result = new ArrayList<>(cards);
|
||||
for (int index = 0; index < result.size(); index++) {
|
||||
final ContextualCard card = cards.get(index);
|
||||
if (card.getCategory() == DEFERRED_SETUP_VALUE) {
|
||||
result.set(index, card.mutate().setViewType(
|
||||
SliceContextualCardRenderer.VIEW_TYPE_DEFERRED_SETUP).build());
|
||||
return result;
|
||||
}
|
||||
}
|
||||
return result;
|
||||
}
|
||||
|
||||
private List<ContextualCard> getCardsToKeep(List<ContextualCard> cards) {
|
||||
if (mSavedCards != null) {
|
||||
//screen rotate
|
||||
|
@@ -52,6 +52,7 @@ import java.util.Set;
|
||||
public class SliceContextualCardRenderer implements ContextualCardRenderer, LifecycleObserver {
|
||||
public static final int VIEW_TYPE_FULL_WIDTH = R.layout.homepage_slice_tile;
|
||||
public static final int VIEW_TYPE_HALF_WIDTH = R.layout.homepage_slice_half_tile;
|
||||
public static final int VIEW_TYPE_DEFERRED_SETUP = R.layout.homepage_slice_deferred_setup_tile;
|
||||
|
||||
private static final String TAG = "SliceCardRenderer";
|
||||
|
||||
@@ -64,6 +65,7 @@ public class SliceContextualCardRenderer implements ContextualCardRenderer, Life
|
||||
private final LifecycleOwner mLifecycleOwner;
|
||||
private final ControllerRendererPool mControllerRendererPool;
|
||||
private final Set<ContextualCard> mCardSet;
|
||||
private final SliceDeferredSetupCardRendererHelper mDeferredSetupCardHelper;
|
||||
private final SliceFullCardRendererHelper mFullCardHelper;
|
||||
private final SliceHalfCardRendererHelper mHalfCardHelper;
|
||||
|
||||
@@ -78,11 +80,14 @@ public class SliceContextualCardRenderer implements ContextualCardRenderer, Life
|
||||
mLifecycleOwner.getLifecycle().addObserver(this);
|
||||
mFullCardHelper = new SliceFullCardRendererHelper(context);
|
||||
mHalfCardHelper = new SliceHalfCardRendererHelper(context);
|
||||
mDeferredSetupCardHelper = new SliceDeferredSetupCardRendererHelper(context);
|
||||
}
|
||||
|
||||
@Override
|
||||
public RecyclerView.ViewHolder createViewHolder(View view, @LayoutRes int viewType) {
|
||||
switch (viewType) {
|
||||
case VIEW_TYPE_DEFERRED_SETUP:
|
||||
return mDeferredSetupCardHelper.createViewHolder(view);
|
||||
case VIEW_TYPE_HALF_WIDTH:
|
||||
return mHalfCardHelper.createViewHolder(view);
|
||||
default:
|
||||
@@ -119,17 +124,25 @@ public class SliceContextualCardRenderer implements ContextualCardRenderer, Life
|
||||
//TODO(b/120629936): Take this out once blank card issue is fixed.
|
||||
Log.d(TAG, "Slice callback - uri = " + slice.getUri());
|
||||
}
|
||||
if (holder.getItemViewType() == VIEW_TYPE_HALF_WIDTH) {
|
||||
mHalfCardHelper.bindView(holder, card, slice);
|
||||
} else {
|
||||
mFullCardHelper.bindView(holder, card, slice, mCardSet);
|
||||
switch (holder.getItemViewType()) {
|
||||
case VIEW_TYPE_DEFERRED_SETUP:
|
||||
mDeferredSetupCardHelper.bindView(holder, card, slice);
|
||||
break;
|
||||
case VIEW_TYPE_HALF_WIDTH:
|
||||
mHalfCardHelper.bindView(holder, card, slice);
|
||||
break;
|
||||
default:
|
||||
mFullCardHelper.bindView(holder, card, slice, mCardSet);
|
||||
}
|
||||
});
|
||||
|
||||
if (holder.getItemViewType() == VIEW_TYPE_HALF_WIDTH) {
|
||||
initDismissalActions(holder, card, R.id.content);
|
||||
} else {
|
||||
initDismissalActions(holder, card, R.id.slice_view);
|
||||
switch (holder.getItemViewType()) {
|
||||
case VIEW_TYPE_DEFERRED_SETUP:
|
||||
case VIEW_TYPE_HALF_WIDTH:
|
||||
initDismissalActions(holder, card, R.id.content);
|
||||
break;
|
||||
default:
|
||||
initDismissalActions(holder, card, R.id.slice_view);
|
||||
}
|
||||
}
|
||||
|
||||
|
@@ -0,0 +1,91 @@
|
||||
/*
|
||||
* Copyright (C) 2019 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.settings.homepage.contextualcards.slices;
|
||||
|
||||
import android.app.PendingIntent;
|
||||
import android.content.Context;
|
||||
import android.util.Log;
|
||||
import android.view.View;
|
||||
import android.widget.Button;
|
||||
import android.widget.ImageView;
|
||||
import android.widget.LinearLayout;
|
||||
import android.widget.TextView;
|
||||
|
||||
import androidx.recyclerview.widget.RecyclerView;
|
||||
import androidx.slice.Slice;
|
||||
import androidx.slice.SliceMetadata;
|
||||
import androidx.slice.core.SliceAction;
|
||||
import androidx.slice.widget.EventInfo;
|
||||
|
||||
import com.android.settings.R;
|
||||
import com.android.settings.homepage.contextualcards.ContextualCard;
|
||||
import com.android.settings.homepage.contextualcards.ContextualCardFeatureProvider;
|
||||
import com.android.settings.overlay.FeatureFactory;
|
||||
|
||||
/**
|
||||
* Card renderer helper for {@link ContextualCard} built as slice deferred setup card.
|
||||
*/
|
||||
class SliceDeferredSetupCardRendererHelper {
|
||||
private static final String TAG = "SliceDSCRendererHelper";
|
||||
|
||||
private final Context mContext;
|
||||
|
||||
SliceDeferredSetupCardRendererHelper(Context context) {
|
||||
mContext = context;
|
||||
}
|
||||
|
||||
RecyclerView.ViewHolder createViewHolder(View view) {
|
||||
return new DeferredSetupCardViewHolder(view);
|
||||
}
|
||||
|
||||
void bindView(RecyclerView.ViewHolder holder, ContextualCard card, Slice slice) {
|
||||
final DeferredSetupCardViewHolder view = (DeferredSetupCardViewHolder) holder;
|
||||
final SliceMetadata sliceMetadata = SliceMetadata.from(mContext, slice);
|
||||
final SliceAction primaryAction = sliceMetadata.getPrimaryAction();
|
||||
view.icon.setImageDrawable(primaryAction.getIcon().loadDrawable(mContext));
|
||||
view.title.setText(primaryAction.getTitle());
|
||||
view.summary.setText(sliceMetadata.getSubtitle());
|
||||
view.button.setOnClickListener(v -> {
|
||||
try {
|
||||
primaryAction.getAction().send();
|
||||
} catch (PendingIntent.CanceledException e) {
|
||||
Log.w(TAG, "Failed to start intent " + primaryAction.getTitle());
|
||||
}
|
||||
final ContextualCardFeatureProvider contextualCardFeatureProvider =
|
||||
FeatureFactory.getFactory(mContext).getContextualCardFeatureProvider(mContext);
|
||||
contextualCardFeatureProvider.logContextualCardClick(card, 0 /* row */,
|
||||
EventInfo.ACTION_TYPE_CONTENT);
|
||||
});
|
||||
}
|
||||
|
||||
static class DeferredSetupCardViewHolder extends RecyclerView.ViewHolder {
|
||||
public final LinearLayout content;
|
||||
public final ImageView icon;
|
||||
public final TextView title;
|
||||
public final TextView summary;
|
||||
public final Button button;
|
||||
|
||||
public DeferredSetupCardViewHolder(View itemView) {
|
||||
super(itemView);
|
||||
content = itemView.findViewById(R.id.content);
|
||||
icon = itemView.findViewById(android.R.id.icon);
|
||||
title = itemView.findViewById(android.R.id.title);
|
||||
summary = itemView.findViewById(android.R.id.summary);
|
||||
button = itemView.findViewById(R.id.finish_setup);
|
||||
}
|
||||
}
|
||||
}
|
Reference in New Issue
Block a user