VerticalGridFragment.java revision 497a47f2ab7fdc8490191b02caa9a68f884d9ac0
1/*
2 * Copyright (C) 2014 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except
5 * in compliance with the License. You may obtain a copy of the License at
6 *
7 * http://www.apache.org/licenses/LICENSE-2.0
8 *
9 * Unless required by applicable law or agreed to in writing, software distributed under the License
10 * is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express
11 * or implied. See the License for the specific language governing permissions and limitations under
12 * the License.
13 */
14package android.support.v17.leanback.app;
15
16import android.support.v17.leanback.R;
17import android.support.v17.leanback.widget.Row;
18import android.support.v17.leanback.widget.TitleView;
19import android.support.v17.leanback.widget.VerticalGridPresenter;
20import android.support.v17.leanback.widget.ObjectAdapter;
21import android.support.v17.leanback.widget.OnItemClickedListener;
22import android.support.v17.leanback.widget.OnItemSelectedListener;
23import android.support.v17.leanback.widget.SearchOrbView;
24import android.app.Fragment;
25import android.graphics.drawable.Drawable;
26import android.os.Bundle;
27import android.util.Log;
28import android.view.LayoutInflater;
29import android.view.View;
30import android.view.ViewGroup;
31import android.view.ViewGroup.MarginLayoutParams;
32import android.widget.ImageView;
33import android.widget.TextView;
34
35/**
36 * Leanback fragment for a vertical grid.
37 *
38 * Renders a vertical grid of objects given a {@link VerticalGridPresenter} and
39 * an {@link ObjectAdapter}.
40 */
41public class VerticalGridFragment extends Fragment {
42    private static final String TAG = "VerticalGridFragment";
43    private static boolean DEBUG = false;
44
45    private BrowseFrameLayout mBrowseFrame;
46    private String mTitle;
47    private Drawable mBadgeDrawable;
48    private ObjectAdapter mAdapter;
49    private VerticalGridPresenter mGridPresenter;
50    private VerticalGridPresenter.ViewHolder mGridViewHolder;
51    private OnItemSelectedListener mOnItemSelectedListener;
52    private OnItemClickedListener mOnItemClickedListener;
53    private View.OnClickListener mExternalOnSearchClickedListener;
54    private int mSelectedPosition = -1;
55
56    private TitleView mTitleView;
57    private int mSearchAffordanceColor;
58    private boolean mSearchAffordanceColorSet;
59    private boolean mShowingTitle = true;
60
61    // transition related
62    private static TransitionHelper sTransitionHelper = TransitionHelper.getInstance();
63    private Object mTitleUpTransition;
64    private Object mTitleDownTransition;
65    private Object mSceneWithTitle;
66    private Object mSceneWithoutTitle;
67
68    /**
69     * Sets the badge drawable displayed in the title area.
70     */
71    public void setBadgeDrawable(Drawable drawable) {
72        if (drawable != mBadgeDrawable) {
73            mBadgeDrawable = drawable;
74            if (mTitleView != null) {
75                mTitleView.setBadgeDrawable(drawable);
76            }
77        }
78    }
79
80    /**
81     * Returns the badge drawable.
82     */
83    public Drawable getBadgeDrawable() {
84        return mBadgeDrawable;
85    }
86
87    /**
88     * Sets a title for the fragment.
89     */
90    public void setTitle(String title) {
91        mTitle = title;
92        if (mTitleView != null) {
93            mTitleView.setTitle(mTitle);
94        }
95    }
96
97    /**
98     * Returns the title for the fragment.
99     */
100    public String getTitle() {
101        return mTitle;
102    }
103
104    /**
105     * Sets the grid presenter.
106     */
107    public void setGridPresenter(VerticalGridPresenter gridPresenter) {
108        if (gridPresenter == null) {
109            throw new IllegalArgumentException("Grid presenter may not be null");
110        }
111        mGridPresenter = gridPresenter;
112        mGridPresenter.setOnItemSelectedListener(mRowSelectedListener);
113        if (mOnItemClickedListener != null) {
114            mGridPresenter.setOnItemClickedListener(mOnItemClickedListener);
115        }
116    }
117
118    /**
119     * Returns the grid presenter.
120     */
121    public VerticalGridPresenter getGridPresenter() {
122        return mGridPresenter;
123    }
124
125    /**
126     * Sets the object adapter for the fragment.
127     */
128    public void setAdapter(ObjectAdapter adapter) {
129        mAdapter = adapter;
130        updateAdapter();
131    }
132
133    /**
134     * Returns the object adapter.
135     */
136    public ObjectAdapter getAdapter() {
137        return mAdapter;
138    }
139
140    final private OnItemSelectedListener mRowSelectedListener = new OnItemSelectedListener() {
141        @Override
142        public void onItemSelected(Object item, Row row) {
143            int position = mGridViewHolder.getGridView().getSelectedPosition();
144            if (DEBUG) Log.v(TAG, "row selected position " + position);
145            onRowSelected(position);
146            if (mOnItemSelectedListener != null) {
147                mOnItemSelectedListener.onItemSelected(item, row);
148            }
149        }
150    };
151
152    /**
153     * Sets an item selection listener.
154     */
155    public void setOnItemSelectedListener(OnItemSelectedListener listener) {
156        mOnItemSelectedListener = listener;
157    }
158
159    private void onRowSelected(int position) {
160        if (position != mSelectedPosition) {
161            if (!mGridViewHolder.getGridView().hasPreviousViewInSameRow(position)) {
162                // if has no sibling in front of it,  show title
163                if (!mShowingTitle) {
164                    sTransitionHelper.runTransition(mSceneWithTitle, mTitleDownTransition);
165                    mShowingTitle = true;
166                }
167            } else if (mShowingTitle) {
168                sTransitionHelper.runTransition(mSceneWithoutTitle, mTitleUpTransition);
169                mShowingTitle = false;
170            }
171            mSelectedPosition = position;
172        }
173    }
174
175    /**
176     * Sets an item clicked listener.
177     */
178    public void setOnItemClickedListener(OnItemClickedListener listener) {
179        mOnItemClickedListener = listener;
180        if (mGridPresenter != null) {
181            mGridPresenter.setOnItemClickedListener(mOnItemClickedListener);
182        }
183    }
184
185    /**
186     * Returns the item clicked listener.
187     */
188    public OnItemClickedListener getOnItemClickedListener() {
189        return mOnItemClickedListener;
190    }
191
192    /**
193     * Sets a click listener for the search affordance.
194     *
195     * The presence of a listener will change the visibility of the search affordance in the
196     * title area. When set to non-null the title area will contain a call to search action.
197     *
198     * The listener onClick method will be invoked when the user click on the search action.
199     *
200     * @param listener The listener.
201     */
202    public void setOnSearchClickedListener(View.OnClickListener listener) {
203        mExternalOnSearchClickedListener = listener;
204        if (mTitleView != null) {
205            mTitleView.setOnSearchClickedListener(listener);
206        }
207    }
208
209    /**
210     * Sets the color used to draw the search affordance.
211     */
212    public void setSearchAffordanceColor(int color) {
213        mSearchAffordanceColor = color;
214        mSearchAffordanceColorSet = true;
215        if (mTitleView != null) {
216            mTitleView.setSearchAffordanceColor(mSearchAffordanceColor);
217        }
218    }
219
220    /**
221     * Returns the color used to draw the search affordance.
222     */
223    public int getSearchAffordanceColor() {
224        if (mSearchAffordanceColorSet) {
225            return mSearchAffordanceColor;
226        }
227        if (mTitleView == null) {
228            throw new IllegalStateException("Fragment views not yet created");
229        }
230        return mTitleView.getSearchAffordanceColor();
231    }
232
233
234    private final BrowseFrameLayout.OnFocusSearchListener mOnFocusSearchListener =
235            new BrowseFrameLayout.OnFocusSearchListener() {
236        @Override
237        public View onFocusSearch(View focused, int direction) {
238            if (DEBUG) Log.v(TAG, "onFocusSearch focused " + focused + " + direction " + direction);
239
240            final View searchOrbView = mTitleView.getSearchAffordanceView();
241            if (focused == searchOrbView && (
242                    direction == View.FOCUS_DOWN || direction == View.FOCUS_RIGHT)) {
243                return mGridViewHolder.view;
244
245            } else if (focused != searchOrbView && searchOrbView.getVisibility() == View.VISIBLE
246                    && direction == View.FOCUS_UP) {
247                return searchOrbView;
248
249            } else {
250                return null;
251            }
252        }
253    };
254
255    @Override
256    public View onCreateView(LayoutInflater inflater, ViewGroup container,
257            Bundle savedInstanceState) {
258        ViewGroup root = (ViewGroup) inflater.inflate(R.layout.lb_vertical_grid_fragment,
259                container, false);
260
261        mBrowseFrame = (BrowseFrameLayout) root.findViewById(R.id.browse_frame);
262        mBrowseFrame.setOnFocusSearchListener(mOnFocusSearchListener);
263
264        mTitleView = (TitleView) root.findViewById(R.id.browse_title_group);
265        mTitleView.setBadgeDrawable(mBadgeDrawable);
266        mTitleView.setTitle(mTitle);
267        if (mSearchAffordanceColorSet) {
268            mTitleView.setSearchAffordanceColor(mSearchAffordanceColor);
269        }
270        if (mExternalOnSearchClickedListener != null) {
271            mTitleView.setOnSearchClickedListener(mExternalOnSearchClickedListener);
272        }
273
274        mSceneWithTitle = sTransitionHelper.createScene(root, new Runnable() {
275            @Override
276            public void run() {
277                TitleTransitionHelper.showTitle(mTitleView, true);
278            }
279        });
280        mSceneWithoutTitle = sTransitionHelper.createScene(root, new Runnable() {
281            @Override
282            public void run() {
283                TitleTransitionHelper.showTitle(mTitleView, false);
284            }
285        });
286        mTitleUpTransition = TitleTransitionHelper.createTransitionTitleUp(sTransitionHelper);
287        mTitleDownTransition = TitleTransitionHelper.createTransitionTitleDown(sTransitionHelper);
288        sTransitionHelper.excludeChildren(mTitleUpTransition, R.id.browse_grid_dock, true);
289        sTransitionHelper.excludeChildren(mTitleDownTransition, R.id.browse_grid_dock, true);
290
291        return root;
292    }
293
294    @Override
295    public void onViewCreated(View view, Bundle savedInstanceState) {
296        ViewGroup gridDock = (ViewGroup) view.findViewById(R.id.browse_grid_dock);
297        mGridViewHolder = mGridPresenter.onCreateViewHolder(gridDock);
298        gridDock.addView(mGridViewHolder.view);
299
300        updateAdapter();
301    }
302
303    @Override
304    public void onStart() {
305        super.onStart();
306        mGridViewHolder.getGridView().requestFocus();
307    }
308
309    @Override
310    public void onDestroyView() {
311        super.onDestroyView();
312        mGridViewHolder = null;
313    }
314
315    /**
316     * Sets the selected item position.
317     */
318    public void setSelectedPosition(int position) {
319        mSelectedPosition = position;
320        if(mGridViewHolder != null && mGridViewHolder.getGridView().getAdapter() != null) {
321            mGridViewHolder.getGridView().setSelectedPositionSmooth(position);
322        }
323    }
324
325    private void updateAdapter() {
326        if (mGridViewHolder != null) {
327            mGridPresenter.onBindViewHolder(mGridViewHolder, mAdapter);
328            if (mSelectedPosition != -1) {
329                mGridViewHolder.getGridView().setSelectedPosition(mSelectedPosition);
330            }
331        }
332    }
333}
334