NavigationMenuPresenter.java revision ca2f07c9cc83b98d73a18da7177044ee147ffb94
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
17package android.support.design.internal;
18
19import android.content.Context;
20import android.content.res.ColorStateList;
21import android.content.res.Resources;
22import android.graphics.drawable.ColorDrawable;
23import android.graphics.drawable.Drawable;
24import android.os.Bundle;
25import android.os.Parcelable;
26import android.support.annotation.LayoutRes;
27import android.support.annotation.NonNull;
28import android.support.annotation.Nullable;
29import android.support.annotation.StyleRes;
30import android.support.design.R;
31import android.support.v7.internal.view.menu.MenuBuilder;
32import android.support.v7.internal.view.menu.MenuItemImpl;
33import android.support.v7.internal.view.menu.MenuPresenter;
34import android.support.v7.internal.view.menu.MenuView;
35import android.support.v7.internal.view.menu.SubMenuBuilder;
36import android.util.SparseArray;
37import android.view.LayoutInflater;
38import android.view.MenuItem;
39import android.view.SubMenu;
40import android.view.View;
41import android.view.ViewGroup;
42import android.widget.AdapterView;
43import android.widget.BaseAdapter;
44import android.widget.LinearLayout;
45import android.widget.TextView;
46
47import java.util.ArrayList;
48
49/**
50 * @hide
51 */
52public class NavigationMenuPresenter implements MenuPresenter, AdapterView.OnItemClickListener {
53
54    private static final String STATE_HIERARCHY = "android:menu:list";
55    private static final String STATE_ADAPTER = "android:menu:adapter";
56
57    private NavigationMenuView mMenuView;
58    private LinearLayout mHeader;
59
60    private Callback mCallback;
61    private MenuBuilder mMenu;
62    private int mId;
63
64    private NavigationMenuAdapter mAdapter;
65    private LayoutInflater mLayoutInflater;
66
67    private int mTextAppearance;
68    private boolean mTextAppearanceSet;
69    private ColorStateList mTextColor;
70    private ColorStateList mIconTintList;
71    private Drawable mItemBackground;
72
73    /**
74     * Padding to be inserted at the top of the list to avoid the first menu item
75     * from being placed underneath the status bar.
76     */
77    private int mPaddingTopDefault;
78
79    /**
80     * Padding for separators between items
81     */
82    private int mPaddingSeparator;
83
84    @Override
85    public void initForMenu(Context context, MenuBuilder menu) {
86        mLayoutInflater = LayoutInflater.from(context);
87        mMenu = menu;
88        Resources res = context.getResources();
89        mPaddingTopDefault = res.getDimensionPixelOffset(
90                R.dimen.design_navigation_padding_top_default);
91        mPaddingSeparator = res.getDimensionPixelOffset(
92                R.dimen.design_navigation_separator_vertical_padding);
93    }
94
95    @Override
96    public MenuView getMenuView(ViewGroup root) {
97        if (mMenuView == null) {
98            mMenuView = (NavigationMenuView) mLayoutInflater.inflate(
99                    R.layout.design_navigation_menu, root, false);
100            if (mAdapter == null) {
101                mAdapter = new NavigationMenuAdapter();
102            }
103            mHeader = (LinearLayout) mLayoutInflater.inflate(R.layout.design_navigation_item_header,
104                    mMenuView, false);
105            mMenuView.addHeaderView(mHeader, null, false);
106            mMenuView.setAdapter(mAdapter);
107            mMenuView.setOnItemClickListener(this);
108        }
109        return mMenuView;
110    }
111
112    @Override
113    public void updateMenuView(boolean cleared) {
114        if (mAdapter != null) {
115            mAdapter.notifyDataSetChanged();
116        }
117    }
118
119    @Override
120    public void setCallback(Callback cb) {
121        mCallback = cb;
122    }
123
124    @Override
125    public boolean onSubMenuSelected(SubMenuBuilder subMenu) {
126        return false;
127    }
128
129    @Override
130    public void onCloseMenu(MenuBuilder menu, boolean allMenusAreClosing) {
131        if (mCallback != null) {
132            mCallback.onCloseMenu(menu, allMenusAreClosing);
133        }
134    }
135
136    @Override
137    public boolean flagActionItems() {
138        return false;
139    }
140
141    @Override
142    public boolean expandItemActionView(MenuBuilder menu, MenuItemImpl item) {
143        return false;
144    }
145
146    @Override
147    public boolean collapseItemActionView(MenuBuilder menu, MenuItemImpl item) {
148        return false;
149    }
150
151    @Override
152    public int getId() {
153        return mId;
154    }
155
156    public void setId(int id) {
157        mId = id;
158    }
159
160    @Override
161    public Parcelable onSaveInstanceState() {
162        Bundle state = new Bundle();
163        if (mMenuView != null) {
164            SparseArray<Parcelable> hierarchy = new SparseArray<>();
165            mMenuView.saveHierarchyState(hierarchy);
166            state.putSparseParcelableArray(STATE_HIERARCHY, hierarchy);
167        }
168        if (mAdapter != null) {
169            state.putBundle(STATE_ADAPTER, mAdapter.createInstanceState());
170        }
171        return state;
172    }
173
174    @Override
175    public void onRestoreInstanceState(Parcelable parcelable) {
176        Bundle state = (Bundle) parcelable;
177        SparseArray<Parcelable> hierarchy = state.getSparseParcelableArray(STATE_HIERARCHY);
178        if (hierarchy != null) {
179            mMenuView.restoreHierarchyState(hierarchy);
180        }
181        Bundle adapterState = state.getBundle(STATE_ADAPTER);
182        if (adapterState != null) {
183            mAdapter.restoreInstanceState(adapterState);
184        }
185    }
186
187    @Override
188    public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
189        int positionInAdapter = position - mMenuView.getHeaderViewsCount();
190        if (positionInAdapter >= 0) {
191            setUpdateSuspended(true);
192            MenuItemImpl item = mAdapter.getItem(positionInAdapter).getMenuItem();
193            boolean result = mMenu.performItemAction(item, this, 0);
194            if (item != null && item.isCheckable() && result) {
195                mAdapter.setCheckedItem(item);
196            }
197            setUpdateSuspended(false);
198            updateMenuView(false);
199        }
200    }
201
202    public void setCheckedItem(MenuItemImpl item) {
203        mAdapter.setCheckedItem(item);
204    }
205
206    public View inflateHeaderView(@LayoutRes int res) {
207        View view = mLayoutInflater.inflate(res, mHeader, false);
208        addHeaderView(view);
209        return view;
210    }
211
212    public void addHeaderView(@NonNull View view) {
213        mHeader.addView(view);
214        // The padding on top should be cleared.
215        mMenuView.setPadding(0, 0, 0, mMenuView.getPaddingBottom());
216    }
217
218    public void removeHeaderView(@NonNull View view) {
219        mHeader.removeView(view);
220        if (mHeader.getChildCount() == 0) {
221            mMenuView.setPadding(0, mPaddingTopDefault, 0, mMenuView.getPaddingBottom());
222        }
223    }
224
225    @Nullable
226    public ColorStateList getItemTintList() {
227        return mIconTintList;
228    }
229
230    public void setItemIconTintList(@Nullable ColorStateList tint) {
231        mIconTintList = tint;
232        updateMenuView(false);
233    }
234
235    @Nullable
236    public ColorStateList getItemTextColor() {
237        return mTextColor;
238    }
239
240    public void setItemTextColor(@Nullable ColorStateList textColor) {
241        mTextColor = textColor;
242        updateMenuView(false);
243    }
244
245    public void setItemTextAppearance(@StyleRes int resId) {
246        mTextAppearance = resId;
247        mTextAppearanceSet = true;
248        updateMenuView(false);
249    }
250
251    public Drawable getItemBackground() {
252        return mItemBackground;
253    }
254
255    public void setItemBackground(Drawable itemBackground) {
256        mItemBackground = itemBackground;
257    }
258
259    public void setUpdateSuspended(boolean updateSuspended) {
260        if (mAdapter != null) {
261            mAdapter.setUpdateSuspended(updateSuspended);
262        }
263    }
264
265    private class NavigationMenuAdapter extends BaseAdapter {
266
267        private static final String STATE_CHECKED_ITEM = "android:menu:checked";
268
269        private static final String STATE_ACTION_VIEWS = "android:menu:action_views";
270        private static final int VIEW_TYPE_NORMAL = 0;
271        private static final int VIEW_TYPE_SUBHEADER = 1;
272        private static final int VIEW_TYPE_SEPARATOR = 2;
273
274        private final ArrayList<NavigationMenuItem> mItems = new ArrayList<>();
275        private MenuItemImpl mCheckedItem;
276        private ColorDrawable mTransparentIcon;
277        private boolean mUpdateSuspended;
278
279        NavigationMenuAdapter() {
280            prepareMenuItems();
281        }
282
283        @Override
284        public int getCount() {
285            return mItems.size();
286        }
287
288        @Override
289        public NavigationMenuItem getItem(int position) {
290            return mItems.get(position);
291        }
292
293        @Override
294        public long getItemId(int position) {
295            return position;
296        }
297
298        @Override
299        public int getViewTypeCount() {
300            return 3;
301        }
302
303        @Override
304        public int getItemViewType(int position) {
305            NavigationMenuItem item = getItem(position);
306            if (item.isSeparator()) {
307                return VIEW_TYPE_SEPARATOR;
308            } else if (item.getMenuItem().hasSubMenu()) {
309                return VIEW_TYPE_SUBHEADER;
310            } else {
311                return VIEW_TYPE_NORMAL;
312            }
313        }
314
315        @Override
316        public View getView(int position, View convertView, ViewGroup parent) {
317            NavigationMenuItem item = getItem(position);
318            int viewType = getItemViewType(position);
319            switch (viewType) {
320                case VIEW_TYPE_NORMAL:
321                    if (convertView == null) {
322                        convertView = mLayoutInflater.inflate(R.layout.design_navigation_item,
323                                parent, false);
324                    }
325                    NavigationMenuItemView itemView = (NavigationMenuItemView) convertView;
326                    itemView.setIconTintList(mIconTintList);
327                    if (mTextAppearanceSet) {
328                        itemView.setTextAppearance(itemView.getContext(), mTextAppearance);
329                    }
330                    if (mTextColor != null) {
331                        itemView.setTextColor(mTextColor);
332                    }
333                    itemView.setBackgroundDrawable(mItemBackground != null ?
334                            mItemBackground.getConstantState().newDrawable() : null);
335                    itemView.initialize(item.getMenuItem(), 0);
336                    break;
337                case VIEW_TYPE_SUBHEADER:
338                    if (convertView == null) {
339                        convertView = mLayoutInflater.inflate(
340                                R.layout.design_navigation_item_subheader, parent, false);
341                    }
342                    TextView subHeader = (TextView) convertView;
343                    subHeader.setText(item.getMenuItem().getTitle());
344                    break;
345                case VIEW_TYPE_SEPARATOR:
346                    if (convertView == null) {
347                        convertView = mLayoutInflater.inflate(
348                                R.layout.design_navigation_item_separator, parent, false);
349                    }
350                    convertView.setPadding(0, item.getPaddingTop(), 0,
351                            item.getPaddingBottom());
352                    break;
353            }
354            return convertView;
355        }
356
357        @Override
358        public boolean areAllItemsEnabled() {
359            return false;
360        }
361
362        @Override
363        public boolean isEnabled(int position) {
364            return getItem(position).isEnabled();
365        }
366
367        @Override
368        public void notifyDataSetChanged() {
369            prepareMenuItems();
370            super.notifyDataSetChanged();
371        }
372
373        /**
374         * Flattens the visible menu items of {@link #mMenu} into {@link #mItems},
375         * while inserting separators between items when necessary.
376         */
377        private void prepareMenuItems() {
378            if (mUpdateSuspended) {
379                return;
380            }
381            mUpdateSuspended = true;
382            mItems.clear();
383            int currentGroupId = -1;
384            int currentGroupStart = 0;
385            boolean currentGroupHasIcon = false;
386            for (int i = 0, totalSize = mMenu.getVisibleItems().size(); i < totalSize; i++) {
387                MenuItemImpl item = mMenu.getVisibleItems().get(i);
388                if (item.isChecked()) {
389                    setCheckedItem(item);
390                }
391                if (item.isCheckable()) {
392                    item.setExclusiveCheckable(false);
393                }
394                if (item.hasSubMenu()) {
395                    SubMenu subMenu = item.getSubMenu();
396                    if (subMenu.hasVisibleItems()) {
397                        if (i != 0) {
398                            mItems.add(NavigationMenuItem.separator(mPaddingSeparator, 0));
399                        }
400                        mItems.add(NavigationMenuItem.of(item));
401                        boolean subMenuHasIcon = false;
402                        int subMenuStart = mItems.size();
403                        for (int j = 0, size = subMenu.size(); j < size; j++) {
404                            MenuItemImpl subMenuItem = (MenuItemImpl) subMenu.getItem(j);
405                            if (subMenuItem.isVisible()) {
406                                if (!subMenuHasIcon && subMenuItem.getIcon() != null) {
407                                    subMenuHasIcon = true;
408                                }
409                                if (subMenuItem.isCheckable()) {
410                                    subMenuItem.setExclusiveCheckable(false);
411                                }
412                                if (item.isChecked()) {
413                                    setCheckedItem(item);
414                                }
415                                mItems.add(NavigationMenuItem.of(subMenuItem));
416                            }
417                        }
418                        if (subMenuHasIcon) {
419                            appendTransparentIconIfMissing(subMenuStart, mItems.size());
420                        }
421                    }
422                } else {
423                    int groupId = item.getGroupId();
424                    if (groupId != currentGroupId) { // first item in group
425                        currentGroupStart = mItems.size();
426                        currentGroupHasIcon = item.getIcon() != null;
427                        if (i != 0) {
428                            currentGroupStart++;
429                            mItems.add(NavigationMenuItem.separator(
430                                    mPaddingSeparator, mPaddingSeparator));
431                        }
432                    } else if (!currentGroupHasIcon && item.getIcon() != null) {
433                        currentGroupHasIcon = true;
434                        appendTransparentIconIfMissing(currentGroupStart, mItems.size());
435                    }
436                    if (currentGroupHasIcon && item.getIcon() == null) {
437                        item.setIcon(android.R.color.transparent);
438                    }
439                    mItems.add(NavigationMenuItem.of(item));
440                    currentGroupId = groupId;
441                }
442            }
443            mUpdateSuspended = false;
444        }
445
446        private void appendTransparentIconIfMissing(int startIndex, int endIndex) {
447            for (int i = startIndex; i < endIndex; i++) {
448                MenuItem item = mItems.get(i).getMenuItem();
449                if (item.getIcon() == null) {
450                    if (mTransparentIcon == null) {
451                        mTransparentIcon = new ColorDrawable(android.R.color.transparent);
452                    }
453                    item.setIcon(mTransparentIcon);
454                }
455            }
456        }
457
458        public void setCheckedItem(MenuItemImpl checkedItem) {
459            if (mCheckedItem == checkedItem || !checkedItem.isCheckable()) {
460                return;
461            }
462            if (mCheckedItem != null) {
463                mCheckedItem.setChecked(false);
464            }
465            mCheckedItem = checkedItem;
466            checkedItem.setChecked(true);
467        }
468
469        public Bundle createInstanceState() {
470            Bundle state = new Bundle();
471            if (mCheckedItem != null) {
472                state.putInt(STATE_CHECKED_ITEM, mCheckedItem.getItemId());
473            }
474            // Store the states of the action views.
475            SparseArray<ParcelableSparseArray> actionViewStates = new SparseArray<>();
476            for (NavigationMenuItem navigationMenuItem : mItems) {
477                MenuItemImpl item = navigationMenuItem.getMenuItem();
478                View actionView = item != null ? item.getActionView() : null;
479                if (actionView != null) {
480                    ParcelableSparseArray container = new ParcelableSparseArray();
481                    actionView.saveHierarchyState(container);
482                    actionViewStates.put(item.getItemId(), container);
483                }
484            }
485            state.putSparseParcelableArray(STATE_ACTION_VIEWS, actionViewStates);
486            return state;
487        }
488
489        public void restoreInstanceState(Bundle state) {
490            int checkedItem = state.getInt(STATE_CHECKED_ITEM, 0);
491            if (checkedItem != 0) {
492                mUpdateSuspended = true;
493                for (NavigationMenuItem item : mItems) {
494                    MenuItemImpl menuItem = item.getMenuItem();
495                    if (menuItem != null && menuItem.getItemId() == checkedItem) {
496                        setCheckedItem(menuItem);
497                        break;
498                    }
499                }
500                mUpdateSuspended = false;
501                prepareMenuItems();
502            }
503            // Restore the states of the action views.
504            SparseArray<ParcelableSparseArray> actionViewStates = state
505                    .getSparseParcelableArray(STATE_ACTION_VIEWS);
506            for (NavigationMenuItem navigationMenuItem : mItems) {
507                MenuItemImpl item = navigationMenuItem.getMenuItem();
508                View actionView = item != null ? item.getActionView() : null;
509                if (actionView != null) {
510                    actionView.restoreHierarchyState(actionViewStates.get(item.getItemId()));
511                }
512            }
513        }
514
515        public void setUpdateSuspended(boolean updateSuspended) {
516            mUpdateSuspended = updateSuspended;
517        }
518
519    }
520
521    /**
522     * Wraps {@link MenuItemImpl}. This allows separators to be counted as items in list.
523     */
524    private static class NavigationMenuItem {
525
526        /** The item; null for separators */
527        private final MenuItemImpl mMenuItem;
528
529        /** Padding top; used only for separators */
530        private final int mPaddingTop;
531
532        /** Padding bottom; used only for separators */
533        private final int mPaddingBottom;
534
535        private NavigationMenuItem(MenuItemImpl item, int paddingTop, int paddingBottom) {
536            mMenuItem = item;
537            mPaddingTop = paddingTop;
538            mPaddingBottom = paddingBottom;
539        }
540
541        public static NavigationMenuItem of(MenuItemImpl item) {
542            return new NavigationMenuItem(item, 0, 0);
543        }
544
545        public static NavigationMenuItem separator(int paddingTop, int paddingBottom) {
546            return new NavigationMenuItem(null, paddingTop, paddingBottom);
547        }
548
549        public boolean isSeparator() {
550            return mMenuItem == null;
551        }
552
553        public int getPaddingTop() {
554            return mPaddingTop;
555        }
556
557        public int getPaddingBottom() {
558            return mPaddingBottom;
559        }
560
561        public MenuItemImpl getMenuItem() {
562            return mMenuItem;
563        }
564
565        public boolean isEnabled() {
566            // Separators and subheaders never respond to click
567            return mMenuItem != null && !mMenuItem.hasSubMenu() && mMenuItem.isEnabled();
568        }
569
570    }
571
572}
573