1 /* 2 * Copyright (C) 2008 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.sdk; 18 19 import com.android.SdkConstants; 20 import com.android.ide.common.resources.platform.AttrsXmlParser; 21 import com.android.ide.common.resources.platform.ViewClassInfo; 22 import com.android.ide.common.resources.platform.ViewClassInfo.LayoutParamsInfo; 23 import com.android.ide.eclipse.adt.AdtPlugin; 24 import com.android.ide.eclipse.adt.internal.sdk.IAndroidClassLoader.IClassDescriptor; 25 26 import org.eclipse.core.runtime.IProgressMonitor; 27 import org.eclipse.core.runtime.SubMonitor; 28 29 import java.io.IOException; 30 import java.util.ArrayList; 31 import java.util.Collection; 32 import java.util.HashMap; 33 import java.util.List; 34 import java.util.SortedMap; 35 import java.util.TreeMap; 36 37 import javax.management.InvalidAttributeValueException; 38 39 /* 40 * TODO: refactor this. Could use some cleanup. 41 */ 42 43 /** 44 * Parser for the framework library. 45 * <p/> 46 * This gather the following information: 47 * <ul> 48 * <li>Resource ID from <code>android.R</code></li> 49 * <li>The list of permissions values from <code>android.Manifest$permission</code></li> 50 * <li></li> 51 * </ul> 52 */ 53 public class LayoutParamsParser { 54 55 /** 56 * Class extending {@link ViewClassInfo} by adding the notion of instantiability. 57 * {@link LayoutParamsParser#getViews()} and {@link LayoutParamsParser#getGroups()} should 58 * only return classes that can be instantiated. 59 */ 60 final static class ExtViewClassInfo extends ViewClassInfo { 61 62 private boolean mIsInstantiable; 63 64 ExtViewClassInfo(boolean instantiable, boolean isLayout, String canonicalClassName, 65 String shortClassName) { 66 super(isLayout, canonicalClassName, shortClassName); 67 mIsInstantiable = instantiable; 68 } 69 70 boolean isInstantiable() { 71 return mIsInstantiable; 72 } 73 } 74 75 /* Note: protected members/methods are overridden in unit tests */ 76 77 /** Reference to android.view.View */ 78 protected IClassDescriptor mTopViewClass; 79 /** Reference to android.view.ViewGroup */ 80 protected IClassDescriptor mTopGroupClass; 81 /** Reference to android.view.ViewGroup$LayoutParams */ 82 protected IClassDescriptor mTopLayoutParamsClass; 83 84 /** Input list of all classes deriving from android.view.View */ 85 protected ArrayList<IClassDescriptor> mViewList; 86 /** Input list of all classes deriving from android.view.ViewGroup */ 87 protected ArrayList<IClassDescriptor> mGroupList; 88 89 /** Output map of FQCN => info on View classes */ 90 protected TreeMap<String, ExtViewClassInfo> mViewMap; 91 /** Output map of FQCN => info on ViewGroup classes */ 92 protected TreeMap<String, ExtViewClassInfo> mGroupMap; 93 /** Output map of FQCN => info on LayoutParams classes */ 94 protected HashMap<String, LayoutParamsInfo> mLayoutParamsMap; 95 96 /** The attrs.xml parser */ 97 protected AttrsXmlParser mAttrsXmlParser; 98 99 /** The android.jar class loader */ 100 protected IAndroidClassLoader mClassLoader; 101 102 /** 103 * Instantiate a new LayoutParamsParser. 104 * @param classLoader The android.jar class loader 105 * @param attrsXmlParser The parser of the attrs.xml file 106 */ 107 public LayoutParamsParser(IAndroidClassLoader classLoader, 108 AttrsXmlParser attrsXmlParser) { 109 mClassLoader = classLoader; 110 mAttrsXmlParser = attrsXmlParser; 111 } 112 113 /** Returns the map of FQCN => info on View classes */ 114 public List<ViewClassInfo> getViews() { 115 return getInstantiables(mViewMap); 116 } 117 118 /** Returns the map of FQCN => info on ViewGroup classes */ 119 public List<ViewClassInfo> getGroups() { 120 return getInstantiables(mGroupMap); 121 } 122 123 /** 124 * TODO: doc here. 125 * <p/> 126 * Note: on output we should have NO dependency on {@link IClassDescriptor}, 127 * otherwise we wouldn't be able to unload the class loader later. 128 * <p/> 129 * Note on Vocabulary: FQCN=Fully Qualified Class Name (e.g. "my.package.class$innerClass") 130 * @param monitor A progress monitor. Can be null. Caller is responsible for calling done. 131 */ 132 public void parseLayoutClasses(IProgressMonitor monitor) { 133 parseClasses(monitor, 134 SdkConstants.CLASS_VIEW, 135 SdkConstants.CLASS_VIEWGROUP, 136 SdkConstants.CLASS_VIEWGROUP_LAYOUTPARAMS); 137 } 138 139 public void parsePreferencesClasses(IProgressMonitor monitor) { 140 parseClasses(monitor, 141 SdkConstants.CLASS_PREFERENCE, 142 SdkConstants.CLASS_PREFERENCEGROUP, 143 null /* paramsClassName */ ); 144 } 145 146 private void parseClasses(IProgressMonitor monitor, 147 String rootClassName, 148 String groupClassName, 149 String paramsClassName) { 150 try { 151 SubMonitor progress = SubMonitor.convert(monitor, 100); 152 153 String[] superClasses = new String[2 + (paramsClassName == null ? 0 : 1)]; 154 superClasses[0] = groupClassName; 155 superClasses[1] = rootClassName; 156 if (paramsClassName != null) { 157 superClasses[2] = paramsClassName; 158 } 159 HashMap<String, ArrayList<IClassDescriptor>> found = 160 mClassLoader.findClassesDerivingFrom("android.", superClasses); //$NON-NLS-1$ 161 mTopViewClass = mClassLoader.getClass(rootClassName); 162 mTopGroupClass = mClassLoader.getClass(groupClassName); 163 if (paramsClassName != null) { 164 mTopLayoutParamsClass = mClassLoader.getClass(paramsClassName); 165 } 166 167 mViewList = found.get(rootClassName); 168 mGroupList = found.get(groupClassName); 169 170 mViewMap = new TreeMap<String, ExtViewClassInfo>(); 171 mGroupMap = new TreeMap<String, ExtViewClassInfo>(); 172 if (mTopLayoutParamsClass != null) { 173 mLayoutParamsMap = new HashMap<String, LayoutParamsInfo>(); 174 } 175 176 // Add top classes to the maps since by design they are not listed in classes deriving 177 // from themselves. 178 if (mTopGroupClass != null) { 179 addGroup(mTopGroupClass); 180 } 181 if (mTopViewClass != null) { 182 addView(mTopViewClass); 183 } 184 185 // ViewGroup derives from View 186 ExtViewClassInfo vg = mGroupMap.get(groupClassName); 187 if (vg != null) { 188 vg.setSuperClass(mViewMap.get(rootClassName)); 189 } 190 191 progress.setWorkRemaining(mGroupList.size() + mViewList.size()); 192 193 for (IClassDescriptor groupChild : mGroupList) { 194 addGroup(groupChild); 195 progress.worked(1); 196 } 197 198 for (IClassDescriptor viewChild : mViewList) { 199 if (viewChild != mTopGroupClass) { 200 addView(viewChild); 201 } 202 progress.worked(1); 203 } 204 } catch (ClassNotFoundException e) { 205 AdtPlugin.log(e, "Problem loading class %1$s or %2$s", //$NON-NLS-1$ 206 rootClassName, groupClassName); 207 } catch (InvalidAttributeValueException e) { 208 AdtPlugin.log(e, "Problem loading classes"); //$NON-NLS-1$ 209 } catch (ClassFormatError e) { 210 AdtPlugin.log(e, "Problem loading classes"); //$NON-NLS-1$ 211 } catch (IOException e) { 212 AdtPlugin.log(e, "Problem loading classes"); //$NON-NLS-1$ 213 } 214 } 215 216 /** 217 * Parses a View class and adds a ExtViewClassInfo for it in mViewMap. 218 * It calls itself recursively to handle super classes which are also Views. 219 */ 220 private ExtViewClassInfo addView(IClassDescriptor viewClass) { 221 String fqcn = viewClass.getFullClassName(); 222 if (mViewMap.containsKey(fqcn)) { 223 return mViewMap.get(fqcn); 224 } else if (mGroupMap.containsKey(fqcn)) { 225 return mGroupMap.get(fqcn); 226 } 227 228 ExtViewClassInfo info = new ExtViewClassInfo(viewClass.isInstantiable(), 229 false /* layout */, fqcn, viewClass.getSimpleName()); 230 mViewMap.put(fqcn, info); 231 232 // All view classes derive from mTopViewClass by design. 233 // Do not lookup the super class for mTopViewClass itself. 234 if (viewClass.equals(mTopViewClass) == false) { 235 IClassDescriptor superClass = viewClass.getSuperclass(); 236 ExtViewClassInfo superClassInfo = addView(superClass); 237 info.setSuperClass(superClassInfo); 238 } 239 240 mAttrsXmlParser.loadViewAttributes(info); 241 return info; 242 } 243 244 /** 245 * Parses a ViewGroup class and adds a ExtViewClassInfo for it in mGroupMap. 246 * It calls itself recursively to handle super classes which are also ViewGroups. 247 */ 248 private ExtViewClassInfo addGroup(IClassDescriptor groupClass) { 249 String fqcn = groupClass.getFullClassName(); 250 if (mGroupMap.containsKey(fqcn)) { 251 return mGroupMap.get(fqcn); 252 } 253 254 ExtViewClassInfo info = new ExtViewClassInfo(groupClass.isInstantiable(), 255 true /* layout */, fqcn, groupClass.getSimpleName()); 256 mGroupMap.put(fqcn, info); 257 258 // All groups derive from android.view.ViewGroup, which in turns derives from 259 // android.view.View (i.e. mTopViewClass here). So the only group that can have View as 260 // its super class is the ViewGroup base class and we don't try to resolve it since groups 261 // are loaded before views. 262 IClassDescriptor superClass = groupClass.getSuperclass(); 263 264 // Assertion: at this point, we should have 265 // superClass != mTopViewClass || fqcn.equals(SdkConstants.CLASS_VIEWGROUP); 266 267 if (superClass != null && superClass.equals(mTopViewClass) == false) { 268 ExtViewClassInfo superClassInfo = addGroup(superClass); 269 270 // Assertion: we should have superClassInfo != null && superClassInfo != info; 271 if (superClassInfo != null && superClassInfo != info) { 272 info.setSuperClass(superClassInfo); 273 } 274 } 275 276 mAttrsXmlParser.loadViewAttributes(info); 277 if (mTopLayoutParamsClass != null) { 278 info.setLayoutParams(addLayoutParams(groupClass)); 279 } 280 return info; 281 } 282 283 /** 284 * Parses a ViewGroup class and returns an info object on its inner LayoutParams. 285 * 286 * @return The {@link LayoutParamsInfo} for the ViewGroup class or null. 287 */ 288 private LayoutParamsInfo addLayoutParams(IClassDescriptor groupClass) { 289 290 // Is there a LayoutParams in this group class? 291 IClassDescriptor layoutParamsClass = findLayoutParams(groupClass); 292 293 // if there's no layout data in the group class, link to the one from the 294 // super class. 295 if (layoutParamsClass == null) { 296 for (IClassDescriptor superClass = groupClass.getSuperclass(); 297 layoutParamsClass == null && 298 superClass != null && 299 superClass.equals(mTopViewClass) == false; 300 superClass = superClass.getSuperclass()) { 301 layoutParamsClass = findLayoutParams(superClass); 302 } 303 } 304 305 if (layoutParamsClass != null) { 306 return getLayoutParamsInfo(layoutParamsClass); 307 } 308 309 return null; 310 } 311 312 /** 313 * Parses a LayoutParams class and returns a LayoutParamsInfo object for it. 314 * It calls itself recursively to handle the super class of the LayoutParams. 315 */ 316 private LayoutParamsInfo getLayoutParamsInfo(IClassDescriptor layoutParamsClass) { 317 String fqcn = layoutParamsClass.getFullClassName(); 318 LayoutParamsInfo layoutParamsInfo = mLayoutParamsMap.get(fqcn); 319 320 if (layoutParamsInfo != null) { 321 return layoutParamsInfo; 322 } 323 324 // Find the link on the LayoutParams super class 325 LayoutParamsInfo superClassInfo = null; 326 if (layoutParamsClass.equals(mTopLayoutParamsClass) == false) { 327 IClassDescriptor superClass = layoutParamsClass.getSuperclass(); 328 superClassInfo = getLayoutParamsInfo(superClass); 329 } 330 331 // Find the link on the enclosing ViewGroup 332 ExtViewClassInfo enclosingGroupInfo = addGroup(layoutParamsClass.getEnclosingClass()); 333 334 layoutParamsInfo = new ExtViewClassInfo.LayoutParamsInfo( 335 enclosingGroupInfo, layoutParamsClass.getSimpleName(), superClassInfo); 336 mLayoutParamsMap.put(fqcn, layoutParamsInfo); 337 338 mAttrsXmlParser.loadLayoutParamsAttributes(layoutParamsInfo); 339 340 return layoutParamsInfo; 341 } 342 343 /** 344 * Given a ViewGroup-derived class, looks for an inner class named LayoutParams 345 * and if found returns its class definition. 346 * <p/> 347 * This uses the actual defined inner classes and does not look at inherited classes. 348 * 349 * @param groupClass The ViewGroup derived class 350 * @return The Class of the inner LayoutParams or null if none is declared. 351 */ 352 private IClassDescriptor findLayoutParams(IClassDescriptor groupClass) { 353 IClassDescriptor[] innerClasses = groupClass.getDeclaredClasses(); 354 for (IClassDescriptor innerClass : innerClasses) { 355 if (innerClass.getSimpleName().equals(SdkConstants.CLASS_NAME_LAYOUTPARAMS)) { 356 return innerClass; 357 } 358 } 359 return null; 360 } 361 362 /** 363 * Computes and return a list of ViewClassInfo from a map by filtering out the class that 364 * cannot be instantiated. 365 */ 366 private List<ViewClassInfo> getInstantiables(SortedMap<String, ExtViewClassInfo> map) { 367 Collection<ExtViewClassInfo> values = map.values(); 368 ArrayList<ViewClassInfo> list = new ArrayList<ViewClassInfo>(); 369 370 for (ExtViewClassInfo info : values) { 371 if (info.isInstantiable()) { 372 list.add(info); 373 } 374 } 375 376 return list; 377 } 378 } 379