1 /*
2  * Copyright (C) 2013 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 package android.view;
17 
18 import android.animation.LayoutTransition;
19 import android.annotation.NonNull;
20 import android.content.Context;
21 import android.graphics.Canvas;
22 import android.graphics.Rect;
23 import android.graphics.drawable.Drawable;
24 
25 import java.util.ArrayList;
26 
27 /**
28  * An overlay is an extra layer that sits on top of a View (the "host view")
29  * which is drawn after all other content in that view (including children,
30  * if the view is a ViewGroup). Interaction with the overlay layer is done
31  * by adding and removing drawables.
32  *
33  * <p>An overlay requested from a ViewGroup is of type {@link ViewGroupOverlay},
34  * which also supports adding and removing views.</p>
35  *
36  * @see View#getOverlay() View.getOverlay()
37  * @see ViewGroup#getOverlay() ViewGroup.getOverlay()
38  * @see ViewGroupOverlay
39  */
40 public class ViewOverlay {
41 
42     /**
43      * The actual container for the drawables (and views, if it's a ViewGroupOverlay).
44      * All of the management and rendering details for the overlay are handled in
45      * OverlayViewGroup.
46      */
47     OverlayViewGroup mOverlayViewGroup;
48 
ViewOverlay(Context context, View hostView)49     ViewOverlay(Context context, View hostView) {
50         mOverlayViewGroup = new OverlayViewGroup(context, hostView);
51     }
52 
53     /**
54      * Used internally by View and ViewGroup to handle drawing and invalidation
55      * of the overlay
56      * @return
57      */
getOverlayView()58     ViewGroup getOverlayView() {
59         return mOverlayViewGroup;
60     }
61 
62     /**
63      * Adds a {@link Drawable} to the overlay. The bounds of the drawable should be relative to
64      * the host view. Any drawable added to the overlay should be removed when it is no longer
65      * needed or no longer visible. Adding an already existing {@link Drawable}
66      * is a no-op. Passing <code>null</code> parameter will result in an
67      * {@link IllegalArgumentException} being thrown.
68      *
69      * @param drawable The {@link Drawable} to be added to the overlay. This drawable will be
70      * drawn when the view redraws its overlay. {@link Drawable}s will be drawn in the order that
71      * they were added.
72      * @see #remove(Drawable)
73      */
add(@onNull Drawable drawable)74     public void add(@NonNull Drawable drawable) {
75         mOverlayViewGroup.add(drawable);
76     }
77 
78     /**
79      * Removes the specified {@link Drawable} from the overlay. Removing a {@link Drawable} that was
80      * not added with {@link #add(Drawable)} is a no-op. Passing <code>null</code> parameter will
81      * result in an {@link IllegalArgumentException} being thrown.
82      *
83      * @param drawable The {@link Drawable} to be removed from the overlay.
84      * @see #add(Drawable)
85      */
remove(@onNull Drawable drawable)86     public void remove(@NonNull Drawable drawable) {
87         mOverlayViewGroup.remove(drawable);
88     }
89 
90     /**
91      * Removes all content from the overlay.
92      */
clear()93     public void clear() {
94         mOverlayViewGroup.clear();
95     }
96 
isEmpty()97     boolean isEmpty() {
98         return mOverlayViewGroup.isEmpty();
99     }
100 
101     /**
102      * OverlayViewGroup is a container that View and ViewGroup use to host
103      * drawables and views added to their overlays  ({@link ViewOverlay} and
104      * {@link ViewGroupOverlay}, respectively). Drawables are added to the overlay
105      * via the add/remove methods in ViewOverlay, Views are added/removed via
106      * ViewGroupOverlay. These drawable and view objects are
107      * drawn whenever the view itself is drawn; first the view draws its own
108      * content (and children, if it is a ViewGroup), then it draws its overlay
109      * (if it has one).
110      *
111      * <p>Besides managing and drawing the list of drawables, this class serves
112      * two purposes:
113      * (1) it noops layout calls because children are absolutely positioned and
114      * (2) it forwards all invalidation calls to its host view. The invalidation
115      * redirect is necessary because the overlay is not a child of the host view
116      * and invalidation cannot therefore follow the normal path up through the
117      * parent hierarchy.</p>
118      *
119      * @see View#getOverlay()
120      * @see ViewGroup#getOverlay()
121      */
122     static class OverlayViewGroup extends ViewGroup {
123 
124         /**
125          * The View for which this is an overlay. Invalidations of the overlay are redirected to
126          * this host view.
127          */
128         final View mHostView;
129 
130         /**
131          * The set of drawables to draw when the overlay is rendered.
132          */
133         ArrayList<Drawable> mDrawables = null;
134 
OverlayViewGroup(Context context, View hostView)135         OverlayViewGroup(Context context, View hostView) {
136             super(context);
137             mHostView = hostView;
138             mAttachInfo = mHostView.mAttachInfo;
139 
140             mRight = hostView.getWidth();
141             mBottom = hostView.getHeight();
142             // pass right+bottom directly to RenderNode, since not going through setters
143             mRenderNode.setLeftTopRightBottom(0, 0, mRight, mBottom);
144         }
145 
add(@onNull Drawable drawable)146         public void add(@NonNull Drawable drawable) {
147             if (drawable == null) {
148                 throw new IllegalArgumentException("drawable must be non-null");
149             }
150             if (mDrawables == null) {
151                 mDrawables = new ArrayList<>();
152             }
153             if (!mDrawables.contains(drawable)) {
154                 // Make each drawable unique in the overlay; can't add it more than once
155                 mDrawables.add(drawable);
156                 invalidate(drawable.getBounds());
157                 drawable.setCallback(this);
158             }
159         }
160 
remove(@onNull Drawable drawable)161         public void remove(@NonNull Drawable drawable) {
162             if (drawable == null) {
163                 throw new IllegalArgumentException("drawable must be non-null");
164             }
165             if (mDrawables != null) {
166                 mDrawables.remove(drawable);
167                 invalidate(drawable.getBounds());
168                 drawable.setCallback(null);
169             }
170         }
171 
172         @Override
verifyDrawable(@onNull Drawable who)173         protected boolean verifyDrawable(@NonNull Drawable who) {
174             return super.verifyDrawable(who) || (mDrawables != null && mDrawables.contains(who));
175         }
176 
add(@onNull View child)177         public void add(@NonNull View child) {
178             if (child == null) {
179                 throw new IllegalArgumentException("view must be non-null");
180             }
181 
182             if (child.getParent() instanceof ViewGroup) {
183                 ViewGroup parent = (ViewGroup) child.getParent();
184                 if (parent != mHostView && parent.getParent() != null &&
185                         parent.mAttachInfo != null) {
186                     // Moving to different container; figure out how to position child such that
187                     // it is in the same location on the screen
188                     int[] parentLocation = new int[2];
189                     int[] hostViewLocation = new int[2];
190                     parent.getLocationOnScreen(parentLocation);
191                     mHostView.getLocationOnScreen(hostViewLocation);
192                     child.offsetLeftAndRight(parentLocation[0] - hostViewLocation[0]);
193                     child.offsetTopAndBottom(parentLocation[1] - hostViewLocation[1]);
194                 }
195                 parent.removeView(child);
196                 if (parent.getLayoutTransition() != null) {
197                     // LayoutTransition will cause the child to delay removal - cancel it
198                     parent.getLayoutTransition().cancel(LayoutTransition.DISAPPEARING);
199                 }
200                 // fail-safe if view is still attached for any reason
201                 if (child.getParent() != null) {
202                     child.mParent = null;
203                 }
204             }
205             super.addView(child);
206         }
207 
remove(@onNull View view)208         public void remove(@NonNull View view) {
209             if (view == null) {
210                 throw new IllegalArgumentException("view must be non-null");
211             }
212 
213             super.removeView(view);
214         }
215 
clear()216         public void clear() {
217             removeAllViews();
218             if (mDrawables != null) {
219                 for (Drawable drawable : mDrawables) {
220                     drawable.setCallback(null);
221                 }
222                 mDrawables.clear();
223             }
224         }
225 
isEmpty()226         boolean isEmpty() {
227             if (getChildCount() == 0 &&
228                     (mDrawables == null || mDrawables.size() == 0)) {
229                 return true;
230             }
231             return false;
232         }
233 
234         @Override
invalidateDrawable(@onNull Drawable drawable)235         public void invalidateDrawable(@NonNull Drawable drawable) {
236             invalidate(drawable.getBounds());
237         }
238 
239         @Override
dispatchDraw(Canvas canvas)240         protected void dispatchDraw(Canvas canvas) {
241             /*
242              * The OverlayViewGroup doesn't draw with a DisplayList, because
243              * draw(Canvas, View, long) is never called on it. This is fine, since it doesn't need
244              * RenderNode/DisplayList features, and can just draw into the owner's Canvas.
245              *
246              * This means that we need to insert reorder barriers manually though, so that children
247              * of the OverlayViewGroup can cast shadows and Z reorder with each other.
248              */
249             canvas.insertReorderBarrier();
250 
251             super.dispatchDraw(canvas);
252 
253             canvas.insertInorderBarrier();
254             final int numDrawables = (mDrawables == null) ? 0 : mDrawables.size();
255             for (int i = 0; i < numDrawables; ++i) {
256                 mDrawables.get(i).draw(canvas);
257             }
258         }
259 
260         @Override
onLayout(boolean changed, int l, int t, int r, int b)261         protected void onLayout(boolean changed, int l, int t, int r, int b) {
262             // Noop: children are positioned absolutely
263         }
264 
265         /*
266          The following invalidation overrides exist for the purpose of redirecting invalidation to
267          the host view. The overlay is not parented to the host view (since a View cannot be a
268          parent), so the invalidation cannot proceed through the normal parent hierarchy.
269          There is a built-in assumption that the overlay exactly covers the host view, therefore
270          the invalidation rectangles received do not need to be adjusted when forwarded to
271          the host view.
272          */
273 
274         @Override
invalidate(Rect dirty)275         public void invalidate(Rect dirty) {
276             super.invalidate(dirty);
277             if (mHostView != null) {
278                 mHostView.invalidate(dirty);
279             }
280         }
281 
282         @Override
invalidate(int l, int t, int r, int b)283         public void invalidate(int l, int t, int r, int b) {
284             super.invalidate(l, t, r, b);
285             if (mHostView != null) {
286                 mHostView.invalidate(l, t, r, b);
287             }
288         }
289 
290         @Override
invalidate()291         public void invalidate() {
292             super.invalidate();
293             if (mHostView != null) {
294                 mHostView.invalidate();
295             }
296         }
297 
298         /** @hide */
299         @Override
invalidate(boolean invalidateCache)300         public void invalidate(boolean invalidateCache) {
301             super.invalidate(invalidateCache);
302             if (mHostView != null) {
303                 mHostView.invalidate(invalidateCache);
304             }
305         }
306 
307         @Override
invalidateViewProperty(boolean invalidateParent, boolean forceRedraw)308         void invalidateViewProperty(boolean invalidateParent, boolean forceRedraw) {
309             super.invalidateViewProperty(invalidateParent, forceRedraw);
310             if (mHostView != null) {
311                 mHostView.invalidateViewProperty(invalidateParent, forceRedraw);
312             }
313         }
314 
315         @Override
invalidateParentCaches()316         protected void invalidateParentCaches() {
317             super.invalidateParentCaches();
318             if (mHostView != null) {
319                 mHostView.invalidateParentCaches();
320             }
321         }
322 
323         @Override
invalidateParentIfNeeded()324         protected void invalidateParentIfNeeded() {
325             super.invalidateParentIfNeeded();
326             if (mHostView != null) {
327                 mHostView.invalidateParentIfNeeded();
328             }
329         }
330 
331         @Override
onDescendantInvalidated(@onNull View child, @NonNull View target)332         public void onDescendantInvalidated(@NonNull View child, @NonNull View target) {
333             if (mHostView != null) {
334                 if (mHostView instanceof ViewGroup) {
335                     // Propagate invalidate through the host...
336                     ((ViewGroup) mHostView).onDescendantInvalidated(mHostView, target);
337 
338                     // ...and also this view, since it will hold the descendant, and must later
339                     // propagate the calls to update display lists if dirty
340                     super.onDescendantInvalidated(child, target);
341                 } else {
342                     // Can't use onDescendantInvalidated because host isn't a ViewGroup - fall back
343                     // to invalidating.
344                     invalidate();
345                 }
346             }
347         }
348 
349         @Override
invalidateChildInParent(int[] location, Rect dirty)350         public ViewParent invalidateChildInParent(int[] location, Rect dirty) {
351             if (mHostView != null) {
352                 dirty.offset(location[0], location[1]);
353                 if (mHostView instanceof ViewGroup) {
354                     location[0] = 0;
355                     location[1] = 0;
356                     super.invalidateChildInParent(location, dirty);
357                     return ((ViewGroup) mHostView).invalidateChildInParent(location, dirty);
358                 } else {
359                     invalidate(dirty);
360                 }
361             }
362             return null;
363         }
364     }
365 
366 }
367