【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 )

浙公网安备 33010602011771号