ActionMenuItemView.java revision 1ebe1d4ab755d1a852413bb0845f45caa000fed4
1/* 2 * Copyright (C) 2010 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.v7.view.menu; 18 19import android.content.Context; 20import android.content.res.Configuration; 21import android.content.res.Resources; 22import android.content.res.TypedArray; 23import android.graphics.Rect; 24import android.graphics.drawable.Drawable; 25import android.os.Build; 26import android.os.Parcelable; 27import android.support.v4.content.res.ConfigurationHelper; 28import android.support.v4.view.GravityCompat; 29import android.support.v4.view.ViewCompat; 30import android.support.v7.appcompat.R; 31import android.support.v7.widget.ActionMenuView; 32import android.support.v7.widget.AppCompatTextView; 33import android.support.v7.widget.ForwardingListener; 34import android.support.v7.widget.ListPopupWindow; 35import android.text.TextUtils; 36import android.util.AttributeSet; 37import android.util.DisplayMetrics; 38import android.view.Gravity; 39import android.view.MotionEvent; 40import android.view.View; 41import android.widget.Toast; 42 43/** 44 * @hide 45 */ 46public class ActionMenuItemView extends AppCompatTextView 47 implements MenuView.ItemView, View.OnClickListener, View.OnLongClickListener, 48 ActionMenuView.ActionMenuChildView { 49 50 private static final String TAG = "ActionMenuItemView"; 51 52 private MenuItemImpl mItemData; 53 private CharSequence mTitle; 54 private Drawable mIcon; 55 private MenuBuilder.ItemInvoker mItemInvoker; 56 private ForwardingListener mForwardingListener; 57 private PopupCallback mPopupCallback; 58 59 private boolean mAllowTextWithIcon; 60 private boolean mExpandedFormat; 61 private int mMinWidth; 62 private int mSavedPaddingLeft; 63 64 private static final int MAX_ICON_SIZE = 32; // dp 65 private int mMaxIconSize; 66 67 public ActionMenuItemView(Context context) { 68 this(context, null); 69 } 70 71 public ActionMenuItemView(Context context, AttributeSet attrs) { 72 this(context, attrs, 0); 73 } 74 75 public ActionMenuItemView(Context context, AttributeSet attrs, int defStyle) { 76 super(context, attrs, defStyle); 77 final Resources res = context.getResources(); 78 mAllowTextWithIcon = shouldAllowTextWithIcon(); 79 TypedArray a = context.obtainStyledAttributes(attrs, 80 R.styleable.ActionMenuItemView, defStyle, 0); 81 mMinWidth = a.getDimensionPixelSize( 82 R.styleable.ActionMenuItemView_android_minWidth, 0); 83 a.recycle(); 84 85 final float density = res.getDisplayMetrics().density; 86 mMaxIconSize = (int) (MAX_ICON_SIZE * density + 0.5f); 87 88 setOnClickListener(this); 89 setOnLongClickListener(this); 90 91 mSavedPaddingLeft = -1; 92 setSaveEnabled(false); 93 } 94 95 public void onConfigurationChanged(Configuration newConfig) { 96 if (Build.VERSION.SDK_INT >= 8) { 97 super.onConfigurationChanged(newConfig); 98 } 99 100 mAllowTextWithIcon = shouldAllowTextWithIcon(); 101 updateTextButtonVisibility(); 102 } 103 104 /** 105 * Whether action menu items should obey the "withText" showAsAction flag. This may be set to 106 * false for situations where space is extremely limited. --> 107 */ 108 private boolean shouldAllowTextWithIcon() { 109 final Configuration config = getContext().getResources().getConfiguration(); 110 final int widthDp = ConfigurationHelper.getScreenWidthDp(getResources()); 111 final int heightDp = ConfigurationHelper.getScreenHeightDp(getResources()); 112 113 return widthDp >= 480 || (widthDp >= 640 && heightDp >= 480) 114 || config.orientation == Configuration.ORIENTATION_LANDSCAPE; 115 } 116 117 @Override 118 public void setPadding(int l, int t, int r, int b) { 119 mSavedPaddingLeft = l; 120 super.setPadding(l, t, r, b); 121 } 122 123 public MenuItemImpl getItemData() { 124 return mItemData; 125 } 126 127 public void initialize(MenuItemImpl itemData, int menuType) { 128 mItemData = itemData; 129 130 setIcon(itemData.getIcon()); 131 setTitle(itemData.getTitleForItemView(this)); // Title only takes effect if there is no icon 132 setId(itemData.getItemId()); 133 134 setVisibility(itemData.isVisible() ? View.VISIBLE : View.GONE); 135 setEnabled(itemData.isEnabled()); 136 if (itemData.hasSubMenu()) { 137 if (mForwardingListener == null) { 138 mForwardingListener = new ActionMenuItemForwardingListener(); 139 } 140 } 141 } 142 143 @Override 144 public boolean onTouchEvent(MotionEvent e) { 145 if (mItemData.hasSubMenu() && mForwardingListener != null 146 && mForwardingListener.onTouch(this, e)) { 147 return true; 148 } 149 return super.onTouchEvent(e); 150 } 151 152 @Override 153 public void onClick(View v) { 154 if (mItemInvoker != null) { 155 mItemInvoker.invokeItem(mItemData); 156 } 157 } 158 159 public void setItemInvoker(MenuBuilder.ItemInvoker invoker) { 160 mItemInvoker = invoker; 161 } 162 163 public void setPopupCallback(PopupCallback popupCallback) { 164 mPopupCallback = popupCallback; 165 } 166 167 public boolean prefersCondensedTitle() { 168 return true; 169 } 170 171 public void setCheckable(boolean checkable) { 172 // TODO Support checkable action items 173 } 174 175 public void setChecked(boolean checked) { 176 // TODO Support checkable action items 177 } 178 179 public void setExpandedFormat(boolean expandedFormat) { 180 if (mExpandedFormat != expandedFormat) { 181 mExpandedFormat = expandedFormat; 182 if (mItemData != null) { 183 mItemData.actionFormatChanged(); 184 } 185 } 186 } 187 188 private void updateTextButtonVisibility() { 189 boolean visible = !TextUtils.isEmpty(mTitle); 190 visible &= mIcon == null || 191 (mItemData.showsTextAsAction() && (mAllowTextWithIcon || mExpandedFormat)); 192 193 setText(visible ? mTitle : null); 194 } 195 196 public void setIcon(Drawable icon) { 197 mIcon = icon; 198 if (icon != null) { 199 int width = icon.getIntrinsicWidth(); 200 int height = icon.getIntrinsicHeight(); 201 if (width > mMaxIconSize) { 202 final float scale = (float) mMaxIconSize / width; 203 width = mMaxIconSize; 204 height *= scale; 205 } 206 if (height > mMaxIconSize) { 207 final float scale = (float) mMaxIconSize / height; 208 height = mMaxIconSize; 209 width *= scale; 210 } 211 icon.setBounds(0, 0, width, height); 212 } 213 setCompoundDrawables(icon, null, null, null); 214 215 updateTextButtonVisibility(); 216 } 217 218 public boolean hasText() { 219 return !TextUtils.isEmpty(getText()); 220 } 221 222 public void setShortcut(boolean showShortcut, char shortcutKey) { 223 // Action buttons don't show text for shortcut keys. 224 } 225 226 public void setTitle(CharSequence title) { 227 mTitle = title; 228 229 setContentDescription(mTitle); 230 updateTextButtonVisibility(); 231 } 232 233 public boolean showsIcon() { 234 return true; 235 } 236 237 public boolean needsDividerBefore() { 238 return hasText() && mItemData.getIcon() == null; 239 } 240 241 public boolean needsDividerAfter() { 242 return hasText(); 243 } 244 245 @Override 246 public boolean onLongClick(View v) { 247 if (hasText()) { 248 // Don't show the cheat sheet for items that already show text. 249 return false; 250 } 251 252 final int[] screenPos = new int[2]; 253 final Rect displayFrame = new Rect(); 254 getLocationOnScreen(screenPos); 255 getWindowVisibleDisplayFrame(displayFrame); 256 257 final Context context = getContext(); 258 final int width = getWidth(); 259 final int height = getHeight(); 260 final int midy = screenPos[1] + height / 2; 261 int referenceX = screenPos[0] + width / 2; 262 if (ViewCompat.getLayoutDirection(v) == ViewCompat.LAYOUT_DIRECTION_LTR) { 263 final int screenWidth = context.getResources().getDisplayMetrics().widthPixels; 264 referenceX = screenWidth - referenceX; // mirror 265 } 266 Toast cheatSheet = Toast.makeText(context, mItemData.getTitle(), Toast.LENGTH_SHORT); 267 if (midy < displayFrame.height()) { 268 // Show along the top; follow action buttons 269 cheatSheet.setGravity(Gravity.TOP | GravityCompat.END, referenceX, 270 screenPos[1] + height - displayFrame.top); 271 } else { 272 // Show along the bottom center 273 cheatSheet.setGravity(Gravity.BOTTOM | Gravity.CENTER_HORIZONTAL, 0, height); 274 } 275 cheatSheet.show(); 276 return true; 277 } 278 279 @Override 280 protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { 281 final boolean textVisible = hasText(); 282 if (textVisible && mSavedPaddingLeft >= 0) { 283 super.setPadding(mSavedPaddingLeft, getPaddingTop(), 284 getPaddingRight(), getPaddingBottom()); 285 } 286 287 super.onMeasure(widthMeasureSpec, heightMeasureSpec); 288 289 final int widthMode = MeasureSpec.getMode(widthMeasureSpec); 290 final int widthSize = MeasureSpec.getSize(widthMeasureSpec); 291 final int oldMeasuredWidth = getMeasuredWidth(); 292 final int targetWidth = widthMode == MeasureSpec.AT_MOST ? Math.min(widthSize, mMinWidth) 293 : mMinWidth; 294 295 if (widthMode != MeasureSpec.EXACTLY && mMinWidth > 0 && oldMeasuredWidth < targetWidth) { 296 // Remeasure at exactly the minimum width. 297 super.onMeasure(MeasureSpec.makeMeasureSpec(targetWidth, MeasureSpec.EXACTLY), 298 heightMeasureSpec); 299 } 300 301 if (!textVisible && mIcon != null) { 302 // TextView won't center compound drawables in both dimensions without 303 // a little coercion. Pad in to center the icon after we've measured. 304 final int w = getMeasuredWidth(); 305 final int dw = mIcon.getBounds().width(); 306 super.setPadding((w - dw) / 2, getPaddingTop(), getPaddingRight(), getPaddingBottom()); 307 } 308 } 309 310 private class ActionMenuItemForwardingListener extends ForwardingListener { 311 public ActionMenuItemForwardingListener() { 312 super(ActionMenuItemView.this); 313 } 314 315 @Override 316 public ShowableListMenu getPopup() { 317 if (mPopupCallback != null) { 318 return mPopupCallback.getPopup(); 319 } 320 return null; 321 } 322 323 @Override 324 protected boolean onForwardingStarted() { 325 // Call the invoker, then check if the expected popup is showing. 326 if (mItemInvoker != null && mItemInvoker.invokeItem(mItemData)) { 327 final ShowableListMenu popup = getPopup(); 328 return popup != null && popup.isShowing(); 329 } 330 return false; 331 } 332 333 // Do not backport the framework impl here. 334 // The framework's ListPopupWindow uses an animation before performing the item click 335 // after selecting an item. As AppCompat doesn't use an animation, the popup is 336 // dismissed and thus null'ed out before onForwardingStopped() has been called. 337 // This messes up ActionMenuItemView's onForwardingStopped() impl since it will now 338 // return false and make ListPopupWindow think it's still forwarding. 339 } 340 341 @Override 342 public void onRestoreInstanceState(Parcelable state) { 343 // This might get called with the state of ActionView since it shares the same ID with 344 // ActionMenuItemView. Do not restore this state as ActionMenuItemView never saved it. 345 super.onRestoreInstanceState(null); 346 } 347 348 public static abstract class PopupCallback { 349 public abstract ShowableListMenu getPopup(); 350 } 351} 352