AdapterTransition / src / com.example.android.adaptertransition /

AdapterTransitionFragment.java

1
/*
2
 * Copyright 2014 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.adaptertransition;
18
 
19
import android.os.Bundle;
20
import android.support.v4.app.ActivityCompat;
21
import android.support.v4.app.Fragment;
22
import android.transition.AutoTransition;
23
import android.transition.Scene;
24
import android.transition.Transition;
25
import android.transition.TransitionManager;
26
import android.view.LayoutInflater;
27
import android.view.Menu;
28
import android.view.MenuInflater;
29
import android.view.MenuItem;
30
import android.view.View;
31
import android.view.ViewGroup;
32
import android.widget.AbsListView;
33
import android.widget.FrameLayout;
34
import android.widget.GridView;
35
import android.widget.ListView;
36
import android.widget.Toast;
37
 
38
/**
39
 * Main screen for AdapterTransition sample.
40
 */
41
public class AdapterTransitionFragment extends Fragment implements Transition.TransitionListener {
42
 
43
    /**
44
     * Since the transition framework requires all relevant views in a view hierarchy to be marked
45
     * with IDs, we use this ID to mark the root view.
46
     */
47
    private static final int ROOT_ID = 1;
48
 
49
    /**
50
     * A tag for saving state whether the mAbsListView is ListView or GridView.
51
     */
52
    private static final String STATE_IS_LISTVIEW = "is_listview";
53
 
54
    /**
55
     * This is where we place our AdapterView (ListView / GridView).
56
     */
57
    private FrameLayout mContent;
58
 
59
    /**
60
     * This is where we carry out the transition.
61
     */
62
    private FrameLayout mCover;
63
 
64
    /**
65
     * This list shows our contents. It can be ListView or GridView, and we toggle between them
66
     * using the transition framework.
67
     */
68
    private AbsListView mAbsListView;
69
 
70
    /**
71
     * This is our contents.
72
     */
73
    private MeatAdapter mAdapter;
74
 
75
    public static AdapterTransitionFragment newInstance() {
76
        return new AdapterTransitionFragment();
77
    }
78
 
79
    public AdapterTransitionFragment() {
80
    }
81
 
82
    @Override
83
    public void onCreate(Bundle savedInstanceState) {
84
        super.onCreate(savedInstanceState);
85
        setHasOptionsMenu(true);
86
    }
87
 
88
    @Override
89
    public View onCreateView(LayoutInflater inflater, ViewGroup container,
90
                             Bundle savedInstanceState) {
91
        // If savedInstanceState is available, we restore the state whether the list is a ListView
92
        // or a GridView.
93
        boolean isListView;
94
        if (null == savedInstanceState) {
95
            isListView = true;
96
        } else {
97
            isListView = savedInstanceState.getBoolean(STATE_IS_LISTVIEW, true);
98
        }
99
        inflateAbsList(inflater, container, isListView);
100
        return inflater.inflate(R.layout.fragment_adapter_transition, container, false);
101
    }
102
 
103
    @Override
104
    public void onSaveInstanceState(Bundle outState) {
105
        super.onSaveInstanceState(outState);
106
        outState.putBoolean(STATE_IS_LISTVIEW, mAbsListView instanceof ListView);
107
    }
108
 
109
    @Override
110
    public void onViewCreated(View view, Bundle savedInstanceState) {
111
        // Retaining references for FrameLayouts that we use later.
112
        mContent = (FrameLayout) view.findViewById(R.id.content);
113
        mCover = (FrameLayout) view.findViewById(R.id.cover);
114
        // We are attaching the list to the screen here.
115
        mContent.addView(mAbsListView);
116
    }
117
 
118
    @Override
119
    public void onCreateOptionsMenu(Menu menu, MenuInflater inflater) {
120
        inflater.inflate(R.menu.fragment_adapter_transition, menu);
121
    }
122
 
123
    @Override
124
    public void onPrepareOptionsMenu(Menu menu) {
125
        // We change the look of the icon every time the user toggles between list and grid.
126
        MenuItem item = menu.findItem(R.id.action_toggle);
127
        if (null != item) {
128
            if (mAbsListView instanceof ListView) {
129
                item.setIcon(R.drawable.ic_action_grid);
130
                item.setTitle(R.string.show_as_grid);
131
            } else {
132
                item.setIcon(R.drawable.ic_action_list);
133
                item.setTitle(R.string.show_as_list);
134
            }
135
        }
136
    }
137
 
138
    @Override
139
    public boolean onOptionsItemSelected(MenuItem item) {
140
        switch (item.getItemId()) {
141
            case R.id.action_toggle: {
142
                toggle();
143
                return true;
144
            }
145
        }
146
        return false;
147
    }
148
 
149
    @Override
150
    public void onTransitionStart(Transition transition) {
151
    }
152
 
154
    @Override
155
    public void onTransitionEnd(Transition transition) {
156
        // When the transition ends, we remove all the views from the overlay and hide it.
157
        mCover.removeAllViews();
158
        mCover.setVisibility(View.INVISIBLE);
159
    }
161
 
162
    @Override
163
    public void onTransitionCancel(Transition transition) {
164
    }
165
 
166
    @Override
167
    public void onTransitionPause(Transition transition) {
168
    }
169
 
170
    @Override
171
    public void onTransitionResume(Transition transition) {
172
    }
173
 
174
    /**
175
     * Inflate a ListView or a GridView with a corresponding ListAdapter.
176
     *
177
     * @param inflater The LayoutInflater.
178
     * @param container The ViewGroup that contains this AbsListView. The AbsListView won't be
179
     *                  attached to it.
180
     * @param inflateListView Pass true to inflate a ListView, or false to inflate a GridView.
181
     */
182
    private void inflateAbsList(LayoutInflater inflater, ViewGroup container,
183
                                boolean inflateListView) {
184
        if (inflateListView) {
185
            mAbsListView = (AbsListView) inflater.inflate(R.layout.fragment_meat_list,
186
                    container, false);
187
            mAdapter = new MeatAdapter(inflater, R.layout.item_meat_list);
188
        } else {
189
            mAbsListView = (AbsListView) inflater.inflate(R.layout.fragment_meat_grid,
190
                    container, false);
191
            mAdapter = new MeatAdapter(inflater, R.layout.item_meat_grid);
192
        }
193
        mAbsListView.setAdapter(mAdapter);
194
        mAbsListView.setOnItemClickListener(mAdapter);
195
    }
196
 
197
    /**
198
     * Toggle the UI between ListView and GridView.
199
     */
200
    private void toggle() {
201
        // We use mCover as the overlay on which we carry out the transition.
202
        mCover.setVisibility(View.VISIBLE);
203
        // This FrameLayout holds all the visible views in the current list or grid. We use this as
204
        // the starting Scene of the Transition later.
205
        FrameLayout before = copyVisibleViews();
206
        FrameLayout.LayoutParams params = new FrameLayout.LayoutParams(
207
                FrameLayout.LayoutParams.MATCH_PARENT, FrameLayout.LayoutParams.MATCH_PARENT);
208
        mCover.addView(before, params);
209
        // Swap the actual list.
210
        swapAbsListView();
211
        // We also swap the icon for the toggle button.
212
        ActivityCompat.invalidateOptionsMenu(getActivity());
213
        // It is now ready to start the transition.
214
        mAbsListView.post(new Runnable() {
215
            @Override
216
            public void run() {
218
                Scene scene = new Scene(mCover, copyVisibleViews());
219
                Transition transition = new AutoTransition();
220
                transition.addListener(AdapterTransitionFragment.this);
221
                TransitionManager.go(scene, transition);
223
            }
224
        });
225
    }
226
 
227
    /**
228
     * Swap ListView with GridView, or GridView with ListView.
229
     */
230
    private void swapAbsListView() {
231
        // We save the current scrolling position before removing the current list.
232
        int first = mAbsListView.getFirstVisiblePosition();
233
        // If the current list is a GridView, we replace it with a ListView. If it is a ListView,
234
        // a GridView.
235
        LayoutInflater inflater = LayoutInflater.from(getActivity());
236
        inflateAbsList(inflater, (ViewGroup) mAbsListView.getParent(),
237
                mAbsListView instanceof GridView);
238
        mAbsListView.setAdapter(mAdapter);
239
        // We restore the scrolling position here.
240
        mAbsListView.setSelection(first);
241
        // The new list is ready, and we replace the existing one with it.
242
        mContent.removeAllViews();
243
        mContent.addView(mAbsListView);
244
    }
245
 
246
    /**
247
     * Copy all the visible views in the mAbsListView into a new FrameLayout and return it.
248
     *
249
     * @return a FrameLayout with all the visible views inside.
250
     */
251
    private FrameLayout copyVisibleViews() {
252
        // This is the FrameLayout we return afterwards.
253
        FrameLayout layout = new FrameLayout(getActivity());
254
        // The transition framework requires to set ID for all views to be animated.
255
        layout.setId(ROOT_ID);
256
        // We only copy visible views.
257
        int first = mAbsListView.getFirstVisiblePosition();
258
        int index = 0;
259
        while (true) {
260
            // This is one of the views that we copy. Note that the argument for getChildAt is a
261
            // zero-oriented index, and it doesn't usually match with its position in the list.
262
            View source = mAbsListView.getChildAt(index);
263
            if (null == source) {
264
                break;
265
            }
266
            // This is the copy of the original view.
267
            View destination = mAdapter.getView(first + index, null, layout);
268
            assert destination != null;
269
            destination.setId(ROOT_ID + first + index);
270
            FrameLayout.LayoutParams params = new FrameLayout.LayoutParams(
271
                    source.getWidth(), source.getHeight());
272
            params.leftMargin = (int) source.getX();
273
            params.topMargin = (int) source.getY();
274
            layout.addView(destination, params);
275
            ++index;
276
        }
277
        return layout;
278
    }
279
 
280
}