1 /* 2 * Copyright (C) 2019 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 android.annotation.NonNull; 20 import android.annotation.Nullable; 21 import android.graphics.Rect; 22 23 import com.android.internal.util.Preconditions; 24 25 import java.lang.ref.WeakReference; 26 import java.util.ArrayList; 27 import java.util.Collections; 28 import java.util.Iterator; 29 import java.util.List; 30 31 /** 32 * Used by {@link ViewRootImpl} to track system gesture exclusion rects reported by views. 33 */ 34 class GestureExclusionTracker { 35 private boolean mGestureExclusionViewsChanged = false; 36 private boolean mRootGestureExclusionRectsChanged = false; 37 private List<Rect> mRootGestureExclusionRects = Collections.emptyList(); 38 private List<GestureExclusionViewInfo> mGestureExclusionViewInfos = new ArrayList<>(); 39 private List<Rect> mGestureExclusionRects = Collections.emptyList(); 40 updateRectsForView(@onNull View view)41 public void updateRectsForView(@NonNull View view) { 42 boolean found = false; 43 final Iterator<GestureExclusionViewInfo> i = mGestureExclusionViewInfos.iterator(); 44 while (i.hasNext()) { 45 final GestureExclusionViewInfo info = i.next(); 46 final View v = info.getView(); 47 if (v == null || !v.isAttachedToWindow()) { 48 mGestureExclusionViewsChanged = true; 49 i.remove(); 50 continue; 51 } 52 if (v == view) { 53 found = true; 54 info.mDirty = true; 55 break; 56 } 57 } 58 if (!found && view.isAttachedToWindow()) { 59 mGestureExclusionViewInfos.add(new GestureExclusionViewInfo(view)); 60 mGestureExclusionViewsChanged = true; 61 } 62 } 63 64 @Nullable computeChangedRects()65 public List<Rect> computeChangedRects() { 66 boolean changed = mRootGestureExclusionRectsChanged; 67 final Iterator<GestureExclusionViewInfo> i = mGestureExclusionViewInfos.iterator(); 68 final List<Rect> rects = new ArrayList<>(mRootGestureExclusionRects); 69 while (i.hasNext()) { 70 final GestureExclusionViewInfo info = i.next(); 71 switch (info.update()) { 72 case GestureExclusionViewInfo.CHANGED: 73 changed = true; 74 // Deliberate fall-through 75 case GestureExclusionViewInfo.UNCHANGED: 76 rects.addAll(info.mExclusionRects); 77 break; 78 case GestureExclusionViewInfo.GONE: 79 mGestureExclusionViewsChanged = true; 80 i.remove(); 81 break; 82 } 83 } 84 if (changed || mGestureExclusionViewsChanged) { 85 mGestureExclusionViewsChanged = false; 86 mRootGestureExclusionRectsChanged = false; 87 if (!mGestureExclusionRects.equals(rects)) { 88 mGestureExclusionRects = rects; 89 return rects; 90 } 91 } 92 return null; 93 } 94 setRootSystemGestureExclusionRects(@onNull List<Rect> rects)95 public void setRootSystemGestureExclusionRects(@NonNull List<Rect> rects) { 96 Preconditions.checkNotNull(rects, "rects must not be null"); 97 mRootGestureExclusionRects = rects; 98 mRootGestureExclusionRectsChanged = true; 99 } 100 101 @NonNull getRootSystemGestureExclusionRects()102 public List<Rect> getRootSystemGestureExclusionRects() { 103 return mRootGestureExclusionRects; 104 } 105 106 private static class GestureExclusionViewInfo { 107 public static final int CHANGED = 0; 108 public static final int UNCHANGED = 1; 109 public static final int GONE = 2; 110 111 private final WeakReference<View> mView; 112 boolean mDirty = true; 113 List<Rect> mExclusionRects = Collections.emptyList(); 114 GestureExclusionViewInfo(View view)115 GestureExclusionViewInfo(View view) { 116 mView = new WeakReference<>(view); 117 } 118 getView()119 public View getView() { 120 return mView.get(); 121 } 122 update()123 public int update() { 124 final View excludedView = getView(); 125 if (excludedView == null || !excludedView.isAttachedToWindow()) return GONE; 126 final List<Rect> localRects = excludedView.getSystemGestureExclusionRects(); 127 final List<Rect> newRects = new ArrayList<>(localRects.size()); 128 for (Rect src : localRects) { 129 Rect mappedRect = new Rect(src); 130 ViewParent p = excludedView.getParent(); 131 if (p != null && p.getChildVisibleRect(excludedView, mappedRect, null)) { 132 newRects.add(mappedRect); 133 } 134 } 135 136 if (mExclusionRects.equals(localRects)) return UNCHANGED; 137 mExclusionRects = newRects; 138 return CHANGED; 139 } 140 } 141 } 142