aboutsummaryrefslogtreecommitdiffstats
path: root/ide_common
diff options
context:
space:
mode:
Diffstat (limited to 'ide_common')
-rw-r--r--ide_common/src/com/android/ide/common/rendering/LayoutLibrary.java185
1 files changed, 160 insertions, 25 deletions
diff --git a/ide_common/src/com/android/ide/common/rendering/LayoutLibrary.java b/ide_common/src/com/android/ide/common/rendering/LayoutLibrary.java
index e1256fe..dd1a3dc 100644
--- a/ide_common/src/com/android/ide/common/rendering/LayoutLibrary.java
+++ b/ide_common/src/com/android/ide/common/rendering/LayoutLibrary.java
@@ -16,7 +16,7 @@
package com.android.ide.common.rendering;
-import static com.android.ide.common.rendering.api.Result.Status.NOT_IMPLEMENTED;
+import static com.android.ide.common.rendering.api.Result.Status.ERROR_REFLECTION;
import com.android.ide.common.log.ILogger;
import com.android.ide.common.rendering.api.Bridge;
@@ -48,11 +48,13 @@ import java.awt.image.BufferedImage;
import java.io.File;
import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
+import java.lang.reflect.Method;
import java.net.URI;
import java.net.URL;
import java.net.URLClassLoader;
import java.util.ArrayList;
import java.util.HashMap;
+import java.util.List;
import java.util.Map;
import java.util.Map.Entry;
@@ -91,6 +93,16 @@ public class LayoutLibrary {
/** classloader used to load the jar file */
private final ClassLoader mClassLoader;
+ // Reflection data for older Layout Libraries.
+ private Method mViewGetParentMethod;
+ private Method mViewGetBaselineMethod;
+ private Method mViewParentIndexOfChildMethod;
+ private Class<?> mMarginLayoutParamClass;
+ private Field mLeftMarginField;
+ private Field mTopMarginField;
+ private Field mRightMarginField;
+ private Field mBottomMarginField;
+
/**
* Returns the {@link LoadStatus} of the loading of the layoutlib jar file.
*/
@@ -282,7 +294,18 @@ public class LayoutLibrary {
*/
public RenderSession createSession(SessionParams params) {
if (mBridge != null) {
- return mBridge.createSession(params);
+ RenderSession session = mBridge.createSession(params);
+ if (params.getExtendedViewInfoMode() &&
+ mBridge.getCapabilities().contains(Capability.EXTENDED_VIEWINFO) == false) {
+ // Extended view info was requested but the layoutlib does not support it.
+ // Add it manually.
+ List<ViewInfo> infoList = session.getRootViews();
+ for (ViewInfo info : infoList) {
+ addExtendedViewInfo(info);
+ }
+ }
+
+ return session;
} else if (mLegacyBridge != null) {
return createLegacySession(params);
}
@@ -333,10 +356,13 @@ public class LayoutLibrary {
*/
public Result getViewParent(Object viewObject) {
if (mBridge != null) {
- return mBridge.getViewParent(viewObject);
+ Result r = mBridge.getViewParent(viewObject);
+ if (r.isSuccess()) {
+ return r;
+ }
}
- return NOT_IMPLEMENTED.createResult();
+ return getViewParentWithReflection(viewObject);
}
/**
@@ -348,30 +374,15 @@ public class LayoutLibrary {
*/
public Result getViewIndex(Object viewObject) {
if (mBridge != null) {
- return mBridge.getViewIndex(viewObject);
- }
-
- return NOT_IMPLEMENTED.createResult();
- }
-
- /**
- * Utility method returning the baseline value for a given view object. This basically returns
- * View.getBaseline().
- *
- * @param viewObject the object for which to return the index.
- *
- * @return the baseline value or -1 if not applicable to the view object or if this layout
- * library does not implement this method.
- */
- public int getViewBaseline(Object viewObject) {
- if (mBridge != null) {
- return mBridge.getViewBaseline(viewObject);
+ Result r = mBridge.getViewIndex(viewObject);
+ if (r.isSuccess()) {
+ return r;
+ }
}
- return -1;
+ return getViewIndexReflection(viewObject);
}
-
// ------ Implementation
private LayoutLibrary(Bridge bridge, ILayoutBridge legacyBridge, ClassLoader classLoader,
@@ -440,7 +451,6 @@ public class LayoutLibrary {
};
-
// convert the map of ResourceValue into IResourceValue. Super ugly but works.
Map<String, Map<String, IResourceValue>> projectMap = convertMap(
@@ -584,4 +594,129 @@ public class LayoutLibrary {
// do nothing.
}
}
+
+ private Result getViewParentWithReflection(Object viewObject) {
+ // default implementation using reflection.
+ try {
+ if (mViewGetParentMethod == null) {
+ Class<?> viewClass = Class.forName("android.view.View");
+ mViewGetParentMethod = viewClass.getMethod("getParent");
+ }
+
+ return Status.SUCCESS.createResult(mViewGetParentMethod.invoke(viewObject));
+ } catch (Exception e) {
+ // Catch all for the reflection calls.
+ return ERROR_REFLECTION.createResult(null, e);
+ }
+ }
+
+ /**
+ * Utility method returning the index of a given view in its parent.
+ * @param viewObject the object for which to return the index.
+ *
+ * @return a {@link Result} indicating the status of the action, and if success, the index in
+ * the parent in {@link Result#getData()}
+ */
+ private Result getViewIndexReflection(Object viewObject) {
+ // default implementation using reflection.
+ try {
+ Class<?> viewClass = Class.forName("android.view.View");
+
+ if (mViewGetParentMethod == null) {
+ mViewGetParentMethod = viewClass.getMethod("getParent");
+ }
+
+ Object parentObject = mViewGetParentMethod.invoke(viewObject);
+
+ if (mViewParentIndexOfChildMethod == null) {
+ Class<?> viewParentClass = Class.forName("android.view.ViewParent");
+ mViewParentIndexOfChildMethod = viewParentClass.getMethod("indexOfChild",
+ viewClass);
+ }
+
+ return Status.SUCCESS.createResult(
+ mViewParentIndexOfChildMethod.invoke(parentObject, viewObject));
+ } catch (Exception e) {
+ // Catch all for the reflection calls.
+ return ERROR_REFLECTION.createResult(null, e);
+ }
+ }
+
+ private void addExtendedViewInfo(ViewInfo info) {
+ computeExtendedViewInfo(info);
+
+ List<ViewInfo> children = info.getChildren();
+ for (ViewInfo child : children) {
+ addExtendedViewInfo(child);
+ }
+ }
+
+ private void computeExtendedViewInfo(ViewInfo info) {
+ Object viewObject = info.getViewObject();
+ Object params = info.getLayoutParamsObject();
+
+ int baseLine = getViewBaselineReflection(viewObject);
+ int leftMargin = 0;
+ int topMargin = 0;
+ int rightMargin = 0;
+ int bottomMargin = 0;
+
+ try {
+ if (mMarginLayoutParamClass == null) {
+ mMarginLayoutParamClass = Class.forName(
+ "android.view.ViewGroup$MarginLayoutParams");
+
+ mLeftMarginField = mMarginLayoutParamClass.getField("leftMargin");
+ mTopMarginField = mMarginLayoutParamClass.getField("topMargin");
+ mRightMarginField = mMarginLayoutParamClass.getField("rightMargin");
+ mBottomMarginField = mMarginLayoutParamClass.getField("bottomMargin");
+ }
+
+ if (mMarginLayoutParamClass.isAssignableFrom(params.getClass())) {
+
+ leftMargin = (Integer)mLeftMarginField.get(params);
+ topMargin = (Integer)mTopMarginField.get(params);
+ rightMargin = (Integer)mRightMarginField.get(params);
+ bottomMargin = (Integer)mBottomMarginField.get(params);
+ }
+
+ } catch (Exception e) {
+ // just use 'unknown' value.
+ leftMargin = Integer.MIN_VALUE;
+ topMargin = Integer.MIN_VALUE;
+ rightMargin = Integer.MIN_VALUE;
+ bottomMargin = Integer.MIN_VALUE;
+ }
+
+ info.setExtendedInfo(baseLine, leftMargin, topMargin, rightMargin, bottomMargin);
+ }
+
+ /**
+ * Utility method returning the baseline value for a given view object. This basically returns
+ * View.getBaseline().
+ *
+ * @param viewObject the object for which to return the index.
+ *
+ * @return the baseline value or -1 if not applicable to the view object or if this layout
+ * library does not implement this method.
+ */
+ private int getViewBaselineReflection(Object viewObject) {
+ // default implementation using reflection.
+ try {
+ if (mViewGetBaselineMethod == null) {
+ Class<?> viewClass = Class.forName("android.view.View");
+ mViewGetBaselineMethod = viewClass.getMethod("getBaseline");
+ }
+
+ Object result = mViewGetBaselineMethod.invoke(viewObject);
+ if (result instanceof Integer) {
+ return ((Integer)result).intValue();
+ }
+
+ } catch (Exception e) {
+ // Catch all for the reflection calls.
+ }
+
+ return Integer.MIN_VALUE;
+ }
}