1 /* 2 * Copyright (C) 2006 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 android.view; 18 19 import android.annotation.MenuRes; 20 import android.app.Activity; 21 import android.content.Context; 22 import android.content.ContextWrapper; 23 import android.content.res.ColorStateList; 24 import android.content.res.TypedArray; 25 import android.content.res.XmlResourceParser; 26 import android.graphics.PorterDuff; 27 import android.graphics.drawable.Drawable; 28 import android.util.AttributeSet; 29 import android.util.Log; 30 import android.util.Xml; 31 32 import com.android.internal.view.menu.MenuItemImpl; 33 34 import org.xmlpull.v1.XmlPullParser; 35 import org.xmlpull.v1.XmlPullParserException; 36 37 import java.io.IOException; 38 import java.lang.reflect.Constructor; 39 import java.lang.reflect.Method; 40 41 /** 42 * This class is used to instantiate menu XML files into Menu objects. 43 * <p> 44 * For performance reasons, menu inflation relies heavily on pre-processing of 45 * XML files that is done at build time. Therefore, it is not currently possible 46 * to use MenuInflater with an XmlPullParser over a plain XML file at runtime; 47 * it only works with an XmlPullParser returned from a compiled resource (R. 48 * <em>something</em> file.) 49 */ 50 public class MenuInflater { 51 private static final String LOG_TAG = "MenuInflater"; 52 53 /** Menu tag name in XML. */ 54 private static final String XML_MENU = "menu"; 55 56 /** Group tag name in XML. */ 57 private static final String XML_GROUP = "group"; 58 59 /** Item tag name in XML. */ 60 private static final String XML_ITEM = "item"; 61 62 private static final int NO_ID = 0; 63 64 private static final Class<?>[] ACTION_VIEW_CONSTRUCTOR_SIGNATURE = new Class[] {Context.class}; 65 66 private static final Class<?>[] ACTION_PROVIDER_CONSTRUCTOR_SIGNATURE = ACTION_VIEW_CONSTRUCTOR_SIGNATURE; 67 68 private final Object[] mActionViewConstructorArguments; 69 70 private final Object[] mActionProviderConstructorArguments; 71 72 private Context mContext; 73 private Object mRealOwner; 74 75 /** 76 * Constructs a menu inflater. 77 * 78 * @see Activity#getMenuInflater() 79 */ MenuInflater(Context context)80 public MenuInflater(Context context) { 81 mContext = context; 82 mActionViewConstructorArguments = new Object[] {context}; 83 mActionProviderConstructorArguments = mActionViewConstructorArguments; 84 } 85 86 /** 87 * Constructs a menu inflater. 88 * 89 * @see Activity#getMenuInflater() 90 * @hide 91 */ MenuInflater(Context context, Object realOwner)92 public MenuInflater(Context context, Object realOwner) { 93 mContext = context; 94 mRealOwner = realOwner; 95 mActionViewConstructorArguments = new Object[] {context}; 96 mActionProviderConstructorArguments = mActionViewConstructorArguments; 97 } 98 99 /** 100 * Inflate a menu hierarchy from the specified XML resource. Throws 101 * {@link InflateException} if there is an error. 102 * 103 * @param menuRes Resource ID for an XML layout resource to load (e.g., 104 * <code>R.menu.main_activity</code>) 105 * @param menu The Menu to inflate into. The items and submenus will be 106 * added to this Menu. 107 */ inflate(@enuRes int menuRes, Menu menu)108 public void inflate(@MenuRes int menuRes, Menu menu) { 109 XmlResourceParser parser = null; 110 try { 111 parser = mContext.getResources().getLayout(menuRes); 112 AttributeSet attrs = Xml.asAttributeSet(parser); 113 114 parseMenu(parser, attrs, menu); 115 } catch (XmlPullParserException e) { 116 throw new InflateException("Error inflating menu XML", e); 117 } catch (IOException e) { 118 throw new InflateException("Error inflating menu XML", e); 119 } finally { 120 if (parser != null) parser.close(); 121 } 122 } 123 124 /** 125 * Called internally to fill the given menu. If a sub menu is seen, it will 126 * call this recursively. 127 */ parseMenu(XmlPullParser parser, AttributeSet attrs, Menu menu)128 private void parseMenu(XmlPullParser parser, AttributeSet attrs, Menu menu) 129 throws XmlPullParserException, IOException { 130 MenuState menuState = new MenuState(menu); 131 132 int eventType = parser.getEventType(); 133 String tagName; 134 boolean lookingForEndOfUnknownTag = false; 135 String unknownTagName = null; 136 137 // This loop will skip to the menu start tag 138 do { 139 if (eventType == XmlPullParser.START_TAG) { 140 tagName = parser.getName(); 141 if (tagName.equals(XML_MENU)) { 142 // Go to next tag 143 eventType = parser.next(); 144 break; 145 } 146 147 throw new RuntimeException("Expecting menu, got " + tagName); 148 } 149 eventType = parser.next(); 150 } while (eventType != XmlPullParser.END_DOCUMENT); 151 152 boolean reachedEndOfMenu = false; 153 while (!reachedEndOfMenu) { 154 switch (eventType) { 155 case XmlPullParser.START_TAG: 156 if (lookingForEndOfUnknownTag) { 157 break; 158 } 159 160 tagName = parser.getName(); 161 if (tagName.equals(XML_GROUP)) { 162 menuState.readGroup(attrs); 163 } else if (tagName.equals(XML_ITEM)) { 164 menuState.readItem(attrs); 165 } else if (tagName.equals(XML_MENU)) { 166 // A menu start tag denotes a submenu for an item 167 SubMenu subMenu = menuState.addSubMenuItem(); 168 registerMenu(subMenu, attrs); 169 170 // Parse the submenu into returned SubMenu 171 parseMenu(parser, attrs, subMenu); 172 } else { 173 lookingForEndOfUnknownTag = true; 174 unknownTagName = tagName; 175 } 176 break; 177 178 case XmlPullParser.END_TAG: 179 tagName = parser.getName(); 180 if (lookingForEndOfUnknownTag && tagName.equals(unknownTagName)) { 181 lookingForEndOfUnknownTag = false; 182 unknownTagName = null; 183 } else if (tagName.equals(XML_GROUP)) { 184 menuState.resetGroup(); 185 } else if (tagName.equals(XML_ITEM)) { 186 // Add the item if it hasn't been added (if the item was 187 // a submenu, it would have been added already) 188 if (!menuState.hasAddedItem()) { 189 if (menuState.itemActionProvider != null && 190 menuState.itemActionProvider.hasSubMenu()) { 191 registerMenu(menuState.addSubMenuItem(), attrs); 192 } else { 193 registerMenu(menuState.addItem(), attrs); 194 } 195 } 196 } else if (tagName.equals(XML_MENU)) { 197 reachedEndOfMenu = true; 198 } 199 break; 200 201 case XmlPullParser.END_DOCUMENT: 202 throw new RuntimeException("Unexpected end of document"); 203 } 204 205 eventType = parser.next(); 206 } 207 } 208 209 /** 210 * The method is a hook for layoutlib to do its magic. 211 * Nothing is needed outside of LayoutLib. However, it should not be deleted because it 212 * appears to do nothing. 213 */ registerMenu(@uppressWarnings"unused") MenuItem item, @SuppressWarnings("unused") AttributeSet set)214 private void registerMenu(@SuppressWarnings("unused") MenuItem item, 215 @SuppressWarnings("unused") AttributeSet set) { 216 } 217 218 /** 219 * The method is a hook for layoutlib to do its magic. 220 * Nothing is needed outside of LayoutLib. However, it should not be deleted because it 221 * appears to do nothing. 222 */ registerMenu(@uppressWarnings"unused") SubMenu subMenu, @SuppressWarnings("unused") AttributeSet set)223 private void registerMenu(@SuppressWarnings("unused") SubMenu subMenu, 224 @SuppressWarnings("unused") AttributeSet set) { 225 } 226 227 // Needed by layoutlib. getContext()228 /*package*/ Context getContext() { 229 return mContext; 230 } 231 232 private static class InflatedOnMenuItemClickListener 233 implements MenuItem.OnMenuItemClickListener { 234 private static final Class<?>[] PARAM_TYPES = new Class[] { MenuItem.class }; 235 236 private Object mRealOwner; 237 private Method mMethod; 238 InflatedOnMenuItemClickListener(Object realOwner, String methodName)239 public InflatedOnMenuItemClickListener(Object realOwner, String methodName) { 240 mRealOwner = realOwner; 241 Class<?> c = realOwner.getClass(); 242 try { 243 mMethod = c.getMethod(methodName, PARAM_TYPES); 244 } catch (Exception e) { 245 InflateException ex = new InflateException( 246 "Couldn't resolve menu item onClick handler " + methodName + 247 " in class " + c.getName()); 248 ex.initCause(e); 249 throw ex; 250 } 251 } 252 onMenuItemClick(MenuItem item)253 public boolean onMenuItemClick(MenuItem item) { 254 try { 255 if (mMethod.getReturnType() == Boolean.TYPE) { 256 return (Boolean) mMethod.invoke(mRealOwner, item); 257 } else { 258 mMethod.invoke(mRealOwner, item); 259 return true; 260 } 261 } catch (Exception e) { 262 throw new RuntimeException(e); 263 } 264 } 265 } 266 getRealOwner()267 private Object getRealOwner() { 268 if (mRealOwner == null) { 269 mRealOwner = findRealOwner(mContext); 270 } 271 return mRealOwner; 272 } 273 findRealOwner(Object owner)274 private Object findRealOwner(Object owner) { 275 if (owner instanceof Activity) { 276 return owner; 277 } 278 if (owner instanceof ContextWrapper) { 279 return findRealOwner(((ContextWrapper) owner).getBaseContext()); 280 } 281 return owner; 282 } 283 284 /** 285 * State for the current menu. 286 * <p> 287 * Groups can not be nested unless there is another menu (which will have 288 * its state class). 289 */ 290 private class MenuState { 291 private Menu menu; 292 293 /* 294 * Group state is set on items as they are added, allowing an item to 295 * override its group state. (As opposed to set on items at the group end tag.) 296 */ 297 private int groupId; 298 private int groupCategory; 299 private int groupOrder; 300 private int groupCheckable; 301 private boolean groupVisible; 302 private boolean groupEnabled; 303 304 private boolean itemAdded; 305 private int itemId; 306 private int itemCategoryOrder; 307 private CharSequence itemTitle; 308 private CharSequence itemTitleCondensed; 309 private int itemIconResId; 310 private ColorStateList itemIconTintList = null; 311 private PorterDuff.Mode itemIconTintMode = null; 312 private char itemAlphabeticShortcut; 313 private int itemAlphabeticModifiers; 314 private char itemNumericShortcut; 315 private int itemNumericModifiers; 316 /** 317 * Sync to attrs.xml enum: 318 * - 0: none 319 * - 1: all 320 * - 2: exclusive 321 */ 322 private int itemCheckable; 323 private boolean itemChecked; 324 private boolean itemVisible; 325 private boolean itemEnabled; 326 327 /** 328 * Sync to attrs.xml enum, values in MenuItem: 329 * - 0: never 330 * - 1: ifRoom 331 * - 2: always 332 * - -1: Safe sentinel for "no value". 333 */ 334 private int itemShowAsAction; 335 336 private int itemActionViewLayout; 337 private String itemActionViewClassName; 338 private String itemActionProviderClassName; 339 340 private String itemListenerMethodName; 341 342 private ActionProvider itemActionProvider; 343 344 private CharSequence itemContentDescription; 345 private CharSequence itemTooltipText; 346 347 private static final int defaultGroupId = NO_ID; 348 private static final int defaultItemId = NO_ID; 349 private static final int defaultItemCategory = 0; 350 private static final int defaultItemOrder = 0; 351 private static final int defaultItemCheckable = 0; 352 private static final boolean defaultItemChecked = false; 353 private static final boolean defaultItemVisible = true; 354 private static final boolean defaultItemEnabled = true; 355 MenuState(final Menu menu)356 public MenuState(final Menu menu) { 357 this.menu = menu; 358 359 resetGroup(); 360 } 361 resetGroup()362 public void resetGroup() { 363 groupId = defaultGroupId; 364 groupCategory = defaultItemCategory; 365 groupOrder = defaultItemOrder; 366 groupCheckable = defaultItemCheckable; 367 groupVisible = defaultItemVisible; 368 groupEnabled = defaultItemEnabled; 369 } 370 371 /** 372 * Called when the parser is pointing to a group tag. 373 */ readGroup(AttributeSet attrs)374 public void readGroup(AttributeSet attrs) { 375 TypedArray a = mContext.obtainStyledAttributes(attrs, 376 com.android.internal.R.styleable.MenuGroup); 377 378 groupId = a.getResourceId(com.android.internal.R.styleable.MenuGroup_id, defaultGroupId); 379 groupCategory = a.getInt(com.android.internal.R.styleable.MenuGroup_menuCategory, defaultItemCategory); 380 groupOrder = a.getInt(com.android.internal.R.styleable.MenuGroup_orderInCategory, defaultItemOrder); 381 groupCheckable = a.getInt(com.android.internal.R.styleable.MenuGroup_checkableBehavior, defaultItemCheckable); 382 groupVisible = a.getBoolean(com.android.internal.R.styleable.MenuGroup_visible, defaultItemVisible); 383 groupEnabled = a.getBoolean(com.android.internal.R.styleable.MenuGroup_enabled, defaultItemEnabled); 384 385 a.recycle(); 386 } 387 388 /** 389 * Called when the parser is pointing to an item tag. 390 */ readItem(AttributeSet attrs)391 public void readItem(AttributeSet attrs) { 392 TypedArray a = mContext.obtainStyledAttributes(attrs, 393 com.android.internal.R.styleable.MenuItem); 394 395 // Inherit attributes from the group as default value 396 itemId = a.getResourceId(com.android.internal.R.styleable.MenuItem_id, defaultItemId); 397 final int category = a.getInt(com.android.internal.R.styleable.MenuItem_menuCategory, groupCategory); 398 final int order = a.getInt(com.android.internal.R.styleable.MenuItem_orderInCategory, groupOrder); 399 itemCategoryOrder = (category & Menu.CATEGORY_MASK) | (order & Menu.USER_MASK); 400 itemTitle = a.getText(com.android.internal.R.styleable.MenuItem_title); 401 itemTitleCondensed = a.getText(com.android.internal.R.styleable.MenuItem_titleCondensed); 402 itemIconResId = a.getResourceId(com.android.internal.R.styleable.MenuItem_icon, 0); 403 if (a.hasValue(com.android.internal.R.styleable.MenuItem_iconTintMode)) { 404 itemIconTintMode = Drawable.parseTintMode(a.getInt( 405 com.android.internal.R.styleable.MenuItem_iconTintMode, -1), 406 itemIconTintMode); 407 } else { 408 // Reset to null so that it's not carried over to the next item 409 itemIconTintMode = null; 410 } 411 if (a.hasValue(com.android.internal.R.styleable.MenuItem_iconTint)) { 412 itemIconTintList = a.getColorStateList( 413 com.android.internal.R.styleable.MenuItem_iconTint); 414 } else { 415 // Reset to null so that it's not carried over to the next item 416 itemIconTintList = null; 417 } 418 419 itemAlphabeticShortcut = 420 getShortcut(a.getString(com.android.internal.R.styleable.MenuItem_alphabeticShortcut)); 421 itemAlphabeticModifiers = 422 a.getInt(com.android.internal.R.styleable.MenuItem_alphabeticModifiers, 423 KeyEvent.META_CTRL_ON); 424 itemNumericShortcut = 425 getShortcut(a.getString(com.android.internal.R.styleable.MenuItem_numericShortcut)); 426 itemNumericModifiers = 427 a.getInt(com.android.internal.R.styleable.MenuItem_numericModifiers, 428 KeyEvent.META_CTRL_ON); 429 if (a.hasValue(com.android.internal.R.styleable.MenuItem_checkable)) { 430 // Item has attribute checkable, use it 431 itemCheckable = a.getBoolean(com.android.internal.R.styleable.MenuItem_checkable, false) ? 1 : 0; 432 } else { 433 // Item does not have attribute, use the group's (group can have one more state 434 // for checkable that represents the exclusive checkable) 435 itemCheckable = groupCheckable; 436 } 437 itemChecked = a.getBoolean(com.android.internal.R.styleable.MenuItem_checked, defaultItemChecked); 438 itemVisible = a.getBoolean(com.android.internal.R.styleable.MenuItem_visible, groupVisible); 439 itemEnabled = a.getBoolean(com.android.internal.R.styleable.MenuItem_enabled, groupEnabled); 440 itemShowAsAction = a.getInt(com.android.internal.R.styleable.MenuItem_showAsAction, -1); 441 itemListenerMethodName = a.getString(com.android.internal.R.styleable.MenuItem_onClick); 442 itemActionViewLayout = a.getResourceId(com.android.internal.R.styleable.MenuItem_actionLayout, 0); 443 itemActionViewClassName = a.getString(com.android.internal.R.styleable.MenuItem_actionViewClass); 444 itemActionProviderClassName = a.getString(com.android.internal.R.styleable.MenuItem_actionProviderClass); 445 446 final boolean hasActionProvider = itemActionProviderClassName != null; 447 if (hasActionProvider && itemActionViewLayout == 0 && itemActionViewClassName == null) { 448 itemActionProvider = newInstance(itemActionProviderClassName, 449 ACTION_PROVIDER_CONSTRUCTOR_SIGNATURE, 450 mActionProviderConstructorArguments); 451 } else { 452 if (hasActionProvider) { 453 Log.w(LOG_TAG, "Ignoring attribute 'actionProviderClass'." 454 + " Action view already specified."); 455 } 456 itemActionProvider = null; 457 } 458 459 itemContentDescription = 460 a.getText(com.android.internal.R.styleable.MenuItem_contentDescription); 461 itemTooltipText = a.getText(com.android.internal.R.styleable.MenuItem_tooltipText); 462 463 a.recycle(); 464 465 itemAdded = false; 466 } 467 getShortcut(String shortcutString)468 private char getShortcut(String shortcutString) { 469 if (shortcutString == null) { 470 return 0; 471 } else { 472 return shortcutString.charAt(0); 473 } 474 } 475 setItem(MenuItem item)476 private void setItem(MenuItem item) { 477 item.setChecked(itemChecked) 478 .setVisible(itemVisible) 479 .setEnabled(itemEnabled) 480 .setCheckable(itemCheckable >= 1) 481 .setTitleCondensed(itemTitleCondensed) 482 .setIcon(itemIconResId) 483 .setAlphabeticShortcut(itemAlphabeticShortcut, itemAlphabeticModifiers) 484 .setNumericShortcut(itemNumericShortcut, itemNumericModifiers); 485 486 if (itemShowAsAction >= 0) { 487 item.setShowAsAction(itemShowAsAction); 488 } 489 490 if (itemIconTintMode != null) { 491 item.setIconTintMode(itemIconTintMode); 492 } 493 494 if (itemIconTintList != null) { 495 item.setIconTintList(itemIconTintList); 496 } 497 498 if (itemListenerMethodName != null) { 499 if (mContext.isRestricted()) { 500 throw new IllegalStateException("The android:onClick attribute cannot " 501 + "be used within a restricted context"); 502 } 503 item.setOnMenuItemClickListener( 504 new InflatedOnMenuItemClickListener(getRealOwner(), itemListenerMethodName)); 505 } 506 507 if (item instanceof MenuItemImpl) { 508 MenuItemImpl impl = (MenuItemImpl) item; 509 if (itemCheckable >= 2) { 510 impl.setExclusiveCheckable(true); 511 } 512 } 513 514 boolean actionViewSpecified = false; 515 if (itemActionViewClassName != null) { 516 View actionView = (View) newInstance(itemActionViewClassName, 517 ACTION_VIEW_CONSTRUCTOR_SIGNATURE, mActionViewConstructorArguments); 518 item.setActionView(actionView); 519 actionViewSpecified = true; 520 } 521 if (itemActionViewLayout > 0) { 522 if (!actionViewSpecified) { 523 item.setActionView(itemActionViewLayout); 524 actionViewSpecified = true; 525 } else { 526 Log.w(LOG_TAG, "Ignoring attribute 'itemActionViewLayout'." 527 + " Action view already specified."); 528 } 529 } 530 if (itemActionProvider != null) { 531 item.setActionProvider(itemActionProvider); 532 } 533 534 item.setContentDescription(itemContentDescription); 535 item.setTooltipText(itemTooltipText); 536 } 537 addItem()538 public MenuItem addItem() { 539 itemAdded = true; 540 MenuItem item = menu.add(groupId, itemId, itemCategoryOrder, itemTitle); 541 setItem(item); 542 return item; 543 } 544 addSubMenuItem()545 public SubMenu addSubMenuItem() { 546 itemAdded = true; 547 SubMenu subMenu = menu.addSubMenu(groupId, itemId, itemCategoryOrder, itemTitle); 548 setItem(subMenu.getItem()); 549 return subMenu; 550 } 551 hasAddedItem()552 public boolean hasAddedItem() { 553 return itemAdded; 554 } 555 556 @SuppressWarnings("unchecked") newInstance(String className, Class<?>[] constructorSignature, Object[] arguments)557 private <T> T newInstance(String className, Class<?>[] constructorSignature, 558 Object[] arguments) { 559 try { 560 Class<?> clazz = mContext.getClassLoader().loadClass(className); 561 Constructor<?> constructor = clazz.getConstructor(constructorSignature); 562 constructor.setAccessible(true); 563 return (T) constructor.newInstance(arguments); 564 } catch (Exception e) { 565 Log.w(LOG_TAG, "Cannot instantiate class: " + className, e); 566 } 567 return null; 568 } 569 } 570 } 571