【Android】Animation动画(二)

Displaying Card Flip Animations

This lesson shows you how to do a card flip animation with custom fragment animations. Card flips animate between views of content by showing an animation that emulates a card flipping over.(说白了就是界面切换,只不过View是通过Fragment实现的)上代码:

  1 /*
  2  * Copyright 2012 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.example.android.animationsdemo;
 18 
 19 import android.app.Activity;
 20 import android.app.Fragment;
 21 import android.app.FragmentManager;
 22 import android.content.Intent;
 23 import android.os.Bundle;
 24 import android.os.Handler;
 25 import android.support.v4.app.NavUtils;
 26 import android.view.LayoutInflater;
 27 import android.view.Menu;
 28 import android.view.MenuItem;
 29 import android.view.View;
 30 import android.view.ViewGroup;
 31 import android.widget.TextView;
 32 
 33 /**
 34  * Demonstrates a "card-flip" animation using custom fragment transactions ({@link
 35  * android.app.FragmentTransaction#setCustomAnimations(int, int)}).
 36  *
 37  * <p>This sample shows an "info" action bar button that shows the back of a "card", rotating the
 38  * front of the card out and the back of the card in. The reverse animation is played when the user
 39  * presses the system Back button or the "photo" action bar button.</p>
 40  */
 41 public class CardFlipActivity extends Activity
 42         implements FragmentManager.OnBackStackChangedListener {
 43     /**
 44      * A handler object, used for deferring UI operations.
 45      */
 46     private Handler mHandler = new Handler();
 47 
 48     /**
 49      * Whether or not we're showing the back of the card (otherwise showing the front).
 50      */
 51     private boolean mShowingBack = false;
 52 
 53     @Override
 54     protected void onCreate(Bundle savedInstanceState) {
 55         super.onCreate(savedInstanceState);
 56         setContentView(R.layout.activity_card_flip);
 57 
 58         if (savedInstanceState == null) {
 59             // If there is no saved instance state, add a fragment representing the
 60             // front of the card to this activity. If there is saved instance state,
 61             // this fragment will have already been added to the activity.
 62             getFragmentManager()
 63                     .beginTransaction()
 64                     .add(R.id.container, new CardFrontFragment())
 65                     .commit();
 66         } else {
 67             mShowingBack = (getFragmentManager().getBackStackEntryCount() > 0);
 68         }
 69 
 70         // Monitor back stack changes to ensure the action bar shows the appropriate
 71         // button (either "photo" or "info").
 72         getFragmentManager().addOnBackStackChangedListener(this);
 73     }
 74 
 75     @Override
 76     public boolean onCreateOptionsMenu(Menu menu) {
 77         super.onCreateOptionsMenu(menu);
 78 
 79         // Add either a "photo" or "finish" button to the action bar, depending on which page
 80         // is currently selected.
 81         MenuItem item = menu.add(Menu.NONE, R.id.action_flip, Menu.NONE,
 82                 mShowingBack
 83                         ? R.string.action_photo
 84                         : R.string.action_info);
 85         item.setIcon(mShowingBack
 86                 ? R.drawable.ic_action_photo
 87                 : R.drawable.ic_action_info);
 88         item.setShowAsAction(MenuItem.SHOW_AS_ACTION_IF_ROOM);
 89         return true;
 90     }
 91 
 92     @Override
 93     public boolean onOptionsItemSelected(MenuItem item) {
 94         switch (item.getItemId()) {
 95             case android.R.id.home:
 96                 // Navigate "up" the demo structure to the launchpad activity.
 97                 // See http://developer.android.com/design/patterns/navigation.html for more.
 98                 NavUtils.navigateUpTo(this, new Intent(this, MainActivity.class));
 99                 return true;
100 
101             case R.id.action_flip:
102                 flipCard();
103                 return true;
104         }
105 
106         return super.onOptionsItemSelected(item);
107     }
108 
109     private void flipCard() {
110         if (mShowingBack) {
111             getFragmentManager().popBackStack();
112             return;
113         }
114 
115         // Flip to the back.
116 
117         mShowingBack = true;
118 
119         // Create and commit a new fragment transaction that adds the fragment for the back of
120         // the card, uses custom animations, and is part of the fragment manager's back stack.
121 
122         getFragmentManager()
123                 .beginTransaction()
124 
125                 // Replace the default fragment animations with animator resources representing
126                 // rotations when switching to the back of the card, as well as animator
127                 // resources representing rotations when flipping back to the front (e.g. when
128                 // the system Back button is pressed).
129                 .setCustomAnimations(
130                         R.animator.card_flip_right_in, R.animator.card_flip_right_out,
131                         R.animator.card_flip_left_in, R.animator.card_flip_left_out)
132 
133                 // Replace any fragments currently in the container view with a fragment
134                 // representing the next page (indicated by the just-incremented currentPage
135                 // variable).
136                 .replace(R.id.container, new CardBackFragment())
137 
138                 // Add this transaction to the back stack, allowing users to press Back
139                 // to get to the front of the card.
140                 .addToBackStack(null)
141 
142                 // Commit the transaction.
143                 .commit();
144 
145         // Defer an invalidation of the options menu (on modern devices, the action bar). This
146         // can't be done immediately because the transaction may not yet be committed. Commits
147         // are asynchronous in that they are posted to the main thread's message loop.
148         mHandler.post(new Runnable() {
149             @Override
150             public void run() {
151                 invalidateOptionsMenu();
152             }
153         });
154     }
155 
156     @Override
157     public void onBackStackChanged() {
158         mShowingBack = (getFragmentManager().getBackStackEntryCount() > 0);
159 
160         // When the back stack changes, invalidate the options menu (action bar).
161         invalidateOptionsMenu();
162     }
163 
164     /**
165      * A fragment representing the front of the card.
166      */
167     public static class CardFrontFragment extends Fragment {
168         public CardFrontFragment() {
169         }
170 
171         @Override
172         public View onCreateView(LayoutInflater inflater, ViewGroup container,
173                 Bundle savedInstanceState) {
174             return inflater.inflate(R.layout.fragment_card_front, container, false);
175         }
176     }
177 
178     /**
179      * A fragment representing the back of the card.
180      */
181     public static class CardBackFragment extends Fragment {
182         public CardBackFragment() {
183         }
184 
185         @Override
186         public View onCreateView(LayoutInflater inflater, ViewGroup container,
187                 Bundle savedInstanceState) {
188             return inflater.inflate(R.layout.fragment_card_back, container, false);
189         }
190     }
191 }

