/* * Copyright (C) 2015 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.tv.onboarding; import android.app.Activity; import android.graphics.Typeface; import android.media.tv.TvInputInfo; import android.media.tv.TvInputManager.TvInputCallback; import android.os.Bundle; import android.support.annotation.NonNull; import android.text.TextUtils; import android.view.LayoutInflater; import android.view.View; import android.view.ViewGroup; import android.widget.TextView; import androidx.leanback.widget.GuidanceStylist.Guidance; import androidx.leanback.widget.GuidedAction; import androidx.leanback.widget.GuidedActionsStylist; import androidx.leanback.widget.VerticalGridView; import com.android.tv.R; import com.android.tv.TvSingletons; import com.android.tv.common.ui.setup.SetupGuidedStepFragment; import com.android.tv.common.ui.setup.SetupMultiPaneFragment; import com.android.tv.data.ChannelDataManager; import com.android.tv.data.TvInputNewComparator; import com.android.tv.tunerinputcontroller.BuiltInTunerManager; import com.android.tv.ui.GuidedActionsStylistWithDivider; import com.android.tv.util.SetupUtils; import com.android.tv.util.TvInputManagerHelper; import com.google.common.base.Optional; import dagger.android.AndroidInjection; import dagger.android.ContributesAndroidInjector; import com.android.tv.common.flags.UiFlags; import java.util.ArrayList; import java.util.Collections; import java.util.List; import javax.inject.Inject; /** A fragment for channel source info/setup. */ public class SetupSourcesFragment extends SetupMultiPaneFragment { /** The action category for the actions which is fired from this fragment. */ public static final String ACTION_CATEGORY = "com.android.tv.onboarding.SetupSourcesFragment"; /** An action to open the merchant collection. */ public static final int ACTION_ONLINE_STORE = 1; /** * An action to show the setup activity of TV input. * *

This action is not added to the action list. This is sent outside of the fragment. Use * {@link #ACTION_PARAM_KEY_INPUT_ID} to get the input ID from the parameter. */ public static final int ACTION_SETUP_INPUT = 2; /** * The key for the action parameter which contains the TV input ID. It's used for the action * {@link #ACTION_SETUP_INPUT}. */ public static final String ACTION_PARAM_KEY_INPUT_ID = "input_id"; private static final String SETUP_TRACKER_LABEL = "Setup fragment"; @Override public View onCreateView( LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) { View view = super.onCreateView(inflater, container, savedInstanceState); TvSingletons.getSingletons(getActivity()).getTracker().sendScreenView(SETUP_TRACKER_LABEL); return view; } @Override protected void onEnterTransitionEnd() { SetupGuidedStepFragment f = getContentFragment(); if (f instanceof ContentFragment) { // If the enter transition is canceled quickly, the child fragment can be null because // the fragment is added asynchronously. ((ContentFragment) f).executePendingAction(); } } @Override protected SetupGuidedStepFragment onCreateContentFragment() { SetupGuidedStepFragment f = new ContentFragment(); Bundle arguments = new Bundle(); arguments.putBoolean(SetupGuidedStepFragment.KEY_THREE_PANE, true); f.setArguments(arguments); return f; } @Override protected String getActionCategory() { return ACTION_CATEGORY; } public static class ContentFragment extends SetupGuidedStepFragment { // ACTION_ONLINE_STORE is defined in the outer class. private static final int ACTION_HEADER = 3; private static final int ACTION_INPUT_START = 4; private static final int PENDING_ACTION_NONE = 0; private static final int PENDING_ACTION_INPUT_CHANGED = 1; private static final int PENDING_ACTION_CHANNEL_CHANGED = 2; @Inject TvInputManagerHelper mInputManager; @Inject ChannelDataManager mChannelDataManager; @Inject SetupUtils mSetupUtils; @Inject Optional mBuiltInTunerManagerOptional; @Inject UiFlags mUiFlags; private List mInputs; private int mKnownInputStartIndex; private int mDoneInputStartIndex; private SetupSourcesFragment mParentFragment; private String mNewlyAddedInputId; private int mPendingAction = PENDING_ACTION_NONE; private final TvInputCallback mInputCallback = new TvInputCallback() { @Override public void onInputAdded(String inputId) { handleInputChanged(); } @Override public void onInputRemoved(String inputId) { handleInputChanged(); } @Override public void onInputUpdated(String inputId) { handleInputChanged(); } @Override public void onTvInputInfoUpdated(TvInputInfo inputInfo) { handleInputChanged(); } private void handleInputChanged() { // The actions created while enter transition is running will not be // included in the // fragment transition. if (mParentFragment.isEnterTransitionRunning()) { mPendingAction = PENDING_ACTION_INPUT_CHANGED; return; } buildInputs(); updateActions(); } }; private final ChannelDataManager.Listener mChannelDataManagerListener = new ChannelDataManager.Listener() { @Override public void onLoadFinished() { handleChannelChanged(); } @Override public void onChannelListUpdated() { handleChannelChanged(); } @Override public void onChannelBrowsableChanged() { handleChannelChanged(); } private void handleChannelChanged() { // The actions created while enter transition is running will not be // included in the // fragment transition. if (mParentFragment.isEnterTransitionRunning()) { if (mPendingAction != PENDING_ACTION_INPUT_CHANGED) { mPendingAction = PENDING_ACTION_CHANNEL_CHANGED; } return; } updateActions(); } }; @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); mParentFragment = (SetupSourcesFragment) getParentFragment(); } @Override public void onAttach(Activity activity) { AndroidInjection.inject(this); super.onAttach(activity); buildInputs(); mInputManager.addCallback(mInputCallback); mChannelDataManager.addListener(mChannelDataManagerListener); mParentFragment = (SetupSourcesFragment) getParentFragment(); if (mBuiltInTunerManagerOptional.isPresent()) { mBuiltInTunerManagerOptional .get() .getTunerInputController() .executeNetworkTunerDiscoveryAsyncTask(activity); } } @Override public void onDetach() { mChannelDataManager.removeListener(mChannelDataManagerListener); mInputManager.removeCallback(mInputCallback); super.onDetach(); } @NonNull @Override public Guidance onCreateGuidance(Bundle savedInstanceState) { String title = getString(R.string.setup_sources_text); String description = getString(R.string.setup_sources_description2); return new Guidance(title, description, null, null); } @Override public GuidedActionsStylist onCreateActionsStylist() { return new SetupSourceGuidedActionsStylist(); } @Override public void onCreateActions( @NonNull List actions, Bundle savedInstanceState) { createActionsInternal(actions); } private void buildInputs() { List oldInputs = mInputs; mInputs = mInputManager.getTvInputInfos(true, true); // Get newly installed input ID. if (oldInputs != null) { List newList = new ArrayList<>(mInputs); for (TvInputInfo input : oldInputs) { newList.remove(input); } if (newList.size() > 0 && mSetupUtils.isNewInput(newList.get(0).getId())) { mNewlyAddedInputId = newList.get(0).getId(); } else { mNewlyAddedInputId = null; } } Collections.sort(mInputs, new TvInputNewComparator(mSetupUtils, mInputManager)); mKnownInputStartIndex = 0; mDoneInputStartIndex = 0; for (TvInputInfo input : mInputs) { if (mSetupUtils.isNewInput(input.getId())) { mSetupUtils.markAsKnownInput(input.getId()); ++mKnownInputStartIndex; } if (!mSetupUtils.isSetupDone(input.getId())) { ++mDoneInputStartIndex; } } } private void updateActions() { List actions = new ArrayList<>(); createActionsInternal(actions); setActions(actions); } private void createActionsInternal(List actions) { int newPosition = -1; int position = 0; if (mDoneInputStartIndex > 0) { // Need a "New" category actions.add( new GuidedAction.Builder(getActivity()) .id(ACTION_HEADER) .title(null) .description(getString(R.string.setup_category_new)) .focusable(false) .infoOnly(true) .build()); } for (int i = 0; i < mInputs.size(); ++i) { if (i == mDoneInputStartIndex) { ++position; actions.add( new GuidedAction.Builder(getActivity()) .id(ACTION_HEADER) .title(null) .description(getString(R.string.setup_category_done)) .focusable(false) .infoOnly(true) .build()); } TvInputInfo input = mInputs.get(i); String inputId = input.getId(); String description; int channelCount = mChannelDataManager.getBrowsableChannelCountForInput(inputId); if (mSetupUtils.isSetupDone(inputId) || channelCount > 0) { if (channelCount == 0) { description = getString(R.string.setup_input_no_channels); } else { description = getResources() .getQuantityString( R.plurals.setup_input_channels, channelCount, channelCount); } } else if (i >= mKnownInputStartIndex) { description = getString(R.string.setup_input_setup_now); } else { description = getString(R.string.setup_input_new); } ++position; if (input.getId().equals(mNewlyAddedInputId)) { newPosition = position; } actions.add( new GuidedAction.Builder(getActivity()) .id(ACTION_INPUT_START + i) .title(input.loadLabel(getActivity()).toString()) .description(description) .build()); } if (mInputs.size() > 0) { // Divider ++position; actions.add(GuidedActionsStylistWithDivider.createDividerAction(getContext())); } if (!TextUtils.isEmpty(mUiFlags.moreChannelsUrl())) { // online store action ++position; actions.add( new GuidedAction.Builder(getActivity()) .id(ACTION_ONLINE_STORE) .title(getString(R.string.setup_store_action_title)) .description(getString(R.string.setup_store_action_description)) .icon(R.drawable.ic_app_store) .build()); } if (newPosition != -1) { VerticalGridView gridView = getGuidedActionsStylist().getActionsGridView(); gridView.setSelectedPosition(newPosition); } } @Override protected String getActionCategory() { return ACTION_CATEGORY; } @Override public void onGuidedActionClicked(GuidedAction action) { if (action.getId() == ACTION_ONLINE_STORE) { mParentFragment.onActionClick(ACTION_CATEGORY, (int) action.getId()); return; } int index = (int) action.getId() - ACTION_INPUT_START; if (index >= 0) { TvInputInfo input = mInputs.get(index); Bundle params = new Bundle(); params.putString(ACTION_PARAM_KEY_INPUT_ID, input.getId()); mParentFragment.onActionClick(ACTION_CATEGORY, ACTION_SETUP_INPUT, params); } } void executePendingAction() { switch (mPendingAction) { case PENDING_ACTION_INPUT_CHANGED: buildInputs(); // Fall through case PENDING_ACTION_CHANNEL_CHANGED: updateActions(); break; default: // fall out } mPendingAction = PENDING_ACTION_NONE; } private class SetupSourceGuidedActionsStylist extends GuidedActionsStylistWithDivider { private static final float ALPHA_CATEGORY = 1.0f; private static final float ALPHA_INPUT_DESCRIPTION = 0.5f; @Override public void onBindViewHolder(ViewHolder vh, GuidedAction action) { super.onBindViewHolder(vh, action); TextView descriptionView = vh.getDescriptionView(); if (descriptionView != null) { if (action.getId() == ACTION_HEADER) { descriptionView.setAlpha(ALPHA_CATEGORY); descriptionView.setTextColor( getResources().getColor(R.color.setup_category, null)); descriptionView.setTypeface( Typeface.create(getString(R.string.condensed_font), 0)); } else { descriptionView.setAlpha(ALPHA_INPUT_DESCRIPTION); descriptionView.setTextColor( getResources() .getColor(R.color.common_setup_input_description, null)); descriptionView.setTypeface(Typeface.create(getString(R.string.font), 0)); } } setAccessibilityDelegate(vh, action); } } /** * Exports {@link ContentFragment} for Dagger codegen to create the appropriate injector. */ @dagger.Module public abstract static class Module { @ContributesAndroidInjector abstract ContentFragment contributesContentFragment(); } } }