1/*
2 * Copyright (C) 2011 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 com.android.internal.view.menu;
18
19import com.android.internal.view.ActionBarPolicy;
20import com.android.internal.view.menu.ActionMenuView.ActionMenuChildView;
21
22import android.content.Context;
23import android.content.res.Configuration;
24import android.content.res.Resources;
25import android.os.Parcel;
26import android.os.Parcelable;
27import android.util.SparseBooleanArray;
28import android.view.ActionProvider;
29import android.view.MenuItem;
30import android.view.SoundEffectConstants;
31import android.view.View;
32import android.view.View.MeasureSpec;
33import android.view.ViewGroup;
34import android.widget.ImageButton;
35
36import java.util.ArrayList;
37
38/**
39 * MenuPresenter for building action menus as seen in the action bar and action modes.
40 */
41public class ActionMenuPresenter extends BaseMenuPresenter
42        implements ActionProvider.SubUiVisibilityListener {
43    private static final String TAG = "ActionMenuPresenter";
44
45    private View mOverflowButton;
46    private boolean mReserveOverflow;
47    private boolean mReserveOverflowSet;
48    private int mWidthLimit;
49    private int mActionItemWidthLimit;
50    private int mMaxItems;
51    private boolean mMaxItemsSet;
52    private boolean mStrictWidthLimit;
53    private boolean mWidthLimitSet;
54    private boolean mExpandedActionViewsExclusive;
55
56    private int mMinCellSize;
57
58    // Group IDs that have been added as actions - used temporarily, allocated here for reuse.
59    private final SparseBooleanArray mActionButtonGroups = new SparseBooleanArray();
60
61    private View mScrapActionButtonView;
62
63    private OverflowPopup mOverflowPopup;
64    private ActionButtonSubmenu mActionButtonPopup;
65
66    private OpenOverflowRunnable mPostedOpenRunnable;
67
68    final PopupPresenterCallback mPopupPresenterCallback = new PopupPresenterCallback();
69    int mOpenSubMenuId;
70
71    public ActionMenuPresenter(Context context) {
72        super(context, com.android.internal.R.layout.action_menu_layout,
73                com.android.internal.R.layout.action_menu_item_layout);
74    }
75
76    @Override
77    public void initForMenu(Context context, MenuBuilder menu) {
78        super.initForMenu(context, menu);
79
80        final Resources res = context.getResources();
81
82        final ActionBarPolicy abp = ActionBarPolicy.get(context);
83        if (!mReserveOverflowSet) {
84            mReserveOverflow = abp.showsOverflowMenuButton();
85        }
86
87        if (!mWidthLimitSet) {
88            mWidthLimit = abp.getEmbeddedMenuWidthLimit();
89        }
90
91        // Measure for initial configuration
92        if (!mMaxItemsSet) {
93            mMaxItems = abp.getMaxActionButtons();
94        }
95
96        int width = mWidthLimit;
97        if (mReserveOverflow) {
98            if (mOverflowButton == null) {
99                mOverflowButton = new OverflowMenuButton(mSystemContext);
100                final int spec = MeasureSpec.makeMeasureSpec(0, MeasureSpec.UNSPECIFIED);
101                mOverflowButton.measure(spec, spec);
102            }
103            width -= mOverflowButton.getMeasuredWidth();
104        } else {
105            mOverflowButton = null;
106        }
107
108        mActionItemWidthLimit = width;
109
110        mMinCellSize = (int) (ActionMenuView.MIN_CELL_SIZE * res.getDisplayMetrics().density);
111
112        // Drop a scrap view as it may no longer reflect the proper context/config.
113        mScrapActionButtonView = null;
114    }
115
116    public void onConfigurationChanged(Configuration newConfig) {
117        if (!mMaxItemsSet) {
118            mMaxItems = mContext.getResources().getInteger(
119                    com.android.internal.R.integer.max_action_buttons);
120        }
121        if (mMenu != null) {
122            mMenu.onItemsChanged(true);
123        }
124    }
125
126    public void setWidthLimit(int width, boolean strict) {
127        mWidthLimit = width;
128        mStrictWidthLimit = strict;
129        mWidthLimitSet = true;
130    }
131
132    public void setReserveOverflow(boolean reserveOverflow) {
133        mReserveOverflow = reserveOverflow;
134        mReserveOverflowSet = true;
135    }
136
137    public void setItemLimit(int itemCount) {
138        mMaxItems = itemCount;
139        mMaxItemsSet = true;
140    }
141
142    public void setExpandedActionViewsExclusive(boolean isExclusive) {
143        mExpandedActionViewsExclusive = isExclusive;
144    }
145
146    @Override
147    public MenuView getMenuView(ViewGroup root) {
148        MenuView result = super.getMenuView(root);
149        ((ActionMenuView) result).setPresenter(this);
150        return result;
151    }
152
153    @Override
154    public View getItemView(MenuItemImpl item, View convertView, ViewGroup parent) {
155        View actionView = item.getActionView();
156        if (actionView == null || item.hasCollapsibleActionView()) {
157            if (!(convertView instanceof ActionMenuItemView)) {
158                convertView = null;
159            }
160            actionView = super.getItemView(item, convertView, parent);
161        }
162        actionView.setVisibility(item.isActionViewExpanded() ? View.GONE : View.VISIBLE);
163
164        final ActionMenuView menuParent = (ActionMenuView) parent;
165        final ViewGroup.LayoutParams lp = actionView.getLayoutParams();
166        if (!menuParent.checkLayoutParams(lp)) {
167            actionView.setLayoutParams(menuParent.generateLayoutParams(lp));
168        }
169        return actionView;
170    }
171
172    @Override
173    public void bindItemView(MenuItemImpl item, MenuView.ItemView itemView) {
174        itemView.initialize(item, 0);
175
176        final ActionMenuView menuView = (ActionMenuView) mMenuView;
177        ActionMenuItemView actionItemView = (ActionMenuItemView) itemView;
178        actionItemView.setItemInvoker(menuView);
179    }
180
181    @Override
182    public boolean shouldIncludeItem(int childIndex, MenuItemImpl item) {
183        return item.isActionButton();
184    }
185
186    @Override
187    public void updateMenuView(boolean cleared) {
188        super.updateMenuView(cleared);
189
190        if (mMenu != null) {
191            final ArrayList<MenuItemImpl> actionItems = mMenu.getActionItems();
192            final int count = actionItems.size();
193            for (int i = 0; i < count; i++) {
194                final ActionProvider provider = actionItems.get(i).getActionProvider();
195                if (provider != null) {
196                    provider.setSubUiVisibilityListener(this);
197                }
198            }
199        }
200
201        final ArrayList<MenuItemImpl> nonActionItems = mMenu != null ?
202                mMenu.getNonActionItems() : null;
203
204        boolean hasOverflow = false;
205        if (mReserveOverflow && nonActionItems != null) {
206            final int count = nonActionItems.size();
207            if (count == 1) {
208                hasOverflow = !nonActionItems.get(0).isActionViewExpanded();
209            } else {
210                hasOverflow = count > 0;
211            }
212        }
213
214        if (hasOverflow) {
215            if (mOverflowButton == null) {
216                mOverflowButton = new OverflowMenuButton(mSystemContext);
217            }
218            ViewGroup parent = (ViewGroup) mOverflowButton.getParent();
219            if (parent != mMenuView) {
220                if (parent != null) {
221                    parent.removeView(mOverflowButton);
222                }
223                ActionMenuView menuView = (ActionMenuView) mMenuView;
224                menuView.addView(mOverflowButton, menuView.generateOverflowButtonLayoutParams());
225            }
226        } else if (mOverflowButton != null && mOverflowButton.getParent() == mMenuView) {
227            ((ViewGroup) mMenuView).removeView(mOverflowButton);
228        }
229
230        ((ActionMenuView) mMenuView).setOverflowReserved(mReserveOverflow);
231    }
232
233    @Override
234    public boolean filterLeftoverView(ViewGroup parent, int childIndex) {
235        if (parent.getChildAt(childIndex) == mOverflowButton) return false;
236        return super.filterLeftoverView(parent, childIndex);
237    }
238
239    public boolean onSubMenuSelected(SubMenuBuilder subMenu) {
240        if (!subMenu.hasVisibleItems()) return false;
241
242        SubMenuBuilder topSubMenu = subMenu;
243        while (topSubMenu.getParentMenu() != mMenu) {
244            topSubMenu = (SubMenuBuilder) topSubMenu.getParentMenu();
245        }
246        View anchor = findViewForItem(topSubMenu.getItem());
247        if (anchor == null) {
248            if (mOverflowButton == null) return false;
249            anchor = mOverflowButton;
250        }
251
252        mOpenSubMenuId = subMenu.getItem().getItemId();
253        mActionButtonPopup = new ActionButtonSubmenu(mContext, subMenu);
254        mActionButtonPopup.setAnchorView(anchor);
255        mActionButtonPopup.show();
256        super.onSubMenuSelected(subMenu);
257        return true;
258    }
259
260    private View findViewForItem(MenuItem item) {
261        final ViewGroup parent = (ViewGroup) mMenuView;
262        if (parent == null) return null;
263
264        final int count = parent.getChildCount();
265        for (int i = 0; i < count; i++) {
266            final View child = parent.getChildAt(i);
267            if (child instanceof MenuView.ItemView &&
268                    ((MenuView.ItemView) child).getItemData() == item) {
269                return child;
270            }
271        }
272        return null;
273    }
274
275    /**
276     * Display the overflow menu if one is present.
277     * @return true if the overflow menu was shown, false otherwise.
278     */
279    public boolean showOverflowMenu() {
280        if (mReserveOverflow && !isOverflowMenuShowing() && mMenu != null && mMenuView != null &&
281                mPostedOpenRunnable == null && !mMenu.getNonActionItems().isEmpty()) {
282            OverflowPopup popup = new OverflowPopup(mContext, mMenu, mOverflowButton, true);
283            mPostedOpenRunnable = new OpenOverflowRunnable(popup);
284            // Post this for later; we might still need a layout for the anchor to be right.
285            ((View) mMenuView).post(mPostedOpenRunnable);
286
287            // ActionMenuPresenter uses null as a callback argument here
288            // to indicate overflow is opening.
289            super.onSubMenuSelected(null);
290
291            return true;
292        }
293        return false;
294    }
295
296    /**
297     * Hide the overflow menu if it is currently showing.
298     *
299     * @return true if the overflow menu was hidden, false otherwise.
300     */
301    public boolean hideOverflowMenu() {
302        if (mPostedOpenRunnable != null && mMenuView != null) {
303            ((View) mMenuView).removeCallbacks(mPostedOpenRunnable);
304            mPostedOpenRunnable = null;
305            return true;
306        }
307
308        MenuPopupHelper popup = mOverflowPopup;
309        if (popup != null) {
310            popup.dismiss();
311            return true;
312        }
313        return false;
314    }
315
316    /**
317     * Dismiss all popup menus - overflow and submenus.
318     * @return true if popups were dismissed, false otherwise. (This can be because none were open.)
319     */
320    public boolean dismissPopupMenus() {
321        boolean result = hideOverflowMenu();
322        result |= hideSubMenus();
323        return result;
324    }
325
326    /**
327     * Dismiss all submenu popups.
328     *
329     * @return true if popups were dismissed, false otherwise. (This can be because none were open.)
330     */
331    public boolean hideSubMenus() {
332        if (mActionButtonPopup != null) {
333            mActionButtonPopup.dismiss();
334            return true;
335        }
336        return false;
337    }
338
339    /**
340     * @return true if the overflow menu is currently showing
341     */
342    public boolean isOverflowMenuShowing() {
343        return mOverflowPopup != null && mOverflowPopup.isShowing();
344    }
345
346    /**
347     * @return true if space has been reserved in the action menu for an overflow item.
348     */
349    public boolean isOverflowReserved() {
350        return mReserveOverflow;
351    }
352
353    public boolean flagActionItems() {
354        final ArrayList<MenuItemImpl> visibleItems = mMenu.getVisibleItems();
355        final int itemsSize = visibleItems.size();
356        int maxActions = mMaxItems;
357        int widthLimit = mActionItemWidthLimit;
358        final int querySpec = MeasureSpec.makeMeasureSpec(0, MeasureSpec.UNSPECIFIED);
359        final ViewGroup parent = (ViewGroup) mMenuView;
360
361        int requiredItems = 0;
362        int requestedItems = 0;
363        int firstActionWidth = 0;
364        boolean hasOverflow = false;
365        for (int i = 0; i < itemsSize; i++) {
366            MenuItemImpl item = visibleItems.get(i);
367            if (item.requiresActionButton()) {
368                requiredItems++;
369            } else if (item.requestsActionButton()) {
370                requestedItems++;
371            } else {
372                hasOverflow = true;
373            }
374            if (mExpandedActionViewsExclusive && item.isActionViewExpanded()) {
375                // Overflow everything if we have an expanded action view and we're
376                // space constrained.
377                maxActions = 0;
378            }
379        }
380
381        // Reserve a spot for the overflow item if needed.
382        if (mReserveOverflow &&
383                (hasOverflow || requiredItems + requestedItems > maxActions)) {
384            maxActions--;
385        }
386        maxActions -= requiredItems;
387
388        final SparseBooleanArray seenGroups = mActionButtonGroups;
389        seenGroups.clear();
390
391        int cellSize = 0;
392        int cellsRemaining = 0;
393        if (mStrictWidthLimit) {
394            cellsRemaining = widthLimit / mMinCellSize;
395            final int cellSizeRemaining = widthLimit % mMinCellSize;
396            cellSize = mMinCellSize + cellSizeRemaining / cellsRemaining;
397        }
398
399        // Flag as many more requested items as will fit.
400        for (int i = 0; i < itemsSize; i++) {
401            MenuItemImpl item = visibleItems.get(i);
402
403            if (item.requiresActionButton()) {
404                View v = getItemView(item, mScrapActionButtonView, parent);
405                if (mScrapActionButtonView == null) {
406                    mScrapActionButtonView = v;
407                }
408                if (mStrictWidthLimit) {
409                    cellsRemaining -= ActionMenuView.measureChildForCells(v,
410                            cellSize, cellsRemaining, querySpec, 0);
411                } else {
412                    v.measure(querySpec, querySpec);
413                }
414                final int measuredWidth = v.getMeasuredWidth();
415                widthLimit -= measuredWidth;
416                if (firstActionWidth == 0) {
417                    firstActionWidth = measuredWidth;
418                }
419                final int groupId = item.getGroupId();
420                if (groupId != 0) {
421                    seenGroups.put(groupId, true);
422                }
423                item.setIsActionButton(true);
424            } else if (item.requestsActionButton()) {
425                // Items in a group with other items that already have an action slot
426                // can break the max actions rule, but not the width limit.
427                final int groupId = item.getGroupId();
428                final boolean inGroup = seenGroups.get(groupId);
429                boolean isAction = (maxActions > 0 || inGroup) && widthLimit > 0 &&
430                        (!mStrictWidthLimit || cellsRemaining > 0);
431
432                if (isAction) {
433                    View v = getItemView(item, mScrapActionButtonView, parent);
434                    if (mScrapActionButtonView == null) {
435                        mScrapActionButtonView = v;
436                    }
437                    if (mStrictWidthLimit) {
438                        final int cells = ActionMenuView.measureChildForCells(v,
439                                cellSize, cellsRemaining, querySpec, 0);
440                        cellsRemaining -= cells;
441                        if (cells == 0) {
442                            isAction = false;
443                        }
444                    } else {
445                        v.measure(querySpec, querySpec);
446                    }
447                    final int measuredWidth = v.getMeasuredWidth();
448                    widthLimit -= measuredWidth;
449                    if (firstActionWidth == 0) {
450                        firstActionWidth = measuredWidth;
451                    }
452
453                    if (mStrictWidthLimit) {
454                        isAction &= widthLimit >= 0;
455                    } else {
456                        // Did this push the entire first item past the limit?
457                        isAction &= widthLimit + firstActionWidth > 0;
458                    }
459                }
460
461                if (isAction && groupId != 0) {
462                    seenGroups.put(groupId, true);
463                } else if (inGroup) {
464                    // We broke the width limit. Demote the whole group, they all overflow now.
465                    seenGroups.put(groupId, false);
466                    for (int j = 0; j < i; j++) {
467                        MenuItemImpl areYouMyGroupie = visibleItems.get(j);
468                        if (areYouMyGroupie.getGroupId() == groupId) {
469                            // Give back the action slot
470                            if (areYouMyGroupie.isActionButton()) maxActions++;
471                            areYouMyGroupie.setIsActionButton(false);
472                        }
473                    }
474                }
475
476                if (isAction) maxActions--;
477
478                item.setIsActionButton(isAction);
479            } else {
480                // Neither requires nor requests an action button.
481                item.setIsActionButton(false);
482            }
483        }
484        return true;
485    }
486
487    @Override
488    public void onCloseMenu(MenuBuilder menu, boolean allMenusAreClosing) {
489        dismissPopupMenus();
490        super.onCloseMenu(menu, allMenusAreClosing);
491    }
492
493    @Override
494    public Parcelable onSaveInstanceState() {
495        SavedState state = new SavedState();
496        state.openSubMenuId = mOpenSubMenuId;
497        return state;
498    }
499
500    @Override
501    public void onRestoreInstanceState(Parcelable state) {
502        SavedState saved = (SavedState) state;
503        if (saved.openSubMenuId > 0) {
504            MenuItem item = mMenu.findItem(saved.openSubMenuId);
505            if (item != null) {
506                SubMenuBuilder subMenu = (SubMenuBuilder) item.getSubMenu();
507                onSubMenuSelected(subMenu);
508            }
509        }
510    }
511
512    @Override
513    public void onSubUiVisibilityChanged(boolean isVisible) {
514        if (isVisible) {
515            // Not a submenu, but treat it like one.
516            super.onSubMenuSelected(null);
517        } else {
518            mMenu.close(false);
519        }
520    }
521
522    private static class SavedState implements Parcelable {
523        public int openSubMenuId;
524
525        SavedState() {
526        }
527
528        SavedState(Parcel in) {
529            openSubMenuId = in.readInt();
530        }
531
532        @Override
533        public int describeContents() {
534            return 0;
535        }
536
537        @Override
538        public void writeToParcel(Parcel dest, int flags) {
539            dest.writeInt(openSubMenuId);
540        }
541
542        public static final Parcelable.Creator<SavedState> CREATOR
543                = new Parcelable.Creator<SavedState>() {
544            public SavedState createFromParcel(Parcel in) {
545                return new SavedState(in);
546            }
547
548            public SavedState[] newArray(int size) {
549                return new SavedState[size];
550            }
551        };
552    }
553
554    private class OverflowMenuButton extends ImageButton implements ActionMenuChildView {
555        public OverflowMenuButton(Context context) {
556            super(context, null, com.android.internal.R.attr.actionOverflowButtonStyle);
557
558            setClickable(true);
559            setFocusable(true);
560            setVisibility(VISIBLE);
561            setEnabled(true);
562        }
563
564        @Override
565        public boolean performClick() {
566            if (super.performClick()) {
567                return true;
568            }
569
570            playSoundEffect(SoundEffectConstants.CLICK);
571            showOverflowMenu();
572            return true;
573        }
574
575        public boolean needsDividerBefore() {
576            return false;
577        }
578
579        public boolean needsDividerAfter() {
580            return false;
581        }
582
583        @Override
584        protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
585            if (MeasureSpec.getMode(heightMeasureSpec) == MeasureSpec.AT_MOST) {
586                // Fill available height
587                heightMeasureSpec = MeasureSpec.makeMeasureSpec(
588                        MeasureSpec.getSize(heightMeasureSpec), MeasureSpec.EXACTLY);
589            }
590            super.onMeasure(widthMeasureSpec, heightMeasureSpec);
591        }
592    }
593
594    private class OverflowPopup extends MenuPopupHelper {
595        public OverflowPopup(Context context, MenuBuilder menu, View anchorView,
596                boolean overflowOnly) {
597            super(context, menu, anchorView, overflowOnly);
598            setCallback(mPopupPresenterCallback);
599        }
600
601        @Override
602        public void onDismiss() {
603            super.onDismiss();
604            mMenu.close();
605            mOverflowPopup = null;
606        }
607    }
608
609    private class ActionButtonSubmenu extends MenuPopupHelper {
610        private SubMenuBuilder mSubMenu;
611
612        public ActionButtonSubmenu(Context context, SubMenuBuilder subMenu) {
613            super(context, subMenu);
614            mSubMenu = subMenu;
615
616            MenuItemImpl item = (MenuItemImpl) subMenu.getItem();
617            if (!item.isActionButton()) {
618                // Give a reasonable anchor to nested submenus.
619                setAnchorView(mOverflowButton == null ? (View) mMenuView : mOverflowButton);
620            }
621
622            setCallback(mPopupPresenterCallback);
623
624            boolean preserveIconSpacing = false;
625            final int count = subMenu.size();
626            for (int i = 0; i < count; i++) {
627                MenuItem childItem = subMenu.getItem(i);
628                if (childItem.isVisible() && childItem.getIcon() != null) {
629                    preserveIconSpacing = true;
630                    break;
631                }
632            }
633            setForceShowIcon(preserveIconSpacing);
634        }
635
636        @Override
637        public void onDismiss() {
638            super.onDismiss();
639            mActionButtonPopup = null;
640            mOpenSubMenuId = 0;
641        }
642    }
643
644    private class PopupPresenterCallback implements MenuPresenter.Callback {
645
646        @Override
647        public boolean onOpenSubMenu(MenuBuilder subMenu) {
648            if (subMenu == null) return false;
649
650            mOpenSubMenuId = ((SubMenuBuilder) subMenu).getItem().getItemId();
651            return false;
652        }
653
654        @Override
655        public void onCloseMenu(MenuBuilder menu, boolean allMenusAreClosing) {
656            if (menu instanceof SubMenuBuilder) {
657                ((SubMenuBuilder) menu).getRootMenu().close(false);
658            }
659        }
660    }
661
662    private class OpenOverflowRunnable implements Runnable {
663        private OverflowPopup mPopup;
664
665        public OpenOverflowRunnable(OverflowPopup popup) {
666            mPopup = popup;
667        }
668
669        public void run() {
670            mMenu.changeMenuMode();
671            final View menuView = (View) mMenuView;
672            if (menuView != null && menuView.getWindowToken() != null && mPopup.tryShow()) {
673                mOverflowPopup = mPopup;
674            }
675            mPostedOpenRunnable = null;
676        }
677    }
678}
679