这段代码修改了Fragment从栈中保存和退出的动画,四个动画分别是:

1)card_flip_left_in.xml

 1 <set xmlns:android="http://schemas.android.com/apk/res/android">
 2     <!-- Before rotating, immediately set the alpha to 0. -->
 3     <objectAnimator
 4         android:valueFrom="1.0"
 5         android:valueTo="0.0"
 6         android:propertyName="alpha"
 7         android:duration="0" />
 8 
 9     <!-- Rotate. -->
10     <objectAnimator
11         android:valueFrom="-180"
12         android:valueTo="0"
13         android:propertyName="rotationY"
14         android:interpolator="@android:interpolator/accelerate_decelerate"
15         android:duration="@integer/card_flip_time_full" />
16 
17     <!-- Half-way through the rotation (see startOffset), set the alpha to 1. -->
18     <objectAnimator
19         android:valueFrom="0.0"
20         android:valueTo="1.0"
21         android:propertyName="alpha"
22         android:startOffset="@integer/card_flip_time_half"
23         android:duration="1" />
24 </set>

2)card_flip_left_out.xml

 1 <set xmlns:android="http://schemas.android.com/apk/res/android">
 2     <!-- Rotate. -->
 3     <objectAnimator
 4         android:valueFrom="0"
 5         android:valueTo="180"
 6         android:propertyName="rotationY"
 7         android:interpolator="@android:interpolator/accelerate_decelerate"
 8         android:duration="@integer/card_flip_time_full" />
 9 
10     <!-- Half-way through the rotation (see startOffset), set the alpha to 0. -->
11     <objectAnimator
12         android:valueFrom="1.0"
13         android:valueTo="0.0"
14         android:propertyName="alpha"
15         android:startOffset="@integer/card_flip_time_half"
16         android:duration="1" />
17 </set>

3)card_flip_right_in.xml

 1 <set xmlns:android="http://schemas.android.com/apk/res/android">
 2     <!-- Before rotating, immediately set the alpha to 0. -->
 3     <objectAnimator
 4         android:valueFrom="1.0"
 5         android:valueTo="0.0"
 6         android:propertyName="alpha"
 7         android:duration="0" />
 8 
 9     <!-- Rotate. -->
10     <objectAnimator
11         android:valueFrom="180"
12         android:valueTo="0"
13         android:propertyName="rotationY"
14         android:interpolator="@android:interpolator/accelerate_decelerate"
15         android:duration="@integer/card_flip_time_full" />
16 
17     <!-- Half-way through the rotation (see startOffset), set the alpha to 1. -->
18     <objectAnimator
19         android:valueFrom="0.0"
20         android:valueTo="1.0"
21         android:propertyName="alpha"
22         android:startOffset="@integer/card_flip_time_half"
23         android:duration="1" />

4)card_flip_right_out.xml

 1 <set xmlns:android="http://schemas.android.com/apk/res/android">
 2     <!-- Rotate. -->
 3     <objectAnimator
 4         android:valueFrom="0"
 5         android:valueTo="-180"
 6         android:propertyName="rotationY"
 7         android:interpolator="@android:interpolator/accelerate_decelerate"
 8         android:duration="@integer/card_flip_time_full" />
 9 
10     <!-- Half-way through the rotation (see startOffset), set the alpha to 0. -->
11     <objectAnimator
12         android:valueFrom="1.0"
13         android:valueTo="0.0"
14         android:propertyName="alpha"
15         android:startOffset="@integer/card_flip_time_half"
16         android:duration="1" />
17 </set>

这边有个奇怪的ObjectAnimator,暂不讨论。

关键代码都在这里了,再加上两个View即可。其实很简单,就是修改了Fragment默认的切换动画,然后通过压栈和退栈实现了效果,不过可以仔细研究一下它的动画xml,比如通过设置duration=0来让某种属性(类似alpha)瞬间改变。

(详见:http://developer.android.com/training/animation/cardflip.html#views )

 

 

 

 

 

posted @ 2012-12-08 13:50  大脚印  阅读(1257)  评论(0编辑  收藏  举报