1 /* 2 * Copyright (C) 2011 The Android Open Source Project 3 * 4 * Licensed under the Eclipse Public License, Version 1.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.eclipse.org/org/documents/epl-v10.php 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 com.android.ide.eclipse.adt.internal.editors.manifest; 18 19 import static com.android.SdkConstants.ANDROID_STYLE_RESOURCE_PREFIX; 20 import static com.android.SdkConstants.CLASS_ACTIVITY; 21 import static com.android.SdkConstants.NS_RESOURCES; 22 import static com.android.xml.AndroidManifest.ATTRIBUTE_ICON; 23 import static com.android.xml.AndroidManifest.ATTRIBUTE_LABEL; 24 import static com.android.xml.AndroidManifest.ATTRIBUTE_MIN_SDK_VERSION; 25 import static com.android.xml.AndroidManifest.ATTRIBUTE_NAME; 26 import static com.android.xml.AndroidManifest.ATTRIBUTE_PACKAGE; 27 import static com.android.xml.AndroidManifest.ATTRIBUTE_TARGET_SDK_VERSION; 28 import static com.android.xml.AndroidManifest.ATTRIBUTE_THEME; 29 import static com.android.xml.AndroidManifest.NODE_ACTIVITY; 30 import static com.android.xml.AndroidManifest.NODE_USES_SDK; 31 import static org.eclipse.jdt.core.search.IJavaSearchConstants.REFERENCES; 32 33 import com.android.annotations.NonNull; 34 import com.android.annotations.Nullable; 35 import com.android.ide.eclipse.adt.AdtPlugin; 36 import com.android.ide.eclipse.adt.internal.project.BaseProjectHelper; 37 import com.android.ide.eclipse.adt.internal.sdk.Sdk; 38 import com.android.ide.eclipse.adt.io.IFolderWrapper; 39 import com.android.io.IAbstractFile; 40 import com.android.io.StreamException; 41 import com.android.resources.ScreenSize; 42 import com.android.sdklib.IAndroidTarget; 43 import com.android.utils.Pair; 44 import com.android.xml.AndroidManifest; 45 46 import org.eclipse.core.resources.IFile; 47 import org.eclipse.core.resources.IProject; 48 import org.eclipse.core.resources.IResource; 49 import org.eclipse.core.resources.IWorkspace; 50 import org.eclipse.core.resources.ResourcesPlugin; 51 import org.eclipse.core.runtime.CoreException; 52 import org.eclipse.core.runtime.IPath; 53 import org.eclipse.core.runtime.NullProgressMonitor; 54 import org.eclipse.core.runtime.OperationCanceledException; 55 import org.eclipse.core.runtime.QualifiedName; 56 import org.eclipse.jdt.core.IField; 57 import org.eclipse.jdt.core.IJavaElement; 58 import org.eclipse.jdt.core.IJavaProject; 59 import org.eclipse.jdt.core.IMethod; 60 import org.eclipse.jdt.core.IPackageFragment; 61 import org.eclipse.jdt.core.IPackageFragmentRoot; 62 import org.eclipse.jdt.core.IType; 63 import org.eclipse.jdt.core.ITypeHierarchy; 64 import org.eclipse.jdt.core.search.IJavaSearchScope; 65 import org.eclipse.jdt.core.search.SearchEngine; 66 import org.eclipse.jdt.core.search.SearchMatch; 67 import org.eclipse.jdt.core.search.SearchParticipant; 68 import org.eclipse.jdt.core.search.SearchPattern; 69 import org.eclipse.jdt.core.search.SearchRequestor; 70 import org.eclipse.jdt.internal.core.BinaryType; 71 import org.eclipse.jface.text.IDocument; 72 import org.eclipse.ui.editors.text.TextFileDocumentProvider; 73 import org.eclipse.ui.texteditor.IDocumentProvider; 74 import org.w3c.dom.Document; 75 import org.w3c.dom.Element; 76 import org.w3c.dom.NodeList; 77 import org.xml.sax.InputSource; 78 import org.xml.sax.SAXException; 79 80 import java.util.ArrayList; 81 import java.util.Collections; 82 import java.util.HashMap; 83 import java.util.LinkedList; 84 import java.util.List; 85 import java.util.Map; 86 import java.util.regex.Matcher; 87 import java.util.regex.Pattern; 88 89 import javax.xml.parsers.DocumentBuilder; 90 import javax.xml.parsers.DocumentBuilderFactory; 91 import javax.xml.xpath.XPathExpressionException; 92 93 /** 94 * Retrieves and caches manifest information such as the themes to be used for 95 * a given activity. 96 * 97 * @see AndroidManifest 98 */ 99 public class ManifestInfo { 100 /** 101 * The maximum number of milliseconds to search for an activity in the codebase when 102 * attempting to associate layouts with activities in 103 * {@link #guessActivity(IFile, String)} 104 */ 105 private static final int SEARCH_TIMEOUT_MS = 3000; 106 107 private final IProject mProject; 108 private String mPackage; 109 private String mManifestTheme; 110 private Map<String, String> mActivityThemes; 111 private IAbstractFile mManifestFile; 112 private long mLastModified; 113 private long mLastChecked; 114 private String mMinSdkName; 115 private int mMinSdk; 116 private int mTargetSdk; 117 private String mApplicationIcon; 118 private String mApplicationLabel; 119 120 /** 121 * Qualified name for the per-project non-persistent property storing the 122 * {@link ManifestInfo} for this project 123 */ 124 final static QualifiedName MANIFEST_FINDER = new QualifiedName(AdtPlugin.PLUGIN_ID, 125 "manifest"); //$NON-NLS-1$ 126 127 /** 128 * Constructs an {@link ManifestInfo} for the given project. Don't use this method; 129 * use the {@link #get} factory method instead. 130 * 131 * @param project project to create an {@link ManifestInfo} for 132 */ ManifestInfo(IProject project)133 private ManifestInfo(IProject project) { 134 mProject = project; 135 } 136 137 /** 138 * Clears the cached manifest information. The next get call on one of the 139 * properties will cause the information to be refreshed. 140 */ clear()141 public void clear() { 142 mLastChecked = 0; 143 } 144 145 /** 146 * Returns the {@link ManifestInfo} for the given project 147 * 148 * @param project the project the finder is associated with 149 * @return a {@ManifestInfo} for the given project, never null 150 */ 151 @NonNull get(IProject project)152 public static ManifestInfo get(IProject project) { 153 ManifestInfo finder = null; 154 try { 155 finder = (ManifestInfo) project.getSessionProperty(MANIFEST_FINDER); 156 } catch (CoreException e) { 157 // Not a problem; we will just create a new one 158 } 159 160 if (finder == null) { 161 finder = new ManifestInfo(project); 162 try { 163 project.setSessionProperty(MANIFEST_FINDER, finder); 164 } catch (CoreException e) { 165 AdtPlugin.log(e, "Can't store ManifestInfo"); 166 } 167 } 168 169 return finder; 170 } 171 172 /** 173 * Ensure that the package, theme and activity maps are initialized and up to date 174 * with respect to the manifest file 175 */ sync()176 private void sync() { 177 // Since each of the accessors call sync(), allow a bunch of immediate 178 // accessors to all bypass the file stat() below 179 long now = System.currentTimeMillis(); 180 if (now - mLastChecked < 50 && mManifestFile != null) { 181 return; 182 } 183 mLastChecked = now; 184 185 if (mManifestFile == null) { 186 IFolderWrapper projectFolder = new IFolderWrapper(mProject); 187 mManifestFile = AndroidManifest.getManifest(projectFolder); 188 if (mManifestFile == null) { 189 return; 190 } 191 } 192 193 // Check to see if our data is up to date 194 long fileModified = mManifestFile.getModificationStamp(); 195 if (fileModified == mLastModified) { 196 // Already have up to date data 197 return; 198 } 199 mLastModified = fileModified; 200 201 mActivityThemes = new HashMap<String, String>(); 202 mManifestTheme = null; 203 mTargetSdk = 1; // Default when not specified 204 mMinSdk = 1; // Default when not specified 205 mMinSdkName = "1"; // Default when not specified 206 mPackage = ""; //$NON-NLS-1$ 207 mApplicationIcon = null; 208 mApplicationLabel = null; 209 210 Document document = null; 211 try { 212 DocumentBuilderFactory factory = DocumentBuilderFactory.newInstance(); 213 InputSource is = new InputSource(mManifestFile.getContents()); 214 215 factory.setNamespaceAware(true); 216 factory.setValidating(false); 217 DocumentBuilder builder = factory.newDocumentBuilder(); 218 document = builder.parse(is); 219 220 Element root = document.getDocumentElement(); 221 mPackage = root.getAttribute(ATTRIBUTE_PACKAGE); 222 NodeList activities = document.getElementsByTagName(NODE_ACTIVITY); 223 for (int i = 0, n = activities.getLength(); i < n; i++) { 224 Element activity = (Element) activities.item(i); 225 String theme = activity.getAttributeNS(NS_RESOURCES, ATTRIBUTE_THEME); 226 if (theme != null && theme.length() > 0) { 227 String name = activity.getAttributeNS(NS_RESOURCES, ATTRIBUTE_NAME); 228 if (name.startsWith(".") //$NON-NLS-1$ 229 && mPackage != null && mPackage.length() > 0) { 230 name = mPackage + name; 231 } 232 mActivityThemes.put(name, theme); 233 } 234 } 235 236 NodeList applications = root.getElementsByTagName(AndroidManifest.NODE_APPLICATION); 237 if (applications.getLength() > 0) { 238 assert applications.getLength() == 1; 239 Element application = (Element) applications.item(0); 240 if (application.hasAttributeNS(NS_RESOURCES, ATTRIBUTE_ICON)) { 241 mApplicationIcon = application.getAttributeNS(NS_RESOURCES, ATTRIBUTE_ICON); 242 } 243 if (application.hasAttributeNS(NS_RESOURCES, ATTRIBUTE_LABEL)) { 244 mApplicationLabel = application.getAttributeNS(NS_RESOURCES, ATTRIBUTE_LABEL); 245 } 246 247 String defaultTheme = application.getAttributeNS(NS_RESOURCES, ATTRIBUTE_THEME); 248 if (defaultTheme != null && !defaultTheme.isEmpty()) { 249 // From manifest theme documentation: 250 // "If that attribute is also not set, the default system theme is used." 251 mManifestTheme = defaultTheme; 252 } 253 } 254 255 // Look up target SDK 256 NodeList usesSdks = root.getElementsByTagName(NODE_USES_SDK); 257 if (usesSdks.getLength() > 0) { 258 Element usesSdk = (Element) usesSdks.item(0); 259 mMinSdk = getApiVersion(usesSdk, ATTRIBUTE_MIN_SDK_VERSION, 1); 260 mTargetSdk = getApiVersion(usesSdk, ATTRIBUTE_TARGET_SDK_VERSION, mMinSdk); 261 } 262 263 } catch (SAXException e) { 264 AdtPlugin.log(e, "Malformed manifest"); 265 } catch (Exception e) { 266 AdtPlugin.log(e, "Could not read Manifest data"); 267 } 268 } 269 getApiVersion(Element usesSdk, String attribute, int defaultApiLevel)270 private int getApiVersion(Element usesSdk, String attribute, int defaultApiLevel) { 271 String valueString = null; 272 if (usesSdk.hasAttributeNS(NS_RESOURCES, attribute)) { 273 valueString = usesSdk.getAttributeNS(NS_RESOURCES, attribute); 274 if (attribute.equals(ATTRIBUTE_MIN_SDK_VERSION)) { 275 mMinSdkName = valueString; 276 } 277 } 278 279 if (valueString != null) { 280 int apiLevel = -1; 281 try { 282 apiLevel = Integer.valueOf(valueString); 283 } catch (NumberFormatException e) { 284 // Handle codename 285 if (Sdk.getCurrent() != null) { 286 IAndroidTarget target = Sdk.getCurrent().getTargetFromHashString( 287 "android-" + valueString); //$NON-NLS-1$ 288 if (target != null) { 289 // codename future API level is current api + 1 290 apiLevel = target.getVersion().getApiLevel() + 1; 291 } 292 } 293 } 294 295 return apiLevel; 296 } 297 298 return defaultApiLevel; 299 } 300 301 /** 302 * Returns the default package registered in the Android manifest 303 * 304 * @return the default package registered in the manifest 305 */ 306 @NonNull getPackage()307 public String getPackage() { 308 sync(); 309 return mPackage; 310 } 311 312 /** 313 * Returns a map from activity full class names to the corresponding theme style to be 314 * used 315 * 316 * @return a map from activity fqcn to theme style 317 */ 318 @NonNull getActivityThemes()319 public Map<String, String> getActivityThemes() { 320 sync(); 321 return mActivityThemes; 322 } 323 324 /** 325 * Returns the manifest theme registered on the application, if any 326 * 327 * @return a manifest theme, or null if none was registered 328 */ 329 @Nullable getManifestTheme()330 public String getManifestTheme() { 331 sync(); 332 return mManifestTheme; 333 } 334 335 /** 336 * Returns the default theme for this project, by looking at the manifest default 337 * theme registration, target SDK, rendering target, etc. 338 * 339 * @param renderingTarget the rendering target use to render the theme, or null 340 * @param screenSize the screen size to obtain a default theme for, or null if unknown 341 * @return the theme to use for this project, never null 342 */ 343 @NonNull getDefaultTheme(IAndroidTarget renderingTarget, ScreenSize screenSize)344 public String getDefaultTheme(IAndroidTarget renderingTarget, ScreenSize screenSize) { 345 sync(); 346 347 if (mManifestTheme != null) { 348 return mManifestTheme; 349 } 350 351 int renderingTargetSdk = mTargetSdk; 352 if (renderingTarget != null) { 353 renderingTargetSdk = renderingTarget.getVersion().getApiLevel(); 354 } 355 356 int apiLevel = Math.min(mTargetSdk, renderingTargetSdk); 357 // For now this theme works only on XLARGE screens. When it works for all sizes, 358 // add that new apiLevel to this check. 359 if (apiLevel >= 11 && screenSize == ScreenSize.XLARGE || apiLevel >= 14) { 360 return ANDROID_STYLE_RESOURCE_PREFIX + "Theme.Holo"; //$NON-NLS-1$ 361 } else { 362 return ANDROID_STYLE_RESOURCE_PREFIX + "Theme"; //$NON-NLS-1$ 363 } 364 } 365 366 /** 367 * Returns the application icon, or null 368 * 369 * @return the application icon, or null 370 */ 371 @Nullable getApplicationIcon()372 public String getApplicationIcon() { 373 sync(); 374 return mApplicationIcon; 375 } 376 377 /** 378 * Returns the application label, or null 379 * 380 * @return the application label, or null 381 */ 382 @Nullable getApplicationLabel()383 public String getApplicationLabel() { 384 sync(); 385 return mApplicationLabel; 386 } 387 388 /** 389 * Returns the target SDK version 390 * 391 * @return the target SDK version 392 */ getTargetSdkVersion()393 public int getTargetSdkVersion() { 394 sync(); 395 return mTargetSdk; 396 } 397 398 /** 399 * Returns the minimum SDK version 400 * 401 * @return the minimum SDK version 402 */ getMinSdkVersion()403 public int getMinSdkVersion() { 404 sync(); 405 return mMinSdk; 406 } 407 408 /** 409 * Returns the minimum SDK version name (which may not be a numeric string, e.g. 410 * it could be a codename). It will never be null or empty; if no min sdk version 411 * was specified in the manifest, the return value will be "1". Use 412 * {@link #getMinSdkCodeName()} instead if you want to look up whether there is a code name. 413 * 414 * @return the minimum SDK version 415 */ 416 @NonNull getMinSdkName()417 public String getMinSdkName() { 418 sync(); 419 if (mMinSdkName == null || mMinSdkName.isEmpty()) { 420 mMinSdkName = "1"; //$NON-NLS-1$ 421 } 422 423 return mMinSdkName; 424 } 425 426 /** 427 * Returns the code name used for the minimum SDK version, if any. 428 * 429 * @return the minSdkVersion codename or null 430 */ 431 @Nullable getMinSdkCodeName()432 public String getMinSdkCodeName() { 433 String minSdkName = getMinSdkName(); 434 if (!Character.isDigit(minSdkName.charAt(0))) { 435 return minSdkName; 436 } 437 438 return null; 439 } 440 441 /** 442 * Returns the {@link IPackageFragment} for the package registered in the manifest 443 * 444 * @return the {@link IPackageFragment} for the package registered in the manifest 445 */ 446 @Nullable getPackageFragment()447 public IPackageFragment getPackageFragment() { 448 sync(); 449 try { 450 IJavaProject javaProject = BaseProjectHelper.getJavaProject(mProject); 451 if (javaProject != null) { 452 IPackageFragmentRoot root = ManifestInfo.getSourcePackageRoot(javaProject); 453 if (root != null) { 454 return root.getPackageFragment(mPackage); 455 } 456 } 457 } catch (CoreException e) { 458 AdtPlugin.log(e, null); 459 } 460 461 return null; 462 } 463 464 /** 465 * Returns the activity associated with the given layout file. Makes an educated guess 466 * by peeking at the usages of the R.layout.name field corresponding to the layout and 467 * if it finds a usage. 468 * 469 * @param project the project containing the layout 470 * @param layoutName the layout whose activity we want to look up 471 * @param pkg the package containing activities 472 * @return the activity name 473 */ 474 @Nullable guessActivity(IProject project, String layoutName, String pkg)475 public static String guessActivity(IProject project, String layoutName, String pkg) { 476 List<String> activities = guessActivities(project, layoutName, pkg); 477 if (activities.size() > 0) { 478 return activities.get(0); 479 } else { 480 return null; 481 } 482 } 483 484 /** 485 * Returns the activities associated with the given layout file. Makes an educated guess 486 * by peeking at the usages of the R.layout.name field corresponding to the layout and 487 * if it finds a usage. 488 * 489 * @param project the project containing the layout 490 * @param layoutName the layout whose activity we want to look up 491 * @param pkg the package containing activities 492 * @return the activity name 493 */ 494 @NonNull guessActivities(IProject project, String layoutName, String pkg)495 public static List<String> guessActivities(IProject project, String layoutName, String pkg) { 496 final LinkedList<String> activities = new LinkedList<String>(); 497 SearchRequestor requestor = new SearchRequestor() { 498 @Override 499 public void acceptSearchMatch(SearchMatch match) throws CoreException { 500 Object element = match.getElement(); 501 if (element instanceof IMethod) { 502 IMethod method = (IMethod) element; 503 IType declaringType = method.getDeclaringType(); 504 String fqcn = declaringType.getFullyQualifiedName(); 505 506 if ((declaringType.getSuperclassName() != null && 507 declaringType.getSuperclassName().endsWith("Activity")) //$NON-NLS-1$ 508 || method.getElementName().equals("onCreate")) { //$NON-NLS-1$ 509 activities.addFirst(fqcn); 510 } else { 511 activities.addLast(fqcn); 512 } 513 } 514 } 515 }; 516 try { 517 IJavaProject javaProject = BaseProjectHelper.getJavaProject(project); 518 if (javaProject == null) { 519 return Collections.emptyList(); 520 } 521 // TODO - look around a bit more and see if we can figure out whether the 522 // call if from within a setContentView call! 523 524 // Search for which java classes call setContentView(R.layout.layoutname); 525 String typeFqcn = "R.layout"; //$NON-NLS-1$ 526 if (pkg != null) { 527 typeFqcn = pkg + '.' + typeFqcn; 528 } 529 530 IType type = javaProject.findType(typeFqcn); 531 if (type != null) { 532 IField field = type.getField(layoutName); 533 if (field.exists()) { 534 SearchPattern pattern = SearchPattern.createPattern(field, REFERENCES); 535 try { 536 search(requestor, javaProject, pattern); 537 } catch (OperationCanceledException canceled) { 538 // pass 539 } 540 } 541 } 542 } catch (CoreException e) { 543 AdtPlugin.log(e, null); 544 } 545 546 return activities; 547 } 548 549 /** 550 * Returns all activities found in the given project (including those in libraries, 551 * except for android.jar itself) 552 * 553 * @param project the project 554 * @return a list of activity classes as fully qualified class names 555 */ 556 @SuppressWarnings("restriction") // BinaryType 557 @NonNull getProjectActivities(IProject project)558 public static List<String> getProjectActivities(IProject project) { 559 final List<String> activities = new ArrayList<String>(); 560 try { 561 final IJavaProject javaProject = BaseProjectHelper.getJavaProject(project); 562 if (javaProject != null) { 563 IType[] activityTypes = new IType[0]; 564 IType activityType = javaProject.findType(CLASS_ACTIVITY); 565 if (activityType != null) { 566 ITypeHierarchy hierarchy = 567 activityType.newTypeHierarchy(javaProject, new NullProgressMonitor()); 568 activityTypes = hierarchy.getAllSubtypes(activityType); 569 for (IType type : activityTypes) { 570 if (type instanceof BinaryType && (type.getClassFile() == null 571 || type.getClassFile().getResource() == null)) { 572 continue; 573 } 574 activities.add(type.getFullyQualifiedName()); 575 } 576 } 577 } 578 } catch (CoreException e) { 579 AdtPlugin.log(e, null); 580 } 581 582 return activities; 583 } 584 585 586 /** 587 * Returns the activity associated with the given layout file. 588 * <p> 589 * This is an alternative to {@link #guessActivity(IFile, String)}. Whereas 590 * guessActivity simply looks for references to "R.layout.foo", this method searches 591 * for all usages of Activity#setContentView(int), and for each match it looks up the 592 * corresponding call text (such as "setContentView(R.layout.foo)"). From this it uses 593 * a regexp to pull out "foo" from this, and stores the association that layout "foo" 594 * is associated with the activity class that contained the setContentView call. 595 * <p> 596 * This has two potential advantages: 597 * <ol> 598 * <li>It can be faster. We do the reference search -once-, and we've built a map of 599 * all the layout-to-activity mappings which we can then immediately look up other 600 * layouts for, which is particularly useful at startup when we have to compute the 601 * layout activity associations to populate the theme choosers. 602 * <li>It can be more accurate. Just because an activity references an "R.layout.foo" 603 * field doesn't mean it's setting it as a content view. 604 * </ol> 605 * However, this second advantage is also its chief problem. There are some common 606 * code constructs which means that the associated layout is not explicitly referenced 607 * in a direct setContentView call; on a couple of sample projects I tested I found 608 * patterns like for example "setContentView(v)" where "v" had been computed earlier. 609 * Therefore, for now we're going to stick with the more general approach of just 610 * looking up each field when needed. We're keeping the code around, though statically 611 * compiled out with the "if (false)" construct below in case we revisit this. 612 * 613 * @param layoutFile the layout whose activity we want to look up 614 * @return the activity name 615 */ 616 @SuppressWarnings("all") 617 @Nullable guessActivityBySetContentView(String layoutName)618 public String guessActivityBySetContentView(String layoutName) { 619 if (false) { 620 // These should be fields 621 final Pattern LAYOUT_FIELD_PATTERN = 622 Pattern.compile("R\\.layout\\.([a-z0-9_]+)"); //$NON-NLS-1$ 623 Map<String, String> mUsages = null; 624 625 sync(); 626 if (mUsages == null) { 627 final Map<String, String> usages = new HashMap<String, String>(); 628 mUsages = usages; 629 SearchRequestor requestor = new SearchRequestor() { 630 @Override 631 public void acceptSearchMatch(SearchMatch match) throws CoreException { 632 Object element = match.getElement(); 633 if (element instanceof IMethod) { 634 IMethod method = (IMethod) element; 635 IType declaringType = method.getDeclaringType(); 636 String fqcn = declaringType.getFullyQualifiedName(); 637 IDocumentProvider provider = new TextFileDocumentProvider(); 638 IResource resource = match.getResource(); 639 try { 640 provider.connect(resource); 641 IDocument document = provider.getDocument(resource); 642 if (document != null) { 643 String matchText = document.get(match.getOffset(), 644 match.getLength()); 645 Matcher matcher = LAYOUT_FIELD_PATTERN.matcher(matchText); 646 if (matcher.find()) { 647 usages.put(matcher.group(1), fqcn); 648 } 649 } 650 } catch (Exception e) { 651 AdtPlugin.log(e, "Can't find range information for %1$s", 652 resource.getName()); 653 } finally { 654 provider.disconnect(resource); 655 } 656 } 657 } 658 }; 659 try { 660 IJavaProject javaProject = BaseProjectHelper.getJavaProject(mProject); 661 if (javaProject == null) { 662 return null; 663 } 664 665 // Search for which java classes call setContentView(R.layout.layoutname); 666 String typeFqcn = "R.layout"; //$NON-NLS-1$ 667 if (mPackage != null) { 668 typeFqcn = mPackage + '.' + typeFqcn; 669 } 670 671 IType activityType = javaProject.findType(CLASS_ACTIVITY); 672 if (activityType != null) { 673 IMethod method = activityType.getMethod( 674 "setContentView", new String[] {"I"}); //$NON-NLS-1$ //$NON-NLS-2$ 675 if (method.exists()) { 676 SearchPattern pattern = SearchPattern.createPattern(method, 677 REFERENCES); 678 search(requestor, javaProject, pattern); 679 } 680 } 681 } catch (CoreException e) { 682 AdtPlugin.log(e, null); 683 } 684 } 685 686 return mUsages.get(layoutName); 687 } 688 689 return null; 690 } 691 692 /** 693 * Performs a search using the given pattern, scope and handler. The search will abort 694 * if it takes longer than {@link #SEARCH_TIMEOUT_MS} milliseconds. 695 */ search(SearchRequestor requestor, IJavaProject javaProject, SearchPattern pattern)696 private static void search(SearchRequestor requestor, IJavaProject javaProject, 697 SearchPattern pattern) throws CoreException { 698 // Find the package fragment specified in the manifest; the activities should 699 // live there. 700 IJavaSearchScope scope = createPackageScope(javaProject); 701 702 SearchParticipant[] participants = new SearchParticipant[] { 703 SearchEngine.getDefaultSearchParticipant() 704 }; 705 SearchEngine engine = new SearchEngine(); 706 707 final long searchStart = System.currentTimeMillis(); 708 NullProgressMonitor monitor = new NullProgressMonitor() { 709 private boolean mCancelled; 710 @Override 711 public void internalWorked(double work) { 712 long searchEnd = System.currentTimeMillis(); 713 if (searchEnd - searchStart > SEARCH_TIMEOUT_MS) { 714 mCancelled = true; 715 } 716 } 717 718 @Override 719 public boolean isCanceled() { 720 return mCancelled; 721 } 722 }; 723 engine.search(pattern, participants, scope, requestor, monitor); 724 } 725 726 /** Creates a package search scope for the first package root in the given java project */ createPackageScope(IJavaProject javaProject)727 private static IJavaSearchScope createPackageScope(IJavaProject javaProject) { 728 IPackageFragmentRoot packageRoot = getSourcePackageRoot(javaProject); 729 730 IJavaSearchScope scope; 731 if (packageRoot != null) { 732 IJavaElement[] scopeElements = new IJavaElement[] { packageRoot }; 733 scope = SearchEngine.createJavaSearchScope(scopeElements); 734 } else { 735 scope = SearchEngine.createWorkspaceScope(); 736 } 737 return scope; 738 } 739 740 /** 741 * Returns the first package root for the given java project 742 * 743 * @param javaProject the project to search in 744 * @return the first package root, or null 745 */ 746 @Nullable getSourcePackageRoot(IJavaProject javaProject)747 public static IPackageFragmentRoot getSourcePackageRoot(IJavaProject javaProject) { 748 IPackageFragmentRoot packageRoot = null; 749 List<IPath> sources = BaseProjectHelper.getSourceClasspaths(javaProject); 750 751 IWorkspace workspace = ResourcesPlugin.getWorkspace(); 752 for (IPath path : sources) { 753 IResource firstSource = workspace.getRoot().findMember(path); 754 if (firstSource != null) { 755 packageRoot = javaProject.getPackageFragmentRoot(firstSource); 756 if (packageRoot != null) { 757 break; 758 } 759 } 760 } 761 return packageRoot; 762 } 763 764 /** 765 * Computes the minimum SDK and target SDK versions for the project 766 * 767 * @param project the project to look up the versions for 768 * @return a pair of (minimum SDK, target SDK) versions, never null 769 */ 770 @NonNull computeSdkVersions(IProject project)771 public static Pair<Integer, Integer> computeSdkVersions(IProject project) { 772 int mMinSdkVersion = 1; 773 int mTargetSdkVersion = 1; 774 775 IAbstractFile manifestFile = AndroidManifest.getManifest(new IFolderWrapper(project)); 776 if (manifestFile != null) { 777 try { 778 Object value = AndroidManifest.getMinSdkVersion(manifestFile); 779 mMinSdkVersion = 1; // Default case if missing 780 if (value instanceof Integer) { 781 mMinSdkVersion = ((Integer) value).intValue(); 782 } else if (value instanceof String) { 783 // handle codename, only if we can resolve it. 784 if (Sdk.getCurrent() != null) { 785 IAndroidTarget target = Sdk.getCurrent().getTargetFromHashString( 786 "android-" + value); //$NON-NLS-1$ 787 if (target != null) { 788 // codename future API level is current api + 1 789 mMinSdkVersion = target.getVersion().getApiLevel() + 1; 790 } 791 } 792 } 793 794 Integer i = AndroidManifest.getTargetSdkVersion(manifestFile); 795 if (i == null) { 796 mTargetSdkVersion = mMinSdkVersion; 797 } else { 798 mTargetSdkVersion = i.intValue(); 799 } 800 } catch (XPathExpressionException e) { 801 // do nothing we'll use 1 below. 802 } catch (StreamException e) { 803 // do nothing we'll use 1 below. 804 } 805 } 806 807 return Pair.of(mMinSdkVersion, mTargetSdkVersion); 808 } 809 } 810