1 /*
2  * Copyright (C) 2008 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 com.android.ide.common.rendering.api.LayoutLog;
20 import com.android.ide.common.rendering.api.LayoutlibCallback;
21 import com.android.ide.common.rendering.api.MergeCookie;
22 import com.android.ide.common.rendering.api.ResourceReference;
23 import com.android.ide.common.rendering.api.ResourceValue;
24 import com.android.layoutlib.bridge.Bridge;
25 import com.android.layoutlib.bridge.BridgeConstants;
26 import com.android.layoutlib.bridge.MockView;
27 import com.android.layoutlib.bridge.android.BridgeContext;
28 import com.android.layoutlib.bridge.android.BridgeXmlBlockParser;
29 import com.android.layoutlib.bridge.android.support.DrawerLayoutUtil;
30 import com.android.layoutlib.bridge.android.support.RecyclerViewUtil;
31 import com.android.layoutlib.bridge.impl.ParserFactory;
32 import com.android.layoutlib.bridge.util.ReflectionUtils;
33 import com.android.resources.ResourceType;
34 import com.android.util.Pair;
35 
36 import org.xmlpull.v1.XmlPullParser;
37 
38 import android.annotation.NonNull;
39 import android.content.Context;
40 import android.content.res.TypedArray;
41 import android.graphics.drawable.Animatable;
42 import android.graphics.drawable.Drawable;
43 import android.util.AttributeSet;
44 import android.widget.ImageView;
45 import android.widget.NumberPicker;
46 
47 import java.io.File;
48 import java.util.Arrays;
49 import java.util.Collections;
50 import java.util.HashMap;
51 import java.util.HashSet;
52 import java.util.Map;
53 import java.util.Set;
54 
55 import static com.android.SdkConstants.AUTO_COMPLETE_TEXT_VIEW;
56 import static com.android.SdkConstants.BUTTON;
57 import static com.android.SdkConstants.CHECKED_TEXT_VIEW;
58 import static com.android.SdkConstants.CHECK_BOX;
59 import static com.android.SdkConstants.EDIT_TEXT;
60 import static com.android.SdkConstants.IMAGE_BUTTON;
61 import static com.android.SdkConstants.IMAGE_VIEW;
62 import static com.android.SdkConstants.MULTI_AUTO_COMPLETE_TEXT_VIEW;
63 import static com.android.SdkConstants.RADIO_BUTTON;
64 import static com.android.SdkConstants.SEEK_BAR;
65 import static com.android.SdkConstants.SPINNER;
66 import static com.android.SdkConstants.TEXT_VIEW;
67 import static com.android.layoutlib.bridge.android.BridgeContext.getBaseContext;
68 
69 /**
70  * Custom implementation of {@link LayoutInflater} to handle custom views.
71  */
72 public final class BridgeInflater extends LayoutInflater {
73 
74     private final LayoutlibCallback mLayoutlibCallback;
75     /**
76      * If true, the inflater will try to replace the framework widgets with the AppCompat versions.
77      * Ideally, this should be based on the activity being an AppCompat activity but since that is
78      * not trivial to check from layoutlib, we currently base the decision on the current theme
79      * being an AppCompat theme.
80      */
81     private boolean mLoadAppCompatViews;
82     /**
83      * This set contains the framework views that have an AppCompat version but failed to load.
84      * This might happen because not all widgets are contained in all versions of the support
85      * library.
86      * This will help us to avoid trying to load the AppCompat version multiple times if it
87      * doesn't exist.
88      */
89     private Set<String> mFailedAppCompatViews = new HashSet<>();
90     private boolean mIsInMerge = false;
91     private ResourceReference mResourceReference;
92     private Map<View, String> mOpenDrawerLayouts;
93 
94     // Keep in sync with the same value in LayoutInflater.
95     private static final int[] ATTRS_THEME = new int[] {com.android.internal.R.attr.theme };
96 
97     private static final String APPCOMPAT_WIDGET_PREFIX = "android.support.v7.widget.AppCompat";
98     /** List of platform widgets that have an AppCompat version */
99     private static final Set<String> APPCOMPAT_VIEWS = Collections.unmodifiableSet(
100             new HashSet<>(
101                     Arrays.asList(TEXT_VIEW, IMAGE_VIEW, BUTTON, EDIT_TEXT, SPINNER,
102                             IMAGE_BUTTON, CHECK_BOX, RADIO_BUTTON, CHECKED_TEXT_VIEW,
103                             AUTO_COMPLETE_TEXT_VIEW, MULTI_AUTO_COMPLETE_TEXT_VIEW, "RatingBar",
104                             SEEK_BAR)));
105 
106     /**
107      * List of class prefixes which are tried first by default.
108      * <p/>
109      * This should match the list in com.android.internal.policy.impl.PhoneLayoutInflater.
110      */
111     private static final String[] sClassPrefixList = {
112         "android.widget.",
113         "android.webkit.",
114         "android.app."
115     };
116 
getClassPrefixList()117     public static String[] getClassPrefixList() {
118         return sClassPrefixList;
119     }
120 
BridgeInflater(LayoutInflater original, Context newContext)121     private BridgeInflater(LayoutInflater original, Context newContext) {
122         super(original, newContext);
123         newContext = getBaseContext(newContext);
124         if (newContext instanceof BridgeContext) {
125             mLayoutlibCallback = ((BridgeContext) newContext).getLayoutlibCallback();
126             mLoadAppCompatViews = ((BridgeContext) newContext).isAppCompatTheme();
127         } else {
128             mLayoutlibCallback = null;
129             mLoadAppCompatViews = false;
130         }
131     }
132 
133     /**
134      * Instantiate a new BridgeInflater with an {@link LayoutlibCallback} object.
135      *
136      * @param context The Android application context.
137      * @param layoutlibCallback the {@link LayoutlibCallback} object.
138      */
BridgeInflater(BridgeContext context, LayoutlibCallback layoutlibCallback)139     public BridgeInflater(BridgeContext context, LayoutlibCallback layoutlibCallback) {
140         super(context);
141         mLayoutlibCallback = layoutlibCallback;
142         mConstructorArgs[0] = context;
143         mLoadAppCompatViews = context.isAppCompatTheme();
144     }
145 
146     @Override
onCreateView(String name, AttributeSet attrs)147     public View onCreateView(String name, AttributeSet attrs) throws ClassNotFoundException {
148         View view = null;
149 
150         try {
151             if (mLoadAppCompatViews
152                     && APPCOMPAT_VIEWS.contains(name)
153                     && !mFailedAppCompatViews.contains(name)) {
154                 // We are using an AppCompat theme so try to load the appcompat views
155                 view = loadCustomView(APPCOMPAT_WIDGET_PREFIX + name, attrs, true);
156 
157                 if (view == null) {
158                     mFailedAppCompatViews.add(name); // Do not try this one anymore
159                 }
160             }
161 
162             if (view == null) {
163                 // First try to find a class using the default Android prefixes
164                 for (String prefix : sClassPrefixList) {
165                     try {
166                         view = createView(name, prefix, attrs);
167                         if (view != null) {
168                             break;
169                         }
170                     } catch (ClassNotFoundException e) {
171                         // Ignore. We'll try again using the base class below.
172                     }
173                 }
174 
175                 // Next try using the parent loader. This will most likely only work for
176                 // fully-qualified class names.
177                 try {
178                     if (view == null) {
179                         view = super.onCreateView(name, attrs);
180                     }
181                 } catch (ClassNotFoundException e) {
182                     // Ignore. We'll try again using the custom view loader below.
183                 }
184             }
185 
186             // Finally try again using the custom view loader
187             if (view == null) {
188                 view = loadCustomView(name, attrs);
189             }
190         } catch (InflateException e) {
191             // Don't catch the InflateException below as that results in hiding the real cause.
192             throw e;
193         } catch (Exception e) {
194             // Wrap the real exception in a ClassNotFoundException, so that the calling method
195             // can deal with it.
196             throw new ClassNotFoundException("onCreateView", e);
197         }
198 
199         setupViewInContext(view, attrs);
200 
201         return view;
202     }
203 
204     @Override
createViewFromTag(View parent, String name, Context context, AttributeSet attrs, boolean ignoreThemeAttr)205     public View createViewFromTag(View parent, String name, Context context, AttributeSet attrs,
206             boolean ignoreThemeAttr) {
207         View view = null;
208         if (name.equals("view")) {
209             // This is usually done by the superclass but this allows us catching the error and
210             // reporting something useful.
211             name = attrs.getAttributeValue(null, "class");
212 
213             if (name == null) {
214                 Bridge.getLog().error(LayoutLog.TAG_BROKEN, "Unable to inflate view tag without " +
215                   "class attribute", null);
216                 // We weren't able to resolve the view so we just pass a mock View to be able to
217                 // continue rendering.
218                 view = new MockView(context, attrs);
219                 ((MockView) view).setText("view");
220             }
221         }
222 
223         try {
224             if (view == null) {
225                 view = super.createViewFromTag(parent, name, context, attrs, ignoreThemeAttr);
226             }
227         } catch (InflateException e) {
228             // Creation of ContextThemeWrapper code is same as in the super method.
229             // Apply a theme wrapper, if allowed and one is specified.
230             if (!ignoreThemeAttr) {
231                 final TypedArray ta = context.obtainStyledAttributes(attrs, ATTRS_THEME);
232                 final int themeResId = ta.getResourceId(0, 0);
233                 if (themeResId != 0) {
234                     context = new ContextThemeWrapper(context, themeResId);
235                 }
236                 ta.recycle();
237             }
238             if (!(e.getCause() instanceof ClassNotFoundException)) {
239                 // There is some unknown inflation exception in inflating a View that was found.
240                 view = new MockView(context, attrs);
241                 ((MockView) view).setText(name);
242                 Bridge.getLog().error(LayoutLog.TAG_BROKEN, e.getMessage(), e, null);
243             } else {
244                 final Object lastContext = mConstructorArgs[0];
245                 mConstructorArgs[0] = context;
246                 // try to load the class from using the custom view loader
247                 try {
248                     view = loadCustomView(name, attrs);
249                 } catch (Exception e2) {
250                     // Wrap the real exception in an InflateException so that the calling
251                     // method can deal with it.
252                     InflateException exception = new InflateException();
253                     if (!e2.getClass().equals(ClassNotFoundException.class)) {
254                         exception.initCause(e2);
255                     } else {
256                         exception.initCause(e);
257                     }
258                     throw exception;
259                 } finally {
260                     mConstructorArgs[0] = lastContext;
261                 }
262             }
263         }
264 
265         setupViewInContext(view, attrs);
266 
267         return view;
268     }
269 
270     @Override
inflate(int resource, ViewGroup root)271     public View inflate(int resource, ViewGroup root) {
272         Context context = getContext();
273         context = getBaseContext(context);
274         if (context instanceof BridgeContext) {
275             BridgeContext bridgeContext = (BridgeContext)context;
276 
277             ResourceValue value = null;
278 
279             @SuppressWarnings("deprecation")
280             Pair<ResourceType, String> layoutInfo = Bridge.resolveResourceId(resource);
281             if (layoutInfo != null) {
282                 value = bridgeContext.getRenderResources().getFrameworkResource(
283                         ResourceType.LAYOUT, layoutInfo.getSecond());
284             } else {
285                 layoutInfo = mLayoutlibCallback.resolveResourceId(resource);
286 
287                 if (layoutInfo != null) {
288                     value = bridgeContext.getRenderResources().getProjectResource(
289                             ResourceType.LAYOUT, layoutInfo.getSecond());
290                 }
291             }
292 
293             if (value != null) {
294                 File f = new File(value.getValue());
295                 if (f.isFile()) {
296                     try {
297                         XmlPullParser parser = ParserFactory.create(f, true);
298 
299                         BridgeXmlBlockParser bridgeParser = new BridgeXmlBlockParser(
300                                 parser, bridgeContext, value.isFramework());
301 
302                         return inflate(bridgeParser, root);
303                     } catch (Exception e) {
304                         Bridge.getLog().error(LayoutLog.TAG_RESOURCES_READ,
305                                 "Failed to parse file " + f.getAbsolutePath(), e, null);
306 
307                         return null;
308                     }
309                 }
310             }
311         }
312         return null;
313     }
314 
315     /**
316      * Instantiates the given view name and returns the instance. If the view doesn't exist, a
317      * MockView or null might be returned.
318      * @param name the custom view name
319      * @param attrs the {@link AttributeSet} to be passed to the view constructor
320      * @param silent if true, errors while loading the view won't be reported and, if the view
321      * doesn't exist, null will be returned.
322      */
loadCustomView(String name, AttributeSet attrs, boolean silent)323     private View loadCustomView(String name, AttributeSet attrs, boolean silent) throws Exception {
324         if (mLayoutlibCallback != null) {
325             // first get the classname in case it's not the node name
326             if (name.equals("view")) {
327                 name = attrs.getAttributeValue(null, "class");
328                 if (name == null) {
329                     return null;
330                 }
331             }
332 
333             mConstructorArgs[1] = attrs;
334 
335             Object customView = silent ?
336                     mLayoutlibCallback.loadClass(name, mConstructorSignature, mConstructorArgs)
337                     : mLayoutlibCallback.loadView(name, mConstructorSignature, mConstructorArgs);
338 
339             if (customView instanceof View) {
340                 return (View)customView;
341             }
342         }
343 
344         return null;
345     }
346 
loadCustomView(String name, AttributeSet attrs)347     private View loadCustomView(String name, AttributeSet attrs) throws Exception {
348         return loadCustomView(name, attrs, false);
349     }
350 
setupViewInContext(View view, AttributeSet attrs)351     private void setupViewInContext(View view, AttributeSet attrs) {
352         Context context = getContext();
353         context = getBaseContext(context);
354         if (context instanceof BridgeContext) {
355             BridgeContext bc = (BridgeContext) context;
356             // get the view key
357             Object viewKey = getViewKeyFromParser(attrs, bc, mResourceReference, mIsInMerge);
358             if (viewKey != null) {
359                 bc.addViewKey(view, viewKey);
360             }
361             String scrollPosX = attrs.getAttributeValue(BridgeConstants.NS_RESOURCES, "scrollX");
362             if (scrollPosX != null && scrollPosX.endsWith("px")) {
363                 int value = Integer.parseInt(scrollPosX.substring(0, scrollPosX.length() - 2));
364                 bc.setScrollXPos(view, value);
365             }
366             String scrollPosY = attrs.getAttributeValue(BridgeConstants.NS_RESOURCES, "scrollY");
367             if (scrollPosY != null && scrollPosY.endsWith("px")) {
368                 int value = Integer.parseInt(scrollPosY.substring(0, scrollPosY.length() - 2));
369                 bc.setScrollYPos(view, value);
370             }
371             if (ReflectionUtils.isInstanceOf(view, RecyclerViewUtil.CN_RECYCLER_VIEW)) {
372                 Integer resourceId = null;
373                 String attrVal = attrs.getAttributeValue(BridgeConstants.NS_TOOLS_URI,
374                         BridgeConstants.ATTR_LIST_ITEM);
375                 if (attrVal != null && !attrVal.isEmpty()) {
376                     ResourceValue resValue = bc.getRenderResources().findResValue(attrVal, false);
377                     if (resValue.isFramework()) {
378                         resourceId = Bridge.getResourceId(resValue.getResourceType(),
379                                 resValue.getName());
380                     } else {
381                         resourceId = mLayoutlibCallback.getResourceId(resValue.getResourceType(),
382                                 resValue.getName());
383                     }
384                 }
385                 if (resourceId == null) {
386                     resourceId = 0;
387                 }
388                 RecyclerViewUtil.setAdapter(view, bc, mLayoutlibCallback, resourceId);
389             } else if (ReflectionUtils.isInstanceOf(view, DrawerLayoutUtil.CN_DRAWER_LAYOUT)) {
390                 String attrVal = attrs.getAttributeValue(BridgeConstants.NS_TOOLS_URI,
391                         BridgeConstants.ATTR_OPEN_DRAWER);
392                 if (attrVal != null) {
393                     getDrawerLayoutMap().put(view, attrVal);
394                 }
395             }
396             else if (view instanceof NumberPicker) {
397                 NumberPicker numberPicker = (NumberPicker) view;
398                 String minValue = attrs.getAttributeValue(BridgeConstants.NS_TOOLS_URI, "minValue");
399                 if (minValue != null) {
400                     numberPicker.setMinValue(Integer.parseInt(minValue));
401                 }
402                 String maxValue = attrs.getAttributeValue(BridgeConstants.NS_TOOLS_URI, "maxValue");
403                 if (maxValue != null) {
404                     numberPicker.setMaxValue(Integer.parseInt(maxValue));
405                 }
406             }
407             else if (view instanceof ImageView) {
408                 ImageView img = (ImageView) view;
409                 Drawable drawable = img.getDrawable();
410                 if (drawable instanceof Animatable) {
411                     if (!((Animatable) drawable).isRunning()) {
412                         ((Animatable) drawable).start();
413                     }
414                 }
415             }
416 
417         }
418     }
419 
setIsInMerge(boolean isInMerge)420     public void setIsInMerge(boolean isInMerge) {
421         mIsInMerge = isInMerge;
422     }
423 
setResourceReference(ResourceReference reference)424     public void setResourceReference(ResourceReference reference) {
425         mResourceReference = reference;
426     }
427 
428     @Override
cloneInContext(Context newContext)429     public LayoutInflater cloneInContext(Context newContext) {
430         return new BridgeInflater(this, newContext);
431     }
432 
getViewKeyFromParser(AttributeSet attrs, BridgeContext bc, ResourceReference resourceReference, boolean isInMerge)433     /*package*/ static Object getViewKeyFromParser(AttributeSet attrs, BridgeContext bc,
434             ResourceReference resourceReference, boolean isInMerge) {
435 
436         if (!(attrs instanceof BridgeXmlBlockParser)) {
437             return null;
438         }
439         BridgeXmlBlockParser parser = ((BridgeXmlBlockParser) attrs);
440 
441         // get the view key
442         Object viewKey = parser.getViewCookie();
443 
444         if (viewKey == null) {
445             int currentDepth = parser.getDepth();
446 
447             // test whether we are in an included file or in a adapter binding view.
448             BridgeXmlBlockParser previousParser = bc.getPreviousParser();
449             if (previousParser != null) {
450                 // looks like we are inside an embedded layout.
451                 // only apply the cookie of the calling node (<include>) if we are at the
452                 // top level of the embedded layout. If there is a merge tag, then
453                 // skip it and look for the 2nd level
454                 int testDepth = isInMerge ? 2 : 1;
455                 if (currentDepth == testDepth) {
456                     viewKey = previousParser.getViewCookie();
457                     // if we are in a merge, wrap the cookie in a MergeCookie.
458                     if (viewKey != null && isInMerge) {
459                         viewKey = new MergeCookie(viewKey);
460                     }
461                 }
462             } else if (resourceReference != null && currentDepth == 1) {
463                 // else if there's a resource reference, this means we are in an adapter
464                 // binding case. Set the resource ref as the view cookie only for the top
465                 // level view.
466                 viewKey = resourceReference;
467             }
468         }
469 
470         return viewKey;
471     }
472 
postInflateProcess(View view)473     public void postInflateProcess(View view) {
474         if (mOpenDrawerLayouts != null) {
475             String gravity = mOpenDrawerLayouts.get(view);
476             if (gravity != null) {
477                 DrawerLayoutUtil.openDrawer(view, gravity);
478             }
479             mOpenDrawerLayouts.remove(view);
480         }
481     }
482 
483     @NonNull
getDrawerLayoutMap()484     private Map<View, String> getDrawerLayoutMap() {
485         if (mOpenDrawerLayouts == null) {
486             mOpenDrawerLayouts = new HashMap<View, String>(4);
487         }
488         return mOpenDrawerLayouts;
489     }
490 
onDoneInflation()491     public void onDoneInflation() {
492         if (mOpenDrawerLayouts != null) {
493             mOpenDrawerLayouts.clear();
494         }
495     }
496 }
497