Home | History | Annotate | Download | only in ui
      1 /*
      2  * Copyright (C) 2015 The Android Open Source Project
      3  *
      4  * Licensed under the Apache License, Version 2.0 (the "License");
      5  * you may not use this file except in compliance with the License.
      6  * You may obtain a copy of the License at
      7  *
      8  *      http://www.apache.org/licenses/LICENSE-2.0
      9  *
     10  * Unless required by applicable law or agreed to in writing, software
     11  * distributed under the License is distributed on an "AS IS" BASIS,
     12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
     13  * See the License for the specific language governing permissions and
     14  * limitations under the License.
     15  */
     16 
     17 package com.android.tv.ui;
     18 
     19 import android.animation.Animator;
     20 import android.animation.AnimatorInflater;
     21 import android.support.annotation.IntDef;
     22 import android.transition.Fade;
     23 import android.transition.Scene;
     24 import android.transition.Transition;
     25 import android.transition.TransitionInflater;
     26 import android.transition.TransitionManager;
     27 import android.transition.TransitionSet;
     28 import android.transition.TransitionValues;
     29 import android.view.View;
     30 import android.view.ViewGroup;
     31 import android.widget.FrameLayout;
     32 import android.widget.FrameLayout.LayoutParams;
     33 
     34 import com.android.tv.MainActivity;
     35 import com.android.tv.R;
     36 import com.android.tv.data.Channel;
     37 
     38 import java.lang.annotation.Retention;
     39 import java.lang.annotation.RetentionPolicy;
     40 
     41 public class TvTransitionManager extends TransitionManager {
     42     @Retention(RetentionPolicy.SOURCE)
     43     @IntDef({SCENE_TYPE_EMPTY, SCENE_TYPE_CHANNEL_BANNER, SCENE_TYPE_INPUT_BANNER,
     44         SCENE_TYPE_KEYPAD_CHANNEL_SWITCH, SCENE_TYPE_SELECT_INPUT})
     45     public @interface SceneType {}
     46     public static final int SCENE_TYPE_EMPTY = 0;
     47     public static final int SCENE_TYPE_CHANNEL_BANNER = 1;
     48     public static final int SCENE_TYPE_INPUT_BANNER = 2;
     49     public static final int SCENE_TYPE_KEYPAD_CHANNEL_SWITCH = 3;
     50     public static final int SCENE_TYPE_SELECT_INPUT = 4;
     51 
     52     private final MainActivity mMainActivity;
     53     private final ViewGroup mSceneContainer;
     54     private final ChannelBannerView mChannelBannerView;
     55     private final InputBannerView mInputBannerView;
     56     private final KeypadChannelSwitchView mKeypadChannelSwitchView;
     57     private final SelectInputView mSelectInputView;
     58     private final FrameLayout mEmptyView;
     59     private ViewGroup mCurrentSceneView;
     60     private Animator mEnterAnimator;
     61     private Animator mExitAnimator;
     62 
     63     private boolean mInitialized;
     64     private Scene mEmptyScene;
     65     private Scene mChannelBannerScene;
     66     private Scene mInputBannerScene;
     67     private Scene mKeypadChannelSwitchScene;
     68     private Scene mSelectInputScene;
     69     private Scene mCurrentScene;
     70 
     71     private Listener mListener;
     72 
     73     public TvTransitionManager(MainActivity mainActivity, ViewGroup sceneContainer,
     74             ChannelBannerView channelBannerView, InputBannerView inputBannerView,
     75             KeypadChannelSwitchView keypadChannelSwitchView, SelectInputView selectInputView) {
     76         mMainActivity = mainActivity;
     77         mSceneContainer = sceneContainer;
     78         mChannelBannerView = channelBannerView;
     79         mInputBannerView = inputBannerView;
     80         mKeypadChannelSwitchView = keypadChannelSwitchView;
     81         mSelectInputView = selectInputView;
     82         mEmptyView = (FrameLayout) mMainActivity.getLayoutInflater().inflate(
     83                 R.layout.empty_info_banner, sceneContainer, false);
     84         mCurrentSceneView = mEmptyView;
     85     }
     86 
     87     public void goToEmptyScene(boolean withAnimation) {
     88         if (mCurrentScene == mEmptyScene) {
     89             return;
     90         }
     91         initIfNeeded();
     92         if (withAnimation) {
     93             mEmptyView.setAlpha(1.0f);
     94             transitionTo(mEmptyScene);
     95         } else {
     96             TransitionManager.go(mEmptyScene, null);
     97             // When transition is null, transition got stuck without calling endTransitions.
     98             TransitionManager.endTransitions(mEmptyScene.getSceneRoot());
     99             // Since Fade.OUT transition doesn't run, we need to set alpha manually.
    100             mEmptyView.setAlpha(0);
    101         }
    102     }
    103 
    104     public void goToChannelBannerScene() {
    105         initIfNeeded();
    106         Channel channel = mMainActivity.getCurrentChannel();
    107         if (channel != null && channel.isPassthrough()) {
    108             if (mCurrentScene != mInputBannerScene) {
    109                 // Show the input banner instead.
    110                 LayoutParams lp = (LayoutParams) mInputBannerView.getLayoutParams();
    111                 lp.width = mCurrentScene == mSelectInputScene ? mSelectInputView.getWidth()
    112                         : FrameLayout.LayoutParams.WRAP_CONTENT;
    113                 mInputBannerView.setLayoutParams(lp);
    114                 mInputBannerView.updateLabel();
    115                 transitionTo(mInputBannerScene);
    116             }
    117         } else if (mCurrentScene != mChannelBannerScene) {
    118             transitionTo(mChannelBannerScene);
    119         }
    120     }
    121 
    122     public void goToKeypadChannelSwitchScene() {
    123         initIfNeeded();
    124         if (mCurrentScene != mKeypadChannelSwitchScene) {
    125             transitionTo(mKeypadChannelSwitchScene);
    126         }
    127     }
    128 
    129     public void goToSelectInputScene() {
    130         initIfNeeded();
    131         if (mCurrentScene != mSelectInputScene) {
    132             mSelectInputView.setCurrentChannel(mMainActivity.getCurrentChannel());
    133             transitionTo(mSelectInputScene);
    134         }
    135     }
    136 
    137     public boolean isSceneActive() {
    138         return mCurrentScene != mEmptyScene;
    139     }
    140 
    141     public boolean isKeypadChannelSwitchActive() {
    142         return mCurrentScene != null && mCurrentScene == mKeypadChannelSwitchScene;
    143     }
    144 
    145     public boolean isSelectInputActive() {
    146         return mCurrentScene != null && mCurrentScene == mSelectInputScene;
    147     }
    148 
    149     public void setListener(Listener listener) {
    150         mListener = listener;
    151     }
    152 
    153     public void initIfNeeded() {
    154         if (mInitialized) {
    155             return;
    156         }
    157         mEnterAnimator = AnimatorInflater.loadAnimator(mMainActivity,
    158                 R.animator.channel_banner_enter);
    159         mExitAnimator = AnimatorInflater.loadAnimator(mMainActivity,
    160                 R.animator.channel_banner_exit);
    161 
    162         mEmptyScene = new Scene(mSceneContainer, (View) mEmptyView);
    163         mEmptyScene.setEnterAction(new Runnable() {
    164             @Override
    165             public void run() {
    166                 FrameLayout.LayoutParams emptySceneLayoutParams =
    167                         (FrameLayout.LayoutParams) mEmptyView.getLayoutParams();
    168                 ViewGroup.MarginLayoutParams lp =
    169                         (ViewGroup.MarginLayoutParams) mCurrentSceneView.getLayoutParams();
    170                 emptySceneLayoutParams.topMargin = mCurrentSceneView.getTop();
    171                 emptySceneLayoutParams.setMarginStart(lp.getMarginStart());
    172                 emptySceneLayoutParams.height = mCurrentSceneView.getHeight();
    173                 emptySceneLayoutParams.width = mCurrentSceneView.getWidth();
    174                 mEmptyView.setLayoutParams(emptySceneLayoutParams);
    175                 setCurrentScene(mEmptyScene, mEmptyView);
    176             }
    177         });
    178         mEmptyScene.setExitAction(new Runnable() {
    179             @Override
    180             public void run() {
    181                 removeAllViewsFromOverlay();
    182             }
    183         });
    184 
    185         mChannelBannerScene = buildScene(mSceneContainer, mChannelBannerView);
    186         mInputBannerScene = buildScene(mSceneContainer, mInputBannerView);
    187         mKeypadChannelSwitchScene = buildScene(mSceneContainer, mKeypadChannelSwitchView);
    188         mSelectInputScene = buildScene(mSceneContainer, mSelectInputView);
    189         mCurrentScene = mEmptyScene;
    190 
    191         // Enter transitions
    192         TransitionSet enter = new TransitionSet()
    193                 .addTransition(new SceneTransition(SceneTransition.ENTER))
    194                 .addTransition(new Fade(Fade.IN));
    195         setTransition(mEmptyScene, mChannelBannerScene, enter);
    196         setTransition(mEmptyScene, mInputBannerScene, enter);
    197         setTransition(mEmptyScene, mKeypadChannelSwitchScene, enter);
    198         setTransition(mEmptyScene, mSelectInputScene, enter);
    199 
    200         // Exit transitions
    201         TransitionSet exit = new TransitionSet()
    202                 .addTransition(new SceneTransition(SceneTransition.EXIT))
    203                 .addTransition(new Fade(Fade.OUT));
    204         setTransition(mChannelBannerScene, mEmptyScene, exit);
    205         setTransition(mInputBannerScene, mEmptyScene, exit);
    206         setTransition(mKeypadChannelSwitchScene, mEmptyScene, exit);
    207         setTransition(mSelectInputScene, mEmptyScene, exit);
    208 
    209         // All other possible transitions between scenes
    210         TransitionInflater ti = TransitionInflater.from(mMainActivity);
    211         Transition transition = ti.inflateTransition(R.transition.transition_between_scenes);
    212         setTransition(mChannelBannerScene, mKeypadChannelSwitchScene, transition);
    213         setTransition(mChannelBannerScene, mSelectInputScene, transition);
    214         setTransition(mInputBannerScene, mSelectInputScene, transition);
    215         setTransition(mKeypadChannelSwitchScene, mChannelBannerScene, transition);
    216         setTransition(mKeypadChannelSwitchScene, mSelectInputScene, transition);
    217         setTransition(mSelectInputScene, mChannelBannerScene, transition);
    218         setTransition(mSelectInputScene, mInputBannerScene, transition);
    219 
    220         mInitialized = true;
    221     }
    222 
    223     /**
    224      * Returns the type of the given scene.
    225      */
    226     @SceneType public int getSceneType(Scene scene) {
    227         if (scene == mChannelBannerScene) {
    228             return SCENE_TYPE_CHANNEL_BANNER;
    229         } else if (scene == mInputBannerScene) {
    230             return SCENE_TYPE_INPUT_BANNER;
    231         } else if (scene == mKeypadChannelSwitchScene) {
    232             return SCENE_TYPE_KEYPAD_CHANNEL_SWITCH;
    233         } else if (scene == mSelectInputScene) {
    234             return SCENE_TYPE_SELECT_INPUT;
    235         }
    236         return SCENE_TYPE_EMPTY;
    237     }
    238 
    239     private void setCurrentScene(Scene scene, ViewGroup sceneView) {
    240         if (mListener != null) {
    241             mListener.onSceneChanged(getSceneType(mCurrentScene), getSceneType(scene));
    242         }
    243         mCurrentScene = scene;
    244         mCurrentSceneView = sceneView;
    245         // TODO: Is this a still valid call?
    246         mMainActivity.updateKeyInputFocus();
    247     }
    248 
    249     public interface TransitionLayout {
    250         // TODO: remove the parameter fromEmptyScene once a bug regarding transition alpha
    251         // is fixed. The bug is that the transition alpha is not reset after the transition is
    252         // canceled.
    253         void onEnterAction(boolean fromEmptyScene);
    254 
    255         void onExitAction();
    256     }
    257 
    258     private Scene buildScene(ViewGroup sceneRoot, final TransitionLayout layout) {
    259         final Scene scene = new Scene(sceneRoot, (View) layout);
    260         scene.setEnterAction(new Runnable() {
    261             @Override
    262             public void run() {
    263                 boolean wasEmptyScene = (mCurrentScene == mEmptyScene);
    264                 setCurrentScene(scene, (ViewGroup) layout);
    265                 layout.onEnterAction(wasEmptyScene);
    266             }
    267         });
    268         scene.setExitAction(new Runnable() {
    269             @Override
    270             public void run() {
    271                 removeAllViewsFromOverlay();
    272                 layout.onExitAction();
    273             }
    274         });
    275         return scene;
    276     }
    277 
    278     private void removeAllViewsFromOverlay() {
    279         // Clean up all the animations which can be still running.
    280         mSceneContainer.getOverlay().remove(mChannelBannerView);
    281         mSceneContainer.getOverlay().remove(mInputBannerView);
    282         mSceneContainer.getOverlay().remove(mKeypadChannelSwitchView);
    283         mSceneContainer.getOverlay().remove(mSelectInputView);
    284     }
    285 
    286     private class SceneTransition extends Transition {
    287         static final int ENTER = 0;
    288         static final int EXIT = 1;
    289 
    290         private final Animator mAnimator;
    291 
    292         SceneTransition(int mode) {
    293             mAnimator = mode == ENTER ? mEnterAnimator : mExitAnimator;
    294         }
    295 
    296         @Override
    297         public void captureStartValues(TransitionValues transitionValues) {
    298         }
    299 
    300         @Override
    301         public void captureEndValues(TransitionValues transitionValues) {
    302         }
    303 
    304         @Override
    305         public Animator createAnimator(
    306                 ViewGroup sceneRoot, TransitionValues startValues, TransitionValues endValues) {
    307             Animator animator = mAnimator.clone();
    308             animator.setTarget(sceneRoot);
    309             animator.addListener(new HardwareLayerAnimatorListenerAdapter(sceneRoot));
    310             return animator;
    311         }
    312     }
    313 
    314     /**
    315      * An interface for notification of the scene transition.
    316      */
    317     public interface Listener {
    318         /**
    319          * Called when the scene changes. This method is called just before the scene transition.
    320          */
    321         void onSceneChanged(@SceneType int fromSceneType, @SceneType int toSceneType);
    322     }
    323 }
    324