blob: 9418aaf0c826ad066ceda15688961edf24f811a7 [file] [log] [blame]
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001/*
2 * Copyright (C) 2006 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
17package android.view;
18
Jorim Jaggic39c7b02016-03-24 10:47:07 -070019import static android.view.WindowCallbacks.RESIZE_MODE_DOCKED_DIVIDER;
20import static android.view.WindowCallbacks.RESIZE_MODE_FREEFORM;
Filip Gruszczynski1a4dfe52015-11-15 10:58:57 -080021import static android.view.WindowManager.LayoutParams.PRIVATE_FLAG_FORCE_DECOR_VIEW_VISIBILITY;
Jorim Jaggi2e95a482016-01-14 17:36:55 -080022import static android.view.WindowManager.LayoutParams.TYPE_DOCK_DIVIDER;
Chong Zhangf6525ce2016-01-14 17:09:56 -080023import static android.view.WindowManager.LayoutParams.TYPE_INPUT_METHOD;
24import static android.view.WindowManager.LayoutParams.TYPE_STATUS_BAR_PANEL;
25import static android.view.WindowManager.LayoutParams.TYPE_VOLUME_OVERLAY;
Filip Gruszczynski1a4dfe52015-11-15 10:58:57 -080026
Romain Guy6b7bd242010-10-06 19:49:23 -070027import android.Manifest;
Chet Haasecca2c982011-05-20 14:34:18 -070028import android.animation.LayoutTransition;
Romain Guy6b7bd242010-10-06 19:49:23 -070029import android.app.ActivityManagerNative;
30import android.content.ClipDescription;
31import android.content.ComponentCallbacks;
32import android.content.Context;
33import android.content.pm.PackageManager;
34import android.content.res.CompatibilityInfo;
35import android.content.res.Configuration;
36import android.content.res.Resources;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080037import android.graphics.Canvas;
Alan Viverettefed3f722013-11-14 14:48:20 -080038import android.graphics.Matrix;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080039import android.graphics.PixelFormat;
Christopher Tate2c095f32010-10-04 14:13:40 -070040import android.graphics.Point;
Christopher Tatea53146c2010-09-07 11:57:52 -070041import android.graphics.PointF;
Romain Guy6b7bd242010-10-06 19:49:23 -070042import android.graphics.PorterDuff;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080043import android.graphics.Rect;
44import android.graphics.Region;
Svetoslav Ganov42138042012-03-20 11:51:39 -070045import android.graphics.drawable.Drawable;
Jeff Brownd912e1f2014-04-11 18:46:22 -070046import android.hardware.display.DisplayManager;
47import android.hardware.display.DisplayManager.DisplayListener;
Jun Mukai347e5d42015-12-03 01:13:31 -080048import android.hardware.input.InputManager;
Romain Guy6b7bd242010-10-06 19:49:23 -070049import android.media.AudioManager;
50import android.os.Binder;
Michael Wright5bd69e62015-05-14 14:48:08 +010051import android.os.Build;
Romain Guy6b7bd242010-10-06 19:49:23 -070052import android.os.Bundle;
53import android.os.Debug;
54import android.os.Handler;
Romain Guy6b7bd242010-10-06 19:49:23 -070055import android.os.Looper;
56import android.os.Message;
57import android.os.ParcelFileDescriptor;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080058import android.os.Process;
Romain Guy6b7bd242010-10-06 19:49:23 -070059import android.os.RemoteException;
Romain Guy6b7bd242010-10-06 19:49:23 -070060import android.os.SystemClock;
Romain Guy59a12ca2011-06-09 17:48:21 -070061import android.os.SystemProperties;
Jeff Brown481c1572012-03-09 14:41:15 -080062import android.os.Trace;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080063import android.util.AndroidRuntimeException;
Mitsuru Oshima9189cab2009-06-03 11:19:12 -070064import android.util.DisplayMetrics;
Romain Guy6b7bd242010-10-06 19:49:23 -070065import android.util.Log;
Chet Haase949dbf72010-08-11 18:41:06 -070066import android.util.Slog;
John Reckba6adf62015-02-19 14:36:50 -080067import android.util.TimeUtils;
Dianne Hackborn711e62a2010-11-29 16:38:22 -080068import android.util.TypedValue;
John Reck44fd8d22014-02-26 11:00:11 -080069import android.view.Surface.OutOfResourcesException;
Jeff Browna175a5b2012-02-15 19:18:31 -080070import android.view.View.AttachInfo;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080071import android.view.View.MeasureSpec;
svetoslavganov75986cf2009-05-14 22:28:01 -070072import android.view.accessibility.AccessibilityEvent;
73import android.view.accessibility.AccessibilityManager;
Svetoslav Ganov8643aa02011-04-20 12:12:33 -070074import android.view.accessibility.AccessibilityManager.AccessibilityStateChangeListener;
Chris Craikcce47eb2014-07-16 15:12:15 -070075import android.view.accessibility.AccessibilityManager.HighTextContrastChangeListener;
Svetoslav Ganov8643aa02011-04-20 12:12:33 -070076import android.view.accessibility.AccessibilityNodeInfo;
Alan Viverette25acc7e2015-05-19 11:32:08 -070077import android.view.accessibility.AccessibilityNodeInfo.AccessibilityAction;
Svetoslav Ganov02107852011-10-03 17:06:56 -070078import android.view.accessibility.AccessibilityNodeProvider;
Svetoslav Ganov8643aa02011-04-20 12:12:33 -070079import android.view.accessibility.IAccessibilityInteractionConnection;
80import android.view.accessibility.IAccessibilityInteractionConnectionCallback;
Dianne Hackborn0f761d62010-11-30 22:06:10 -080081import android.view.animation.AccelerateDecelerateInterpolator;
82import android.view.animation.Interpolator;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080083import android.view.inputmethod.InputConnection;
84import android.view.inputmethod.InputMethodManager;
85import android.widget.Scroller;
Svetoslav Ganov8643aa02011-04-20 12:12:33 -070086
Svetoslav Ganov42138042012-03-20 11:51:39 -070087import com.android.internal.R;
Clara Bayarri75e09792015-07-29 16:20:40 +010088import com.android.internal.os.IResultReceiver;
Svetoslav Ganov758143e2012-08-06 16:40:27 -070089import com.android.internal.os.SomeArgs;
Adam Powell6711f3b2015-05-06 15:57:09 -070090import com.android.internal.policy.PhoneFallbackEventHandler;
Romain Guy6b7bd242010-10-06 19:49:23 -070091import com.android.internal.view.BaseSurfaceHolder;
Romain Guy6b7bd242010-10-06 19:49:23 -070092import com.android.internal.view.RootViewSurfaceTaker;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080093
Jeff Brown5182c782013-10-15 20:31:52 -070094import java.io.FileDescriptor;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080095import java.io.IOException;
96import java.io.OutputStream;
Jeff Brown5182c782013-10-15 20:31:52 -070097import java.io.PrintWriter;
Romain Guy6b7bd242010-10-06 19:49:23 -070098import java.lang.ref.WeakReference;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080099import java.util.ArrayList;
Svetoslav Ganov42138042012-03-20 11:51:39 -0700100import java.util.HashSet;
Jorim Jaggic39c7b02016-03-24 10:47:07 -0700101import java.util.concurrent.CountDownLatch;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800102
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800103/**
104 * The top of a view hierarchy, implementing the needed protocol between View
105 * and the WindowManager. This is for the most part an internal implementation
Jeff Brown98365d72012-08-19 20:30:52 -0700106 * detail of {@link WindowManagerGlobal}.
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800107 *
108 * {@hide}
109 */
Romain Guy812ccbe2010-06-01 14:07:24 -0700110@SuppressWarnings({"EmptyCatchBlock", "PointlessBooleanExpression"})
Jeff Browna175a5b2012-02-15 19:18:31 -0800111public final class ViewRootImpl implements ViewParent,
John Reck51aaf902015-12-02 15:08:07 -0800112 View.AttachInfo.Callbacks, ThreadedRenderer.HardwareDrawCallbacks {
Dianne Hackborn70a3f672011-08-08 14:32:41 -0700113 private static final String TAG = "ViewRootImpl";
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800114 private static final boolean DBG = false;
Romain Guy812ccbe2010-06-01 14:07:24 -0700115 private static final boolean LOCAL_LOGV = false;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800116 /** @noinspection PointlessBooleanExpression*/
117 private static final boolean DEBUG_DRAW = false || LOCAL_LOGV;
118 private static final boolean DEBUG_LAYOUT = false || LOCAL_LOGV;
Dianne Hackborn711e62a2010-11-29 16:38:22 -0800119 private static final boolean DEBUG_DIALOG = false || LOCAL_LOGV;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800120 private static final boolean DEBUG_INPUT_RESIZE = false || LOCAL_LOGV;
121 private static final boolean DEBUG_ORIENTATION = false || LOCAL_LOGV;
122 private static final boolean DEBUG_TRACKBALL = false || LOCAL_LOGV;
123 private static final boolean DEBUG_IMF = false || LOCAL_LOGV;
Dianne Hackborn694f79b2010-03-17 19:44:59 -0700124 private static final boolean DEBUG_CONFIGURATION = false || LOCAL_LOGV;
Chet Haase2f2022a2011-10-11 06:41:59 -0700125 private static final boolean DEBUG_FPS = false;
Michael Wright06a79252014-05-05 17:45:29 -0700126 private static final boolean DEBUG_INPUT_STAGES = false || LOCAL_LOGV;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800127
Romain Guy59a12ca2011-06-09 17:48:21 -0700128 /**
Skuhneb8160872015-09-22 09:51:39 -0700129 * Set to false if we do not want to use the multi threaded renderer. Note that by disabling
130 * this, WindowCallbacks will not fire.
131 */
132 private static final boolean USE_MT_RENDERER = true;
133
134 /**
Romain Guy59a12ca2011-06-09 17:48:21 -0700135 * Set this system property to true to force the view hierarchy to render
136 * at 60 Hz. This can be used to measure the potential framerate.
137 */
Romain Guye9bc11f2013-05-23 12:47:26 -0700138 private static final String PROPERTY_PROFILE_RENDERING = "viewroot.profile_rendering";
Michael Chan53071d62009-05-13 17:29:48 -0700139
Griff Hazena0938022015-03-13 10:01:41 -0700140 // properties used by emulator to determine display shape
Griff Hazena0938022015-03-13 10:01:41 -0700141 public static final String PROPERTY_EMULATOR_WIN_OUTSET_BOTTOM_PX =
142 "ro.emu.win_outset_bottom_px";
Michael Kolb437d3132014-06-20 13:28:44 -0700143
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800144 /**
145 * Maximum time we allow the user to roll the trackball enough to generate
146 * a key event, before resetting the counters.
147 */
148 static final int MAX_TRACKBALL_DELAY = 250;
149
Alan Viverettebea0c7da2015-09-01 16:00:20 -0400150 static final ThreadLocal<HandlerActionQueue> sRunQueues = new ThreadLocal<HandlerActionQueue>();
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800151
Skuhneb8160872015-09-22 09:51:39 -0700152 static final ArrayList<Runnable> sFirstDrawHandlers = new ArrayList();
Dianne Hackborn2a9094d2010-02-03 19:20:09 -0800153 static boolean sFirstDrawComplete = false;
Craig Mautner8f303ad2013-06-14 11:32:22 -0700154
Skuhneb8160872015-09-22 09:51:39 -0700155 static final ArrayList<ComponentCallbacks> sConfigCallbacks = new ArrayList();
Romain Guy59a12ca2011-06-09 17:48:21 -0700156
Chong Zhangdcee1de2015-10-06 10:26:00 -0700157 final ArrayList<WindowCallbacks> mWindowCallbacks = new ArrayList();
Jeff Brownf9e989d2013-04-04 23:04:03 -0700158 final Context mContext;
Jeff Brown98365d72012-08-19 20:30:52 -0700159 final IWindowSession mWindowSession;
160 final Display mDisplay;
Jeff Brownd912e1f2014-04-11 18:46:22 -0700161 final DisplayManager mDisplayManager;
Dianne Hackbornc2293022013-02-06 23:14:49 -0800162 final String mBasePackageName;
Jeff Brown98365d72012-08-19 20:30:52 -0700163
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800164 final int[] mTmpLocation = new int[2];
Romain Guy8506ab42009-06-11 17:35:47 -0700165
Dianne Hackborn711e62a2010-11-29 16:38:22 -0800166 final TypedValue mTmpValue = new TypedValue();
Jeff Brownf9e989d2013-04-04 23:04:03 -0700167
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800168 final Thread mThread;
169
170 final WindowLeaked mLocation;
171
172 final WindowManager.LayoutParams mWindowAttributes = new WindowManager.LayoutParams();
173
174 final W mWindow;
175
Dianne Hackborn180c4842011-09-13 12:39:25 -0700176 final int mTargetSdkVersion;
177
Dianne Hackborn9a230e02011-10-06 11:51:27 -0700178 int mSeq;
179
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800180 View mView;
Svetoslav Ganov42138042012-03-20 11:51:39 -0700181
182 View mAccessibilityFocusedHost;
183 AccessibilityNodeInfo mAccessibilityFocusedVirtualView;
184
Jun Mukai347e5d42015-12-03 01:13:31 -0800185 // The view which captures mouse input, or null when no one is capturing.
186 View mCapturingView;
187
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800188 int mViewVisibility;
189 boolean mAppVisible = true;
Filip Gruszczynski1a4dfe52015-11-15 10:58:57 -0800190 // For recents to freeform transition we need to keep drawing after the app receives information
191 // that it became invisible. This will ignore that information and depend on the decor view
192 // visibility to control drawing. The decor view visibility will get adjusted when the app get
193 // stopped and that's when the app will stop drawing further frames.
194 private boolean mForceDecorViewVisibility = false;
Dianne Hackborn180c4842011-09-13 12:39:25 -0700195 int mOrigWindowType = -1;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800196
Alan Viverette64bf97a2015-09-18 16:42:00 -0400197 /** Whether the window had focus during the most recent traversal. */
198 boolean mHadWindowFocus;
199
200 /**
201 * Whether the window lost focus during a previous traversal and has not
202 * yet gained it back. Used to determine whether a WINDOW_STATE_CHANGE
203 * accessibility events should be sent during traversal.
204 */
205 boolean mLostWindowFocus;
206
Dianne Hackbornce418e62011-03-01 14:31:38 -0800207 // Set to true if the owner of this window is in the stopped state,
208 // so the window should no longer be active.
209 boolean mStopped = false;
Craig Mautner8f303ad2013-06-14 11:32:22 -0700210
Daniel Koulomzin087ae472015-12-16 17:52:25 -0500211 // Set to true if the owner of this window is in ambient mode,
212 // which means it won't receive input events.
213 boolean mIsAmbientMode = false;
214
George Mount41725de2015-04-09 08:23:05 -0700215 // Set to true to stop input during an Activity Transition.
216 boolean mPausedForTransition = false;
217
Dianne Hackborn5fd21692011-06-07 14:09:47 -0700218 boolean mLastInCompatMode = false;
219
Dianne Hackbornd76b67c2010-07-13 17:48:30 -0700220 SurfaceHolder.Callback2 mSurfaceHolderCallback;
Dianne Hackborndc8a7f62010-05-10 11:29:34 -0700221 BaseSurfaceHolder mSurfaceHolder;
222 boolean mIsCreating;
223 boolean mDrawingAllowed;
Craig Mautner8f303ad2013-06-14 11:32:22 -0700224
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800225 final Region mTransparentRegion;
226 final Region mPreviousTransparentRegion;
227
228 int mWidth;
229 int mHeight;
Romain Guy7d7b5492011-01-24 16:33:45 -0800230 Rect mDirty;
Romain Guybb93d552009-03-24 21:04:15 -0700231 boolean mIsAnimating;
Romain Guy8506ab42009-06-11 17:35:47 -0700232
Chong Zhang0275e392015-09-17 10:41:44 -0700233 private boolean mDragResizing;
Jorim Jaggic39c7b02016-03-24 10:47:07 -0700234 private boolean mInvalidateRootRequested;
Jorim Jaggi4846ee32016-01-07 17:39:12 +0100235 private int mResizeMode;
Chong Zhang0275e392015-09-17 10:41:44 -0700236 private int mCanvasOffsetX;
237 private int mCanvasOffsetY;
Jorim Jaggi4846ee32016-01-07 17:39:12 +0100238 private boolean mActivityRelaunched;
Chong Zhang0275e392015-09-17 10:41:44 -0700239
Mitsuru Oshima64f59342009-06-21 00:03:11 -0700240 CompatibilityInfo.Translator mTranslator;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800241
242 final View.AttachInfo mAttachInfo;
Jeff Brown46b9ac02010-04-22 18:58:52 -0700243 InputChannel mInputChannel;
Dianne Hackborn1e4b9f32010-06-23 14:10:57 -0700244 InputQueue.Callback mInputQueueCallback;
245 InputQueue mInputQueue;
Joe Onorato86f67862010-11-05 18:57:34 -0700246 FallbackEventHandler mFallbackEventHandler;
Jeff Brown96e942d2011-11-30 19:55:01 -0800247 Choreographer mChoreographer;
Igor Murashkina86ab6402013-08-30 12:58:36 -0700248
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800249 final Rect mTempRect; // used in the transaction to not thrash the heap.
250 final Rect mVisRect; // used to retrieve visible rect of focused view.
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800251
252 boolean mTraversalScheduled;
Jeff Browne0dbd002012-02-15 19:34:58 -0800253 int mTraversalBarrier;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800254 boolean mWillDrawSoon;
Craig Mautnerdf2390a2012-08-29 20:59:22 -0700255 /** Set to true while in performTraversals for detecting when die(true) is called from internal
256 * callbacks such as onMeasure, onPreDraw, onDraw and deferring doDie() until later. */
257 boolean mIsInTraversal;
Adrian Roosfa104232014-06-20 16:10:14 -0700258 boolean mApplyInsetsRequested;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800259 boolean mLayoutRequested;
260 boolean mFirst;
261 boolean mReportNextDraw;
262 boolean mFullRedrawNeeded;
263 boolean mNewSurfaceNeeded;
264 boolean mHasHadWindowFocus;
265 boolean mLastWasImTarget;
Jorim Jaggia4a58ef2016-01-27 02:10:08 -0800266 boolean mForceNextWindowRelayout;
Chong Zhang8dbd9ad2015-10-09 10:06:11 -0700267 CountDownLatch mWindowDrawCountDown;
Jorim Jaggi827e0fa2015-05-07 11:41:41 -0700268
Romain Guy1f59e5c2012-05-06 14:11:16 -0700269 boolean mIsDrawing;
Dianne Hackborn3a3a6cf2012-03-26 10:24:04 -0700270 int mLastSystemUiVisibility;
Dianne Hackborn9d090892012-06-11 18:35:41 -0700271 int mClientWindowLayoutFlags;
Dianne Hackbornc4aad012013-02-22 15:05:25 -0800272 boolean mLastOverscanRequested;
Jeff Brown4952dfd2011-11-30 19:23:22 -0800273
274 // Pool of queued input events.
275 private static final int MAX_QUEUED_INPUT_EVENT_POOL_SIZE = 10;
276 private QueuedInputEvent mQueuedInputEventPool;
277 private int mQueuedInputEventPoolSize;
Jeff Brown4952dfd2011-11-30 19:23:22 -0800278
Michael Wrightc8a7e542013-03-20 17:58:33 -0700279 /* Input event queue.
Jeff Brownf9e989d2013-04-04 23:04:03 -0700280 * Pending input events are input events waiting to be delivered to the input stages
281 * and handled by the application.
Michael Wrightc8a7e542013-03-20 17:58:33 -0700282 */
283 QueuedInputEvent mPendingInputEventHead;
284 QueuedInputEvent mPendingInputEventTail;
Michael Wright95ae9422013-03-14 10:58:50 -0700285 int mPendingInputEventCount;
Jeff Brown96e942d2011-11-30 19:55:01 -0800286 boolean mProcessInputEventsScheduled;
Michael Wright9d744c72014-02-18 21:27:42 -0800287 boolean mUnbufferedInputDispatch;
Michael Wright95ae9422013-03-14 10:58:50 -0700288 String mPendingInputEventQueueLengthCounterName = "pq";
Jeff Brownf9e989d2013-04-04 23:04:03 -0700289
290 InputStage mFirstInputStage;
291 InputStage mFirstPostImeInputStage;
Michael Wright899d7052014-04-23 17:23:39 -0700292 InputStage mSyntheticInputStage;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800293
294 boolean mWindowAttributesChanged = false;
Romain Guyf21c9b02011-09-06 16:56:54 -0700295 int mWindowAttributesChangesFlag = 0;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800296
297 // These can be accessed by any thread, must be protected with a lock.
Mathias Agopian5583dc62009-07-09 16:28:11 -0700298 // Surface can never be reassigned or cleared (use Surface.clear()).
John Reckb13de072014-11-19 16:33:47 -0800299 final Surface mSurface = new Surface();
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800300
301 boolean mAdded;
302 boolean mAddedTouchMode;
303
Craig Mautner48d0d182013-06-11 07:53:06 -0700304 final DisplayAdjustments mDisplayAdjustments;
Dianne Hackborn5fd21692011-06-07 14:09:47 -0700305
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800306 // These are accessed by multiple threads.
307 final Rect mWinFrame; // frame given by window manager.
308
Dianne Hackbornc4aad012013-02-22 15:05:25 -0800309 final Rect mPendingOverscanInsets = new Rect();
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800310 final Rect mPendingVisibleInsets = new Rect();
Adrian Roosfa104232014-06-20 16:10:14 -0700311 final Rect mPendingStableInsets = new Rect();
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800312 final Rect mPendingContentInsets = new Rect();
Filip Gruszczynski2217f612015-05-26 11:32:08 -0700313 final Rect mPendingOutsets = new Rect();
Jorim Jaggia7262a82015-11-03 15:15:40 +0100314 final Rect mPendingBackDropFrame = new Rect();
Jorim Jaggi0ffd49c2016-02-12 15:04:21 -0800315 boolean mPendingAlwaysConsumeNavBar;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800316 final ViewTreeObserver.InternalInsetsInfo mLastGivenInsets
317 = new ViewTreeObserver.InternalInsetsInfo();
318
Adrian Roosfa104232014-06-20 16:10:14 -0700319 final Rect mDispatchContentInsets = new Rect();
320 final Rect mDispatchStableInsets = new Rect();
Dianne Hackborn3a3a6cf2012-03-26 10:24:04 -0700321
Filip Gruszczynski954289d2015-02-26 15:46:47 -0800322 private WindowInsets mLastWindowInsets;
323
Dianne Hackborn694f79b2010-03-17 19:44:59 -0700324 final Configuration mLastConfiguration = new Configuration();
325 final Configuration mPendingConfiguration = new Configuration();
Romain Guy59a12ca2011-06-09 17:48:21 -0700326
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800327 boolean mScrollMayChange;
328 int mSoftInputMode;
Svetoslav Ganov149567f2013-01-08 15:23:34 -0800329 WeakReference<View> mLastScrolledFocus;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800330 int mScrollY;
331 int mCurScrollY;
332 Scroller mScroller;
John Recke56e9df2014-02-21 15:45:10 -0800333 static final Interpolator mResizeInterpolator = new AccelerateDecelerateInterpolator();
Chet Haasecca2c982011-05-20 14:34:18 -0700334 private ArrayList<LayoutTransition> mPendingTransitions;
Romain Guy8506ab42009-06-11 17:35:47 -0700335
Romain Guy8506ab42009-06-11 17:35:47 -0700336 final ViewConfiguration mViewConfiguration;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800337
Christopher Tatea53146c2010-09-07 11:57:52 -0700338 /* Drag/drop */
339 ClipDescription mDragDescription;
340 View mCurrentDragView;
Christopher Tate7fb8b562011-01-20 13:46:41 -0800341 volatile Object mLocalDragState;
Christopher Tatea53146c2010-09-07 11:57:52 -0700342 final PointF mDragPoint = new PointF();
Christopher Tate2c095f32010-10-04 14:13:40 -0700343 final PointF mLastTouchPoint = new PointF();
Vladislav Kaznacheevba761122016-01-22 12:09:45 -0800344 int mLastTouchSource;
Igor Murashkina86ab6402013-08-30 12:58:36 -0700345
346 private boolean mProfileRendering;
Romain Guyd0750312012-11-29 11:34:43 -0800347 private Choreographer.FrameCallback mRenderProfiler;
348 private boolean mRenderProfilingEnabled;
Christopher Tatea53146c2010-09-07 11:57:52 -0700349
Chet Haase2f2022a2011-10-11 06:41:59 -0700350 // Variables to track frames per second, enabled via DEBUG_FPS flag
351 private long mFpsStartTime = -1;
352 private long mFpsPrevTime = -1;
353 private int mFpsNumFrames;
354
Jun Mukai1db53972015-09-11 18:08:31 -0700355 private int mPointerIconShape = PointerIcon.STYLE_NOT_SPECIFIED;
Jun Mukaid4eaef72015-10-30 15:54:33 -0700356 private PointerIcon mCustomPointerIcon = null;
Jun Mukai1db53972015-09-11 18:08:31 -0700357
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800358 /**
359 * see {@link #playSoundEffect(int)}
360 */
361 AudioManager mAudioManager;
362
Svetoslav Ganov8643aa02011-04-20 12:12:33 -0700363 final AccessibilityManager mAccessibilityManager;
364
Gilles Debunne5ac84422011-10-19 09:35:58 -0700365 AccessibilityInteractionController mAccessibilityInteractionController;
Svetoslav Ganov8643aa02011-04-20 12:12:33 -0700366
367 AccessibilityInteractionConnectionManager mAccessibilityInteractionConnectionManager;
Chris Craikcce47eb2014-07-16 15:12:15 -0700368 HighContrastTextManager mHighContrastTextManager;
Svetoslav Ganov8643aa02011-04-20 12:12:33 -0700369
Svetoslav Ganova0156172011-06-26 17:55:44 -0700370 SendWindowContentChangedAccessibilityEvent mSendWindowContentChangedAccessibilityEvent;
Svetoslav Ganoveeee4d22011-06-10 20:51:30 -0700371
Svetoslav Ganov42138042012-03-20 11:51:39 -0700372 HashSet<View> mTempHashSet;
Svetoslav Ganov79311c42012-01-17 20:24:26 -0800373
Dianne Hackborn11ea3342009-07-22 21:48:55 -0700374 private final int mDensity;
Dianne Hackborn908aecc2012-07-31 16:37:34 -0700375 private final int mNoncompatDensity;
Adam Powellb08013c2010-09-16 16:28:11 -0700376
Chet Haase97140572012-09-13 14:56:47 -0700377 private boolean mInLayout = false;
378 ArrayList<View> mLayoutRequesters = new ArrayList<View>();
379 boolean mHandlingLayoutInLayoutRequest = false;
380
Fabrice Di Megliob003e282012-10-17 17:20:19 -0700381 private int mViewLayoutDirectionInitial;
Chet Haase97140572012-09-13 14:56:47 -0700382
Craig Mautner8f303ad2013-06-14 11:32:22 -0700383 /** Set to true once doDie() has been called. */
384 private boolean mRemoved;
385
Jeff Brown21bc5c92011-02-28 18:27:14 -0800386 /**
387 * Consistency verifier for debugging purposes.
388 */
389 protected final InputEventConsistencyVerifier mInputEventConsistencyVerifier =
390 InputEventConsistencyVerifier.isInstrumentationEnabled() ?
391 new InputEventConsistencyVerifier(this, 0) : null;
392
Dianne Hackborn9a230e02011-10-06 11:51:27 -0700393 static final class SystemUiVisibilityInfo {
394 int seq;
395 int globalVisibility;
396 int localValue;
397 int localChanges;
398 }
Igor Murashkina86ab6402013-08-30 12:58:36 -0700399
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -0800400 private String mTag = TAG;
401
Jeff Brown98365d72012-08-19 20:30:52 -0700402 public ViewRootImpl(Context context, Display display) {
Jeff Brownf9e989d2013-04-04 23:04:03 -0700403 mContext = context;
404 mWindowSession = WindowManagerGlobal.getWindowSession();
Jeff Brown98365d72012-08-19 20:30:52 -0700405 mDisplay = display;
Dianne Hackbornc2293022013-02-06 23:14:49 -0800406 mBasePackageName = context.getBasePackageName();
Jeff Brown98365d72012-08-19 20:30:52 -0700407
Craig Mautner48d0d182013-06-11 07:53:06 -0700408 mDisplayAdjustments = display.getDisplayAdjustments();
Svetoslav Ganov8643aa02011-04-20 12:12:33 -0700409
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800410 mThread = Thread.currentThread();
411 mLocation = new WindowLeaked(null);
412 mLocation.fillInStackTrace();
413 mWidth = -1;
414 mHeight = -1;
415 mDirty = new Rect();
416 mTempRect = new Rect();
417 mVisRect = new Rect();
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800418 mWinFrame = new Rect();
Romain Guyfb8b7632010-08-23 21:05:08 -0700419 mWindow = new W(this);
Dianne Hackborn180c4842011-09-13 12:39:25 -0700420 mTargetSdkVersion = context.getApplicationInfo().targetSdkVersion;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800421 mViewVisibility = View.GONE;
422 mTransparentRegion = new Region();
423 mPreviousTransparentRegion = new Region();
424 mFirst = true; // true for the first time the view is added
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800425 mAdded = false;
Chris Craikcce47eb2014-07-16 15:12:15 -0700426 mAttachInfo = new View.AttachInfo(mWindowSession, mWindow, display, this, mHandler, this);
Svetoslav Ganov8643aa02011-04-20 12:12:33 -0700427 mAccessibilityManager = AccessibilityManager.getInstance(context);
428 mAccessibilityInteractionConnectionManager =
429 new AccessibilityInteractionConnectionManager();
Svetoslav Ganov00d0c142012-02-24 11:30:49 -0800430 mAccessibilityManager.addAccessibilityStateChangeListener(
431 mAccessibilityInteractionConnectionManager);
Chris Craikcce47eb2014-07-16 15:12:15 -0700432 mHighContrastTextManager = new HighContrastTextManager();
433 mAccessibilityManager.addHighTextContrastStateChangeListener(
434 mHighContrastTextManager);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800435 mViewConfiguration = ViewConfiguration.get(context);
Dianne Hackborn11ea3342009-07-22 21:48:55 -0700436 mDensity = context.getResources().getDisplayMetrics().densityDpi;
Dianne Hackborn908aecc2012-07-31 16:37:34 -0700437 mNoncompatDensity = context.getResources().getDisplayMetrics().noncompatDensityDpi;
Jorim Jaggib10e33f2015-02-04 21:57:40 +0100438 mFallbackEventHandler = new PhoneFallbackEventHandler(context);
Jeff Brown96e942d2011-11-30 19:55:01 -0800439 mChoreographer = Choreographer.getInstance();
Jeff Brownd912e1f2014-04-11 18:46:22 -0700440 mDisplayManager = (DisplayManager)context.getSystemService(Context.DISPLAY_SERVICE);
Dianne Hackborna53de062012-05-08 18:53:51 -0700441 loadSystemProperties();
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800442 }
443
Dianne Hackborn2a9094d2010-02-03 19:20:09 -0800444 public static void addFirstDrawHandler(Runnable callback) {
445 synchronized (sFirstDrawHandlers) {
446 if (!sFirstDrawComplete) {
447 sFirstDrawHandlers.add(callback);
448 }
449 }
450 }
Igor Murashkina86ab6402013-08-30 12:58:36 -0700451
Dianne Hackborne36d6e22010-02-17 19:46:25 -0800452 public static void addConfigCallback(ComponentCallbacks callback) {
453 synchronized (sConfigCallbacks) {
454 sConfigCallbacks.add(callback);
455 }
456 }
Igor Murashkina86ab6402013-08-30 12:58:36 -0700457
Chong Zhangdcee1de2015-10-06 10:26:00 -0700458 public void addWindowCallbacks(WindowCallbacks callback) {
Skuhneb8160872015-09-22 09:51:39 -0700459 if (USE_MT_RENDERER) {
Chong Zhangdcee1de2015-10-06 10:26:00 -0700460 synchronized (mWindowCallbacks) {
461 mWindowCallbacks.add(callback);
Skuhneb8160872015-09-22 09:51:39 -0700462 }
463 }
464 }
465
Chong Zhangdcee1de2015-10-06 10:26:00 -0700466 public void removeWindowCallbacks(WindowCallbacks callback) {
Skuhneb8160872015-09-22 09:51:39 -0700467 if (USE_MT_RENDERER) {
Chong Zhangdcee1de2015-10-06 10:26:00 -0700468 synchronized (mWindowCallbacks) {
469 mWindowCallbacks.remove(callback);
Skuhneb8160872015-09-22 09:51:39 -0700470 }
471 }
472 }
473
Chong Zhang8dbd9ad2015-10-09 10:06:11 -0700474 public void reportDrawFinish() {
475 if (mWindowDrawCountDown != null) {
476 mWindowDrawCountDown.countDown();
477 }
478 }
479
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800480 // FIXME for perf testing only
481 private boolean mProfile = false;
482
483 /**
484 * Call this to profile the next traversal call.
485 * FIXME for perf testing only. Remove eventually
486 */
487 public void profile() {
488 mProfile = true;
489 }
490
491 /**
492 * Indicates whether we are in touch mode. Calling this method triggers an IPC
493 * call and should be avoided whenever possible.
494 *
495 * @return True, if the device is in touch mode, false otherwise.
496 *
497 * @hide
498 */
499 static boolean isInTouchMode() {
Jeff Brown98365d72012-08-19 20:30:52 -0700500 IWindowSession windowSession = WindowManagerGlobal.peekWindowSession();
501 if (windowSession != null) {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800502 try {
Jeff Brown98365d72012-08-19 20:30:52 -0700503 return windowSession.getInTouchMode();
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800504 } catch (RemoteException e) {
505 }
506 }
507 return false;
508 }
509
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800510 /**
511 * We have one child
512 */
Romain Guye4d01122010-06-16 18:44:05 -0700513 public void setView(View view, WindowManager.LayoutParams attrs, View panelParentView) {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800514 synchronized (this) {
515 if (mView == null) {
Mitsuru Oshima8169dae2009-04-28 18:12:09 -0700516 mView = view;
Jeff Brownd912e1f2014-04-11 18:46:22 -0700517
518 mAttachInfo.mDisplayState = mDisplay.getState();
519 mDisplayManager.registerDisplayListener(mDisplayListener, mHandler);
520
Fabrice Di Megliob003e282012-10-17 17:20:19 -0700521 mViewLayoutDirectionInitial = mView.getRawLayoutDirection();
Joe Onorato86f67862010-11-05 18:57:34 -0700522 mFallbackEventHandler.setView(view);
Mitsuru Oshima9189cab2009-06-03 11:19:12 -0700523 mWindowAttributes.copyFrom(attrs);
Dianne Hackbornc2293022013-02-06 23:14:49 -0800524 if (mWindowAttributes.packageName == null) {
525 mWindowAttributes.packageName = mBasePackageName;
526 }
Mitsuru Oshima1ecf5d22009-07-06 17:20:38 -0700527 attrs = mWindowAttributes;
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -0800528 setTag();
Dianne Hackborn9d090892012-06-11 18:35:41 -0700529 // Keep track of the actual window flags supplied by the client.
530 mClientWindowLayoutFlags = attrs.flags;
Svetoslav Ganov791fd312012-05-14 15:12:30 -0700531
Svetoslav Ganov45a02e02012-06-17 15:07:29 -0700532 setAccessibilityFocus(null, null);
Svetoslav Ganov791fd312012-05-14 15:12:30 -0700533
Dianne Hackborndc8a7f62010-05-10 11:29:34 -0700534 if (view instanceof RootViewSurfaceTaker) {
535 mSurfaceHolderCallback =
536 ((RootViewSurfaceTaker)view).willYouTakeTheSurface();
537 if (mSurfaceHolderCallback != null) {
538 mSurfaceHolder = new TakenSurfaceHolder();
Dianne Hackborn289b9b62010-07-09 11:44:11 -0700539 mSurfaceHolder.setFormat(PixelFormat.UNKNOWN);
Dianne Hackborndc8a7f62010-05-10 11:29:34 -0700540 }
541 }
Romain Guy1aec9a22011-01-05 09:37:12 -0800542
Alan Viverette49a22e82014-07-12 20:01:27 -0700543 // Compute surface insets required to draw at specified Z value.
544 // TODO: Use real shadow insets for a constant max Z.
Alan Viverette5435a302015-01-29 10:25:34 -0800545 if (!attrs.hasManualSurfaceInsets) {
546 final int surfaceInset = (int) Math.ceil(view.getZ() * 2);
547 attrs.surfaceInsets.set(surfaceInset, surfaceInset, surfaceInset, surfaceInset);
548 }
Alan Viverette49a22e82014-07-12 20:01:27 -0700549
Craig Mautner48d0d182013-06-11 07:53:06 -0700550 CompatibilityInfo compatibilityInfo = mDisplayAdjustments.getCompatibilityInfo();
Romain Guy856d4e12011-10-14 15:47:55 -0700551 mTranslator = compatibilityInfo.getTranslator();
552
Romain Guy1aec9a22011-01-05 09:37:12 -0800553 // If the application owns the surface, don't enable hardware acceleration
554 if (mSurfaceHolder == null) {
Romain Guy3b748a42013-04-17 18:54:38 -0700555 enableHardwareAcceleration(attrs);
Romain Guy1aec9a22011-01-05 09:37:12 -0800556 }
557
Mitsuru Oshimae5fb3282009-06-09 21:16:08 -0700558 boolean restore = false;
Romain Guy35b38ce2009-10-07 13:38:55 -0700559 if (mTranslator != null) {
Romain Guy856d4e12011-10-14 15:47:55 -0700560 mSurface.setCompatibilityTranslator(mTranslator);
Mitsuru Oshimae5fb3282009-06-09 21:16:08 -0700561 restore = true;
Mitsuru Oshima64f59342009-06-21 00:03:11 -0700562 attrs.backup();
563 mTranslator.translateWindowLayout(attrs);
Mitsuru Oshima3d914922009-05-13 22:29:15 -0700564 }
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -0800565 if (DEBUG_LAYOUT) Log.d(mTag, "WindowLayout in setView:" + attrs);
Mitsuru Oshima64f59342009-06-21 00:03:11 -0700566
Mitsuru Oshima1ecf5d22009-07-06 17:20:38 -0700567 if (!compatibilityInfo.supportsScreen()) {
Adam Lesinski95c42972013-10-02 10:13:27 -0700568 attrs.privateFlags |= WindowManager.LayoutParams.PRIVATE_FLAG_COMPATIBLE_WINDOW;
Dianne Hackborn5fd21692011-06-07 14:09:47 -0700569 mLastInCompatMode = true;
Mitsuru Oshima1ecf5d22009-07-06 17:20:38 -0700570 }
571
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800572 mSoftInputMode = attrs.softInputMode;
573 mWindowAttributesChanged = true;
Romain Guyf21c9b02011-09-06 16:56:54 -0700574 mWindowAttributesChangesFlag = WindowManager.LayoutParams.EVERYTHING_CHANGED;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800575 mAttachInfo.mRootView = view;
Romain Guy35b38ce2009-10-07 13:38:55 -0700576 mAttachInfo.mScalingRequired = mTranslator != null;
Mitsuru Oshima64f59342009-06-21 00:03:11 -0700577 mAttachInfo.mApplicationScale =
578 mTranslator == null ? 1.0f : mTranslator.applicationScale;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800579 if (panelParentView != null) {
580 mAttachInfo.mPanelParentWindowToken
581 = panelParentView.getApplicationWindowToken();
582 }
583 mAdded = true;
584 int res; /* = WindowManagerImpl.ADD_OKAY; */
Romain Guy8506ab42009-06-11 17:35:47 -0700585
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800586 // Schedule the first layout -before- adding to the window
587 // manager, to make sure we do the relayout before receiving
588 // any other events from the system.
589 requestLayout();
Jeff Browncc4f7db2011-08-30 20:34:48 -0700590 if ((mWindowAttributes.inputFeatures
591 & WindowManager.LayoutParams.INPUT_FEATURE_NO_INPUT_CHANNEL) == 0) {
592 mInputChannel = new InputChannel();
593 }
Filip Gruszczynski1a4dfe52015-11-15 10:58:57 -0800594 mForceDecorViewVisibility = (mWindowAttributes.privateFlags
595 & PRIVATE_FLAG_FORCE_DECOR_VIEW_VISIBILITY) != 0;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800596 try {
Dianne Hackborn180c4842011-09-13 12:39:25 -0700597 mOrigWindowType = mWindowAttributes.type;
Dianne Hackborn3a3a6cf2012-03-26 10:24:04 -0700598 mAttachInfo.mRecomputeGlobalAttributes = true;
599 collectViewAttributes();
Jeff Brown98365d72012-08-19 20:30:52 -0700600 res = mWindowSession.addToDisplay(mWindow, mSeq, mWindowAttributes,
601 getHostVisibility(), mDisplay.getDisplayId(),
Filip Gruszczynski0ec13282015-06-25 11:26:01 -0700602 mAttachInfo.mContentInsets, mAttachInfo.mStableInsets,
603 mAttachInfo.mOutsets, mInputChannel);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800604 } catch (RemoteException e) {
605 mAdded = false;
606 mView = null;
607 mAttachInfo.mRootView = null;
Jeff Brown46b9ac02010-04-22 18:58:52 -0700608 mInputChannel = null;
Joe Onorato86f67862010-11-05 18:57:34 -0700609 mFallbackEventHandler.setView(null);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800610 unscheduleTraversals();
Svetoslav Ganov45a02e02012-06-17 15:07:29 -0700611 setAccessibilityFocus(null, null);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800612 throw new RuntimeException("Adding window failed", e);
Mitsuru Oshima64f59342009-06-21 00:03:11 -0700613 } finally {
614 if (restore) {
615 attrs.restore();
616 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800617 }
Igor Murashkina86ab6402013-08-30 12:58:36 -0700618
Mitsuru Oshima64f59342009-06-21 00:03:11 -0700619 if (mTranslator != null) {
620 mTranslator.translateRectInScreenToAppWindow(mAttachInfo.mContentInsets);
Mitsuru Oshima9189cab2009-06-03 11:19:12 -0700621 }
Dianne Hackbornc4aad012013-02-22 15:05:25 -0800622 mPendingOverscanInsets.set(0, 0, 0, 0);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800623 mPendingContentInsets.set(mAttachInfo.mContentInsets);
Adrian Roosfa104232014-06-20 16:10:14 -0700624 mPendingStableInsets.set(mAttachInfo.mStableInsets);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800625 mPendingVisibleInsets.set(0, 0, 0, 0);
Jorim Jaggi0ffd49c2016-02-12 15:04:21 -0800626 mAttachInfo.mAlwaysConsumeNavBar =
627 (res & WindowManagerGlobal.ADD_FLAG_ALWAYS_CONSUME_NAV_BAR) != 0;
628 mPendingAlwaysConsumeNavBar = mAttachInfo.mAlwaysConsumeNavBar;
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -0800629 if (DEBUG_LAYOUT) Log.v(mTag, "Added window " + mWindow);
Jeff Brown98365d72012-08-19 20:30:52 -0700630 if (res < WindowManagerGlobal.ADD_OKAY) {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800631 mAttachInfo.mRootView = null;
632 mAdded = false;
Joe Onorato86f67862010-11-05 18:57:34 -0700633 mFallbackEventHandler.setView(null);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800634 unscheduleTraversals();
Svetoslav Ganov45a02e02012-06-17 15:07:29 -0700635 setAccessibilityFocus(null, null);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800636 switch (res) {
Jeff Brown98365d72012-08-19 20:30:52 -0700637 case WindowManagerGlobal.ADD_BAD_APP_TOKEN:
638 case WindowManagerGlobal.ADD_BAD_SUBWINDOW_TOKEN:
639 throw new WindowManager.BadTokenException(
Wale Ogunwale74bf0652015-01-12 10:24:36 -0800640 "Unable to add window -- token " + attrs.token
641 + " is not valid; is your activity running?");
Jeff Brown98365d72012-08-19 20:30:52 -0700642 case WindowManagerGlobal.ADD_NOT_APP_TOKEN:
643 throw new WindowManager.BadTokenException(
Wale Ogunwale74bf0652015-01-12 10:24:36 -0800644 "Unable to add window -- token " + attrs.token
645 + " is not for an application");
Jeff Brown98365d72012-08-19 20:30:52 -0700646 case WindowManagerGlobal.ADD_APP_EXITING:
647 throw new WindowManager.BadTokenException(
Wale Ogunwale74bf0652015-01-12 10:24:36 -0800648 "Unable to add window -- app for token " + attrs.token
649 + " is exiting");
Jeff Brown98365d72012-08-19 20:30:52 -0700650 case WindowManagerGlobal.ADD_DUPLICATE_ADD:
651 throw new WindowManager.BadTokenException(
Wale Ogunwale74bf0652015-01-12 10:24:36 -0800652 "Unable to add window -- window " + mWindow
653 + " has already been added");
Jeff Brown98365d72012-08-19 20:30:52 -0700654 case WindowManagerGlobal.ADD_STARTING_NOT_NEEDED:
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800655 // Silently ignore -- we would have just removed it
656 // right away, anyway.
657 return;
Jeff Brown98365d72012-08-19 20:30:52 -0700658 case WindowManagerGlobal.ADD_MULTIPLE_SINGLETON:
Alan Viverette73f6d602015-09-14 16:01:19 -0400659 throw new WindowManager.BadTokenException("Unable to add window "
660 + mWindow + " -- another window of type "
661 + mWindowAttributes.type + " already exists");
Jeff Brown98365d72012-08-19 20:30:52 -0700662 case WindowManagerGlobal.ADD_PERMISSION_DENIED:
Alan Viverette73f6d602015-09-14 16:01:19 -0400663 throw new WindowManager.BadTokenException("Unable to add window "
664 + mWindow + " -- permission denied for window type "
665 + mWindowAttributes.type);
Craig Mautner6018aee2012-10-23 14:27:49 -0700666 case WindowManagerGlobal.ADD_INVALID_DISPLAY:
Alan Viverette73f6d602015-09-14 16:01:19 -0400667 throw new WindowManager.InvalidDisplayException("Unable to add window "
668 + mWindow + " -- the specified display can not be found");
Wale Ogunwale74bf0652015-01-12 10:24:36 -0800669 case WindowManagerGlobal.ADD_INVALID_TYPE:
Alan Viverette73f6d602015-09-14 16:01:19 -0400670 throw new WindowManager.InvalidDisplayException("Unable to add window "
671 + mWindow + " -- the specified window type "
672 + mWindowAttributes.type + " is not valid");
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800673 }
674 throw new RuntimeException(
Wale Ogunwale74bf0652015-01-12 10:24:36 -0800675 "Unable to add window -- unknown error code " + res);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800676 }
Jeff Brown46b9ac02010-04-22 18:58:52 -0700677
Jeff Brown00fa7bd2010-07-02 15:37:36 -0700678 if (view instanceof RootViewSurfaceTaker) {
679 mInputQueueCallback =
680 ((RootViewSurfaceTaker)view).willYouTakeTheInputQueue();
681 }
Jeff Browncc4f7db2011-08-30 20:34:48 -0700682 if (mInputChannel != null) {
683 if (mInputQueueCallback != null) {
Michael Wrighta44dd262013-04-10 21:12:00 -0700684 mInputQueue = new InputQueue();
Jeff Browncc4f7db2011-08-30 20:34:48 -0700685 mInputQueueCallback.onInputQueueCreated(mInputQueue);
Jeff Browncc4f7db2011-08-30 20:34:48 -0700686 }
Michael Wrighta44dd262013-04-10 21:12:00 -0700687 mInputEventReceiver = new WindowInputEventReceiver(mInputChannel,
688 Looper.myLooper());
Jeff Brown46b9ac02010-04-22 18:58:52 -0700689 }
Svetoslav Ganov8643aa02011-04-20 12:12:33 -0700690
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800691 view.assignParent(this);
Jeff Brown98365d72012-08-19 20:30:52 -0700692 mAddedTouchMode = (res & WindowManagerGlobal.ADD_FLAG_IN_TOUCH_MODE) != 0;
693 mAppVisible = (res & WindowManagerGlobal.ADD_FLAG_APP_VISIBLE) != 0;
Svetoslav Ganov8643aa02011-04-20 12:12:33 -0700694
695 if (mAccessibilityManager.isEnabled()) {
696 mAccessibilityInteractionConnectionManager.ensureConnection();
697 }
Svetoslav Ganov42138042012-03-20 11:51:39 -0700698
699 if (view.getImportantForAccessibility() == View.IMPORTANT_FOR_ACCESSIBILITY_AUTO) {
700 view.setImportantForAccessibility(View.IMPORTANT_FOR_ACCESSIBILITY_YES);
701 }
Michael Wright95ae9422013-03-14 10:58:50 -0700702
Jeff Brownf9e989d2013-04-04 23:04:03 -0700703 // Set up the input pipeline.
704 CharSequence counterSuffix = attrs.getTitle();
Michael Wright899d7052014-04-23 17:23:39 -0700705 mSyntheticInputStage = new SyntheticInputStage();
706 InputStage viewPostImeStage = new ViewPostImeInputStage(mSyntheticInputStage);
Jeff Brownf9e989d2013-04-04 23:04:03 -0700707 InputStage nativePostImeStage = new NativePostImeInputStage(viewPostImeStage,
708 "aq:native-post-ime:" + counterSuffix);
709 InputStage earlyPostImeStage = new EarlyPostImeInputStage(nativePostImeStage);
710 InputStage imeStage = new ImeInputStage(earlyPostImeStage,
711 "aq:ime:" + counterSuffix);
712 InputStage viewPreImeStage = new ViewPreImeInputStage(imeStage);
713 InputStage nativePreImeStage = new NativePreImeInputStage(viewPreImeStage,
714 "aq:native-pre-ime:" + counterSuffix);
715
716 mFirstInputStage = nativePreImeStage;
717 mFirstPostImeInputStage = earlyPostImeStage;
718 mPendingInputEventQueueLengthCounterName = "aq:pending:" + counterSuffix;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800719 }
720 }
721 }
722
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -0800723 private void setTag() {
724 final String[] split = mWindowAttributes.getTitle().toString().split("\\.");
725 if (split.length > 0) {
726 mTag = TAG + "[" + split[split.length - 1] + "]";
727 }
728 }
729
keunyoung30f420f2013-08-02 14:23:10 -0700730 /** Whether the window is in local focus mode or not */
731 private boolean isInLocalFocusMode() {
732 return (mWindowAttributes.flags & WindowManager.LayoutParams.FLAG_LOCAL_FOCUS_MODE) != 0;
733 }
734
Dianne Hackborn49b043f2015-05-07 14:21:38 -0700735 public int getWindowFlags() {
736 return mWindowAttributes.flags;
737 }
738
Dianne Hackbornece0f4f2015-06-11 13:29:01 -0700739 public int getDisplayId() {
740 return mDisplay.getDisplayId();
741 }
742
Dianne Hackborna7bb6fb2015-02-03 18:13:40 -0800743 public CharSequence getTitle() {
744 return mWindowAttributes.getTitle();
745 }
746
Romain Guy8ff6b9e2011-11-09 20:10:18 -0800747 void destroyHardwareResources() {
Romain Guy31f2c2e2011-11-21 10:55:41 -0800748 if (mAttachInfo.mHardwareRenderer != null) {
749 mAttachInfo.mHardwareRenderer.destroyHardwareResources(mView);
John Reckf47a5942014-06-30 16:20:04 -0700750 mAttachInfo.mHardwareRenderer.destroy();
Romain Guy31f2c2e2011-11-21 10:55:41 -0800751 }
752 }
753
Bo Liu845535a2014-03-21 12:06:23 -0700754 public void detachFunctor(long functor) {
John Reck44ac42a2014-05-16 14:46:07 -0700755 if (mAttachInfo.mHardwareRenderer != null) {
756 // Fence so that any pending invokeFunctor() messages will be processed
757 // before we return from detachFunctor.
John Reckf47a5942014-06-30 16:20:04 -0700758 mAttachInfo.mHardwareRenderer.stopDrawing();
John Reck44ac42a2014-05-16 14:46:07 -0700759 }
Romain Guyba6be8a2012-04-23 18:22:09 -0700760 }
761
John Reck3b202512014-06-23 13:13:08 -0700762 /**
763 * Schedules the functor for execution in either kModeProcess or
764 * kModeProcessNoContext, depending on whether or not there is an EGLContext.
765 *
766 * @param functor The native functor to invoke
767 * @param waitForCompletion If true, this will not return until the functor
768 * has invoked. If false, the functor may be invoked
769 * asynchronously.
770 */
Hui Shu9970aee2014-06-23 17:10:30 -0700771 public void invokeFunctor(long functor, boolean waitForCompletion) {
John Reck3b202512014-06-23 13:13:08 -0700772 ThreadedRenderer.invokeFunctor(functor, waitForCompletion);
Bo Liuae738a72014-04-27 16:22:04 -0700773 }
774
John Reck119907c2014-08-14 09:02:01 -0700775 public void registerAnimatingRenderNode(RenderNode animator) {
776 if (mAttachInfo.mHardwareRenderer != null) {
777 mAttachInfo.mHardwareRenderer.registerAnimatingRenderNode(animator);
778 } else {
779 if (mAttachInfo.mPendingAnimatingRenderNodes == null) {
780 mAttachInfo.mPendingAnimatingRenderNodes = new ArrayList<RenderNode>();
781 }
782 mAttachInfo.mPendingAnimatingRenderNodes.add(animator);
783 }
784 }
785
Romain Guy3b748a42013-04-17 18:54:38 -0700786 private void enableHardwareAcceleration(WindowManager.LayoutParams attrs) {
Dianne Hackborn7eec10e2010-11-12 18:03:47 -0800787 mAttachInfo.mHardwareAccelerated = false;
788 mAttachInfo.mHardwareAccelerationRequested = false;
Romain Guy4f6aff32011-01-12 16:21:41 -0800789
Romain Guy856d4e12011-10-14 15:47:55 -0700790 // Don't enable hardware acceleration when the application is in compatibility mode
John Reckdd58e792014-04-02 16:54:28 +0000791 if (mTranslator != null) return;
Romain Guy856d4e12011-10-14 15:47:55 -0700792
Dianne Hackborn7eec10e2010-11-12 18:03:47 -0800793 // Try to enable hardware acceleration if requested
Igor Murashkina86ab6402013-08-30 12:58:36 -0700794 final boolean hardwareAccelerated =
Jim Miller1b365922011-03-09 19:38:07 -0800795 (attrs.flags & WindowManager.LayoutParams.FLAG_HARDWARE_ACCELERATED) != 0;
796
John Reckdd58e792014-04-02 16:54:28 +0000797 if (hardwareAccelerated) {
John Reck51aaf902015-12-02 15:08:07 -0800798 if (!ThreadedRenderer.isAvailable()) {
Romain Guy1af23a32011-03-24 16:03:55 -0700799 return;
800 }
801
Dianne Hackborn5d927c22011-09-02 12:22:18 -0700802 // Persistent processes (including the system) should not do
803 // accelerated rendering on low-end devices. In that case,
804 // sRendererDisabled will be set. In addition, the system process
805 // itself should never do accelerated rendering. In that case, both
806 // sRendererDisabled and sSystemRendererDisabled are set. When
807 // sSystemRendererDisabled is set, PRIVATE_FLAG_FORCE_HARDWARE_ACCELERATED
808 // can be used by code on the system process to escape that and enable
809 // HW accelerated drawing. (This is basically for the lock screen.)
Jim Miller1b365922011-03-09 19:38:07 -0800810
John Reck61375a82014-09-18 19:27:48 +0000811 final boolean fakeHwAccelerated = (attrs.privateFlags &
812 WindowManager.LayoutParams.PRIVATE_FLAG_FAKE_HARDWARE_ACCELERATED) != 0;
Dianne Hackborn5d927c22011-09-02 12:22:18 -0700813 final boolean forceHwAccelerated = (attrs.privateFlags &
814 WindowManager.LayoutParams.PRIVATE_FLAG_FORCE_HARDWARE_ACCELERATED) != 0;
Jim Miller1b365922011-03-09 19:38:07 -0800815
John Reck61375a82014-09-18 19:27:48 +0000816 if (fakeHwAccelerated) {
817 // This is exclusively for the preview windows the window manager
818 // shows for launching applications, so they will look more like
819 // the app being launched.
820 mAttachInfo.mHardwareAccelerationRequested = true;
John Reck51aaf902015-12-02 15:08:07 -0800821 } else if (!ThreadedRenderer.sRendererDisabled
822 || (ThreadedRenderer.sSystemRendererDisabled && forceHwAccelerated)) {
Romain Guyb051e892010-09-28 19:09:36 -0700823 if (mAttachInfo.mHardwareRenderer != null) {
John Reckf47a5942014-06-30 16:20:04 -0700824 mAttachInfo.mHardwareRenderer.destroy();
Romain Guy211370f2012-02-01 16:10:55 -0800825 }
826
Alan Viverette2b12b582014-10-29 11:11:40 -0700827 final Rect insets = attrs.surfaceInsets;
Alan Viverette2cd23e62014-11-04 17:04:02 -0800828 final boolean hasSurfaceInsets = insets.left != 0 || insets.right != 0
829 || insets.top != 0 || insets.bottom != 0;
Alan Viverette2b12b582014-10-29 11:11:40 -0700830 final boolean translucent = attrs.format != PixelFormat.OPAQUE || hasSurfaceInsets;
John Reck51aaf902015-12-02 15:08:07 -0800831 mAttachInfo.mHardwareRenderer = ThreadedRenderer.create(mContext, translucent);
Romain Guye55945e2013-04-04 15:26:04 -0700832 if (mAttachInfo.mHardwareRenderer != null) {
833 mAttachInfo.mHardwareRenderer.setName(attrs.getTitle().toString());
834 mAttachInfo.mHardwareAccelerated =
835 mAttachInfo.mHardwareAccelerationRequested = true;
836 }
Romain Guye4d01122010-06-16 18:44:05 -0700837 }
838 }
839 }
840
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800841 public View getView() {
842 return mView;
843 }
844
845 final WindowLeaked getLocation() {
846 return mLocation;
847 }
848
849 void setLayoutParams(WindowManager.LayoutParams attrs, boolean newView) {
850 synchronized (this) {
Alan Viverettedbed8932014-08-06 17:54:52 -0700851 final int oldInsetLeft = mWindowAttributes.surfaceInsets.left;
852 final int oldInsetTop = mWindowAttributes.surfaceInsets.top;
853 final int oldInsetRight = mWindowAttributes.surfaceInsets.right;
854 final int oldInsetBottom = mWindowAttributes.surfaceInsets.bottom;
855 final int oldSoftInputMode = mWindowAttributes.softInputMode;
Alan Viverette5435a302015-01-29 10:25:34 -0800856 final boolean oldHasManualSurfaceInsets = mWindowAttributes.hasManualSurfaceInsets;
Alan Viverettedbed8932014-08-06 17:54:52 -0700857
Dianne Hackborn9d090892012-06-11 18:35:41 -0700858 // Keep track of the actual window flags supplied by the client.
859 mClientWindowLayoutFlags = attrs.flags;
Alan Viverettedbed8932014-08-06 17:54:52 -0700860
861 // Preserve compatible window flag if exists.
862 final int compatibleWindowFlag = mWindowAttributes.privateFlags
Adam Lesinski95c42972013-10-02 10:13:27 -0700863 & WindowManager.LayoutParams.PRIVATE_FLAG_COMPATIBLE_WINDOW;
Alan Viverettedbed8932014-08-06 17:54:52 -0700864
865 // Transfer over system UI visibility values as they carry current state.
Craig Mautner3fe38c02012-05-03 17:28:09 -0700866 attrs.systemUiVisibility = mWindowAttributes.systemUiVisibility;
867 attrs.subtreeSystemUiVisibility = mWindowAttributes.subtreeSystemUiVisibility;
Alan Viverettedbed8932014-08-06 17:54:52 -0700868
Romain Guyf21c9b02011-09-06 16:56:54 -0700869 mWindowAttributesChangesFlag = mWindowAttributes.copyFrom(attrs);
John Spurlockbd957402013-10-03 11:38:39 -0400870 if ((mWindowAttributesChangesFlag
871 & WindowManager.LayoutParams.TRANSLUCENT_FLAGS_CHANGED) != 0) {
872 // Recompute system ui visibility.
873 mAttachInfo.mRecomputeGlobalAttributes = true;
874 }
Dianne Hackbornc2293022013-02-06 23:14:49 -0800875 if (mWindowAttributes.packageName == null) {
876 mWindowAttributes.packageName = mBasePackageName;
877 }
Adam Lesinski95c42972013-10-02 10:13:27 -0700878 mWindowAttributes.privateFlags |= compatibleWindowFlag;
Dianne Hackborn9d090892012-06-11 18:35:41 -0700879
Alan Viverettedbed8932014-08-06 17:54:52 -0700880 // Restore old surface insets.
881 mWindowAttributes.surfaceInsets.set(
882 oldInsetLeft, oldInsetTop, oldInsetRight, oldInsetBottom);
Alan Viverette5435a302015-01-29 10:25:34 -0800883 mWindowAttributes.hasManualSurfaceInsets = oldHasManualSurfaceInsets;
Alan Viverettedbed8932014-08-06 17:54:52 -0700884
Dianne Hackborn9d090892012-06-11 18:35:41 -0700885 applyKeepScreenOnFlag(mWindowAttributes);
886
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800887 if (newView) {
888 mSoftInputMode = attrs.softInputMode;
889 requestLayout();
890 }
Alan Viverettedbed8932014-08-06 17:54:52 -0700891
The Android Open Source Project10592532009-03-18 17:39:46 -0700892 // Don't lose the mode we last auto-computed.
Alan Viverettedbed8932014-08-06 17:54:52 -0700893 if ((attrs.softInputMode & WindowManager.LayoutParams.SOFT_INPUT_MASK_ADJUST)
The Android Open Source Project10592532009-03-18 17:39:46 -0700894 == WindowManager.LayoutParams.SOFT_INPUT_ADJUST_UNSPECIFIED) {
895 mWindowAttributes.softInputMode = (mWindowAttributes.softInputMode
896 & ~WindowManager.LayoutParams.SOFT_INPUT_MASK_ADJUST)
Alan Viverettedbed8932014-08-06 17:54:52 -0700897 | (oldSoftInputMode & WindowManager.LayoutParams.SOFT_INPUT_MASK_ADJUST);
The Android Open Source Project10592532009-03-18 17:39:46 -0700898 }
Alan Viverettedbed8932014-08-06 17:54:52 -0700899
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800900 mWindowAttributesChanged = true;
901 scheduleTraversals();
902 }
903 }
904
905 void handleAppVisibility(boolean visible) {
906 if (mAppVisible != visible) {
907 mAppVisible = visible;
908 scheduleTraversals();
John Reck73840ea2014-09-22 07:39:18 -0700909 if (!mAppVisible) {
910 WindowManagerGlobal.trimForeground();
911 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800912 }
913 }
914
915 void handleGetNewSurface() {
916 mNewSurfaceNeeded = true;
917 mFullRedrawNeeded = true;
918 scheduleTraversals();
919 }
920
Jeff Brownd912e1f2014-04-11 18:46:22 -0700921 private final DisplayListener mDisplayListener = new DisplayListener() {
922 @Override
923 public void onDisplayChanged(int displayId) {
924 if (mView != null && mDisplay.getDisplayId() == displayId) {
925 final int oldDisplayState = mAttachInfo.mDisplayState;
926 final int newDisplayState = mDisplay.getState();
927 if (oldDisplayState != newDisplayState) {
928 mAttachInfo.mDisplayState = newDisplayState;
Jeff Brownc2932a12014-11-20 18:04:05 -0800929 pokeDrawLockIfNeeded();
Jeff Brownd912e1f2014-04-11 18:46:22 -0700930 if (oldDisplayState != Display.STATE_UNKNOWN) {
931 final int oldScreenState = toViewScreenState(oldDisplayState);
932 final int newScreenState = toViewScreenState(newDisplayState);
933 if (oldScreenState != newScreenState) {
934 mView.dispatchScreenStateChanged(newScreenState);
935 }
936 if (oldDisplayState == Display.STATE_OFF) {
937 // Draw was suppressed so we need to for it to happen here.
938 mFullRedrawNeeded = true;
939 scheduleTraversals();
940 }
941 }
942 }
Romain Guy7e4e5612012-03-05 14:37:29 -0800943 }
944 }
Jeff Brownd912e1f2014-04-11 18:46:22 -0700945
946 @Override
947 public void onDisplayRemoved(int displayId) {
948 }
949
950 @Override
951 public void onDisplayAdded(int displayId) {
952 }
953
954 private int toViewScreenState(int displayState) {
955 return displayState == Display.STATE_OFF ?
956 View.SCREEN_STATE_OFF : View.SCREEN_STATE_ON;
957 }
958 };
Romain Guy7e4e5612012-03-05 14:37:29 -0800959
Jeff Brownc2932a12014-11-20 18:04:05 -0800960 void pokeDrawLockIfNeeded() {
961 final int displayState = mAttachInfo.mDisplayState;
962 if (mView != null && mAdded && mTraversalScheduled
963 && (displayState == Display.STATE_DOZE
964 || displayState == Display.STATE_DOZE_SUSPEND)) {
965 try {
966 mWindowSession.pokeDrawLock(mWindow);
967 } catch (RemoteException ex) {
968 // System server died, oh well.
969 }
970 }
971 }
972
Craig Mautner6018aee2012-10-23 14:27:49 -0700973 @Override
Dianne Hackborn3a3a6cf2012-03-26 10:24:04 -0700974 public void requestFitSystemWindows() {
975 checkThread();
Adrian Roosfa104232014-06-20 16:10:14 -0700976 mApplyInsetsRequested = true;
Dianne Hackborn3a3a6cf2012-03-26 10:24:04 -0700977 scheduleTraversals();
978 }
979
Craig Mautner6018aee2012-10-23 14:27:49 -0700980 @Override
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800981 public void requestLayout() {
Chet Haase3efa7b52012-12-03 08:33:17 -0800982 if (!mHandlingLayoutInLayoutRequest) {
983 checkThread();
984 mLayoutRequested = true;
985 scheduleTraversals();
986 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800987 }
988
Craig Mautner6018aee2012-10-23 14:27:49 -0700989 @Override
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800990 public boolean isLayoutRequested() {
991 return mLayoutRequested;
992 }
993
Romain Guycfef1232012-02-23 13:50:37 -0800994 void invalidate() {
995 mDirty.set(0, 0, mWidth, mHeight);
John Recke4267ea2014-06-03 15:53:15 -0700996 if (!mWillDrawSoon) {
997 scheduleTraversals();
998 }
Romain Guycfef1232012-02-23 13:50:37 -0800999 }
1000
Dianne Hackborna53de062012-05-08 18:53:51 -07001001 void invalidateWorld(View view) {
1002 view.invalidate();
1003 if (view instanceof ViewGroup) {
Romain Guyf84208f2012-09-13 22:50:18 -07001004 ViewGroup parent = (ViewGroup) view;
1005 for (int i = 0; i < parent.getChildCount(); i++) {
Dianne Hackborna53de062012-05-08 18:53:51 -07001006 invalidateWorld(parent.getChildAt(i));
1007 }
1008 }
1009 }
1010
Craig Mautner6018aee2012-10-23 14:27:49 -07001011 @Override
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001012 public void invalidateChild(View child, Rect dirty) {
Romain Guycfef1232012-02-23 13:50:37 -08001013 invalidateChildInParent(null, dirty);
1014 }
1015
Craig Mautner8f303ad2013-06-14 11:32:22 -07001016 @Override
Romain Guycfef1232012-02-23 13:50:37 -08001017 public ViewParent invalidateChildInParent(int[] location, Rect dirty) {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001018 checkThread();
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08001019 if (DEBUG_DRAW) Log.v(mTag, "Invalidate child: " + dirty);
Romain Guycfef1232012-02-23 13:50:37 -08001020
Chet Haase70d4ba12010-10-06 09:46:45 -07001021 if (dirty == null) {
Chet Haase70d4ba12010-10-06 09:46:45 -07001022 invalidate();
Romain Guycfef1232012-02-23 13:50:37 -08001023 return null;
Chet Haase3561d062012-10-23 12:54:51 -07001024 } else if (dirty.isEmpty() && !mIsAnimating) {
Chet Haase05e91ed2012-07-03 14:17:57 -07001025 return null;
Chet Haase70d4ba12010-10-06 09:46:45 -07001026 }
Romain Guycfef1232012-02-23 13:50:37 -08001027
Mitsuru Oshima64f59342009-06-21 00:03:11 -07001028 if (mCurScrollY != 0 || mTranslator != null) {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001029 mTempRect.set(dirty);
Romain Guy1e095972009-07-07 11:22:45 -07001030 dirty = mTempRect;
Mitsuru Oshima8169dae2009-04-28 18:12:09 -07001031 if (mCurScrollY != 0) {
Romain Guycfef1232012-02-23 13:50:37 -08001032 dirty.offset(0, -mCurScrollY);
Mitsuru Oshima8169dae2009-04-28 18:12:09 -07001033 }
Mitsuru Oshima64f59342009-06-21 00:03:11 -07001034 if (mTranslator != null) {
Romain Guy1e095972009-07-07 11:22:45 -07001035 mTranslator.translateRectInAppWindowToScreen(dirty);
Mitsuru Oshima8169dae2009-04-28 18:12:09 -07001036 }
Romain Guy1e095972009-07-07 11:22:45 -07001037 if (mAttachInfo.mScalingRequired) {
1038 dirty.inset(-1, -1);
1039 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001040 }
Romain Guycfef1232012-02-23 13:50:37 -08001041
Alan Viverettea7ea65e2015-05-15 11:30:21 -07001042 invalidateRectOnScreen(dirty);
1043
1044 return null;
1045 }
1046
1047 private void invalidateRectOnScreen(Rect dirty) {
Romain Guycfef1232012-02-23 13:50:37 -08001048 final Rect localDirty = mDirty;
1049 if (!localDirty.isEmpty() && !localDirty.contains(dirty)) {
Romain Guy02ccac62011-06-24 13:20:23 -07001050 mAttachInfo.mSetIgnoreDirtyState = true;
Romain Guy7d695942010-12-01 17:22:29 -08001051 mAttachInfo.mIgnoreDirtyState = true;
1052 }
Romain Guycfef1232012-02-23 13:50:37 -08001053
1054 // Add the new dirty rect to the current one
1055 localDirty.union(dirty.left, dirty.top, dirty.right, dirty.bottom);
1056 // Intersect with the bounds of the window to skip
1057 // updates that lie outside of the visible region
Romain Guy7b2f8b82012-03-19 17:18:54 -07001058 final float appScale = mAttachInfo.mApplicationScale;
Chet Haase3561d062012-10-23 12:54:51 -07001059 final boolean intersected = localDirty.intersect(0, 0,
1060 (int) (mWidth * appScale + 0.5f), (int) (mHeight * appScale + 0.5f));
1061 if (!intersected) {
Chet Haaseb78ee0e2012-10-17 11:32:01 -07001062 localDirty.setEmpty();
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001063 }
Chet Haase3561d062012-10-23 12:54:51 -07001064 if (!mWillDrawSoon && (intersected || mIsAnimating)) {
1065 scheduleTraversals();
1066 }
Romain Guy0d9275e2010-10-26 14:22:30 -07001067 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001068
Daniel Koulomzin087ae472015-12-16 17:52:25 -05001069 public void setIsAmbientMode(boolean ambient) {
1070 mIsAmbientMode = ambient;
1071 }
1072
George Mount41725de2015-04-09 08:23:05 -07001073 void setWindowStopped(boolean stopped) {
Dianne Hackbornce418e62011-03-01 14:31:38 -08001074 if (mStopped != stopped) {
1075 mStopped = stopped;
George Mount41725de2015-04-09 08:23:05 -07001076 if (!mStopped) {
Dianne Hackbornce418e62011-03-01 14:31:38 -08001077 scheduleTraversals();
John Reckd9b16072016-02-23 10:35:19 -08001078 } else {
John Reckf4db3d22016-02-26 12:58:17 -08001079 if (mAttachInfo.mHardwareRenderer != null) {
Wale Ogunwalee4f131a2016-03-07 07:33:56 -08001080 if (DEBUG_DRAW) Log.d(mTag, "WindowStopped on " + getTitle());
John Reckf4db3d22016-02-26 12:58:17 -08001081 mAttachInfo.mHardwareRenderer.updateSurface(null);
John Reckfc736862016-02-26 15:01:52 -08001082 mAttachInfo.mHardwareRenderer.destroyHardwareResources(mView);
John Reckf4db3d22016-02-26 12:58:17 -08001083 }
Dianne Hackbornce418e62011-03-01 14:31:38 -08001084 }
1085 }
1086 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001087
George Mount41725de2015-04-09 08:23:05 -07001088 /**
1089 * Block the input events during an Activity Transition. The KEYCODE_BACK event is allowed
1090 * through to allow quick reversal of the Activity Transition.
1091 *
1092 * @param paused true to pause, false to resume.
1093 */
1094 public void setPausedForTransition(boolean paused) {
1095 mPausedForTransition = paused;
1096 }
1097
Craig Mautner8f303ad2013-06-14 11:32:22 -07001098 @Override
Romain Guycfef1232012-02-23 13:50:37 -08001099 public ViewParent getParent() {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001100 return null;
1101 }
1102
Craig Mautner8f303ad2013-06-14 11:32:22 -07001103 @Override
Mitsuru Oshima64f59342009-06-21 00:03:11 -07001104 public boolean getChildVisibleRect(View child, Rect r, android.graphics.Point offset) {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001105 if (child != mView) {
1106 throw new RuntimeException("child is not mine, honest!");
1107 }
1108 // Note: don't apply scroll offset, because we want to know its
1109 // visibility in the virtual canvas being given to the view hierarchy.
1110 return r.intersect(0, 0, mWidth, mHeight);
1111 }
1112
Igor Murashkina86ab6402013-08-30 12:58:36 -07001113 @Override
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001114 public void bringChildToFront(View child) {
1115 }
1116
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001117 int getHostVisibility() {
Filip Gruszczynski1a4dfe52015-11-15 10:58:57 -08001118 return (mAppVisible || mForceDecorViewVisibility) ? mView.getVisibility() : View.GONE;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001119 }
Romain Guy8506ab42009-06-11 17:35:47 -07001120
Chet Haasecca2c982011-05-20 14:34:18 -07001121 /**
1122 * Add LayoutTransition to the list of transitions to be started in the next traversal.
1123 * This list will be cleared after the transitions on the list are start()'ed. These
1124 * transitionsa re added by LayoutTransition itself when it sets up animations. The setup
1125 * happens during the layout phase of traversal, which we want to complete before any of the
1126 * animations are started (because those animations may side-effect properties that layout
1127 * depends upon, like the bounding rectangles of the affected views). So we add the transition
1128 * to the list and it is started just prior to starting the drawing phase of traversal.
1129 *
1130 * @param transition The LayoutTransition to be started on the next traversal.
1131 *
1132 * @hide
1133 */
1134 public void requestTransitionStart(LayoutTransition transition) {
1135 if (mPendingTransitions == null || !mPendingTransitions.contains(transition)) {
1136 if (mPendingTransitions == null) {
1137 mPendingTransitions = new ArrayList<LayoutTransition>();
1138 }
1139 mPendingTransitions.add(transition);
1140 }
1141 }
1142
John Recka5dda642014-05-22 15:43:54 -07001143 /**
1144 * Notifies the HardwareRenderer that a new frame will be coming soon.
1145 * Currently only {@link ThreadedRenderer} cares about this, and uses
1146 * this knowledge to adjust the scheduling of off-thread animations
1147 */
1148 void notifyRendererOfFramePending() {
1149 if (mAttachInfo.mHardwareRenderer != null) {
1150 mAttachInfo.mHardwareRenderer.notifyFramePending();
1151 }
1152 }
1153
Jeff Brownebb2d8d2012-03-23 17:14:34 -07001154 void scheduleTraversals() {
1155 if (!mTraversalScheduled) {
1156 mTraversalScheduled = true;
Jeff Brown3d4e7efe2015-02-26 15:34:16 -08001157 mTraversalBarrier = mHandler.getLooper().getQueue().postSyncBarrier();
Jeff Brownebb2d8d2012-03-23 17:14:34 -07001158 mChoreographer.postCallback(
1159 Choreographer.CALLBACK_TRAVERSAL, mTraversalRunnable, null);
Michael Wright9d744c72014-02-18 21:27:42 -08001160 if (!mUnbufferedInputDispatch) {
1161 scheduleConsumeBatchedInput();
1162 }
John Recka5dda642014-05-22 15:43:54 -07001163 notifyRendererOfFramePending();
Jeff Brownc2932a12014-11-20 18:04:05 -08001164 pokeDrawLockIfNeeded();
Jeff Brown96e942d2011-11-30 19:55:01 -08001165 }
Jeff Brownebb2d8d2012-03-23 17:14:34 -07001166 }
Jeff Brown96e942d2011-11-30 19:55:01 -08001167
Jeff Brownebb2d8d2012-03-23 17:14:34 -07001168 void unscheduleTraversals() {
1169 if (mTraversalScheduled) {
1170 mTraversalScheduled = false;
Jeff Brown3d4e7efe2015-02-26 15:34:16 -08001171 mHandler.getLooper().getQueue().removeSyncBarrier(mTraversalBarrier);
Jeff Brownebb2d8d2012-03-23 17:14:34 -07001172 mChoreographer.removeCallbacks(
1173 Choreographer.CALLBACK_TRAVERSAL, mTraversalRunnable, null);
1174 }
1175 }
1176
1177 void doTraversal() {
1178 if (mTraversalScheduled) {
1179 mTraversalScheduled = false;
Jeff Brown3d4e7efe2015-02-26 15:34:16 -08001180 mHandler.getLooper().getQueue().removeSyncBarrier(mTraversalBarrier);
Jeff Brownebb2d8d2012-03-23 17:14:34 -07001181
Jeff Brownebb2d8d2012-03-23 17:14:34 -07001182 if (mProfile) {
1183 Debug.startMethodTracing("ViewAncestor");
Jeff Brown96e942d2011-11-30 19:55:01 -08001184 }
Jeff Brown96e942d2011-11-30 19:55:01 -08001185
Chris Craike22c59b2015-05-21 18:33:37 -07001186 performTraversals();
Jeff Brown96e942d2011-11-30 19:55:01 -08001187
Jeff Brownebb2d8d2012-03-23 17:14:34 -07001188 if (mProfile) {
1189 Debug.stopMethodTracing();
1190 mProfile = false;
1191 }
Jeff Brown96e942d2011-11-30 19:55:01 -08001192 }
1193 }
1194
Dianne Hackborn9d090892012-06-11 18:35:41 -07001195 private void applyKeepScreenOnFlag(WindowManager.LayoutParams params) {
1196 // Update window's global keep screen on flag: if a view has requested
1197 // that the screen be kept on, then it is always set; otherwise, it is
1198 // set to whatever the client last requested for the global state.
1199 if (mAttachInfo.mKeepScreenOn) {
1200 params.flags |= WindowManager.LayoutParams.FLAG_KEEP_SCREEN_ON;
1201 } else {
1202 params.flags = (params.flags&~WindowManager.LayoutParams.FLAG_KEEP_SCREEN_ON)
1203 | (mClientWindowLayoutFlags&WindowManager.LayoutParams.FLAG_KEEP_SCREEN_ON);
1204 }
1205 }
1206
Dianne Hackborn3a3a6cf2012-03-26 10:24:04 -07001207 private boolean collectViewAttributes() {
Chris Craikd36a81f2014-07-17 10:16:51 -07001208 if (mAttachInfo.mRecomputeGlobalAttributes) {
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08001209 //Log.i(mTag, "Computing view hierarchy attributes!");
Chris Craikd36a81f2014-07-17 10:16:51 -07001210 mAttachInfo.mRecomputeGlobalAttributes = false;
1211 boolean oldScreenOn = mAttachInfo.mKeepScreenOn;
1212 mAttachInfo.mKeepScreenOn = false;
1213 mAttachInfo.mSystemUiVisibility = 0;
1214 mAttachInfo.mHasSystemUiListeners = false;
1215 mView.dispatchCollectViewAttributes(mAttachInfo, 0);
1216 mAttachInfo.mSystemUiVisibility &= ~mAttachInfo.mDisabledSystemUiVisibility;
Craig Mautner7eac0f52012-09-13 13:14:14 -07001217 WindowManager.LayoutParams params = mWindowAttributes;
Chris Craikd36a81f2014-07-17 10:16:51 -07001218 mAttachInfo.mSystemUiVisibility |= getImpliedSystemUiVisibility(params);
1219 if (mAttachInfo.mKeepScreenOn != oldScreenOn
1220 || mAttachInfo.mSystemUiVisibility != params.subtreeSystemUiVisibility
1221 || mAttachInfo.mHasSystemUiListeners != params.hasSystemUiListeners) {
Dianne Hackborn9d090892012-06-11 18:35:41 -07001222 applyKeepScreenOnFlag(params);
Chris Craikd36a81f2014-07-17 10:16:51 -07001223 params.subtreeSystemUiVisibility = mAttachInfo.mSystemUiVisibility;
1224 params.hasSystemUiListeners = mAttachInfo.mHasSystemUiListeners;
1225 mView.dispatchWindowSystemUiVisiblityChanged(mAttachInfo.mSystemUiVisibility);
Dianne Hackborn3a3a6cf2012-03-26 10:24:04 -07001226 return true;
1227 }
1228 }
1229 return false;
1230 }
1231
John Spurlockbd957402013-10-03 11:38:39 -04001232 private int getImpliedSystemUiVisibility(WindowManager.LayoutParams params) {
1233 int vis = 0;
1234 // Translucent decor window flags imply stable system ui visibility.
1235 if ((params.flags & WindowManager.LayoutParams.FLAG_TRANSLUCENT_STATUS) != 0) {
1236 vis |= View.SYSTEM_UI_FLAG_LAYOUT_STABLE | View.SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN;
1237 }
1238 if ((params.flags & WindowManager.LayoutParams.FLAG_TRANSLUCENT_NAVIGATION) != 0) {
1239 vis |= View.SYSTEM_UI_FLAG_LAYOUT_STABLE | View.SYSTEM_UI_FLAG_LAYOUT_HIDE_NAVIGATION;
1240 }
1241 return vis;
1242 }
1243
Dianne Hackborn3a3a6cf2012-03-26 10:24:04 -07001244 private boolean measureHierarchy(final View host, final WindowManager.LayoutParams lp,
1245 final Resources res, final int desiredWindowWidth, final int desiredWindowHeight) {
1246 int childWidthMeasureSpec;
1247 int childHeightMeasureSpec;
1248 boolean windowSizeMayChange = false;
1249
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08001250 if (DEBUG_ORIENTATION || DEBUG_LAYOUT) Log.v(mTag,
Dianne Hackborn3a3a6cf2012-03-26 10:24:04 -07001251 "Measuring " + host + " in display " + desiredWindowWidth
1252 + "x" + desiredWindowHeight + "...");
1253
1254 boolean goodMeasure = false;
1255 if (lp.width == ViewGroup.LayoutParams.WRAP_CONTENT) {
1256 // On large screens, we don't want to allow dialogs to just
1257 // stretch to fill the entire width of the screen to display
1258 // one line of text. First try doing the layout at a smaller
1259 // size to see if it will fit.
1260 final DisplayMetrics packageMetrics = res.getDisplayMetrics();
1261 res.getValue(com.android.internal.R.dimen.config_prefDialogWidth, mTmpValue, true);
1262 int baseSize = 0;
1263 if (mTmpValue.type == TypedValue.TYPE_DIMENSION) {
1264 baseSize = (int)mTmpValue.getDimension(packageMetrics);
1265 }
Filip Gruszczynski1937a4c2016-01-19 16:17:13 -08001266 if (DEBUG_DIALOG) Log.v(mTag, "Window " + mView + ": baseSize=" + baseSize
1267 + ", desiredWindowWidth=" + desiredWindowWidth);
Dianne Hackborn3a3a6cf2012-03-26 10:24:04 -07001268 if (baseSize != 0 && desiredWindowWidth > baseSize) {
1269 childWidthMeasureSpec = getRootMeasureSpec(baseSize, lp.width);
1270 childHeightMeasureSpec = getRootMeasureSpec(desiredWindowHeight, lp.height);
Jeff Brownc8d2668b2012-05-16 17:23:59 -07001271 performMeasure(childWidthMeasureSpec, childHeightMeasureSpec);
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08001272 if (DEBUG_DIALOG) Log.v(mTag, "Window " + mView + ": measured ("
Filip Gruszczynski1937a4c2016-01-19 16:17:13 -08001273 + host.getMeasuredWidth() + "," + host.getMeasuredHeight()
1274 + ") from width spec: " + MeasureSpec.toString(childWidthMeasureSpec)
1275 + " and height spec: " + MeasureSpec.toString(childHeightMeasureSpec));
Dianne Hackborn3a3a6cf2012-03-26 10:24:04 -07001276 if ((host.getMeasuredWidthAndState()&View.MEASURED_STATE_TOO_SMALL) == 0) {
1277 goodMeasure = true;
1278 } else {
1279 // Didn't fit in that size... try expanding a bit.
1280 baseSize = (baseSize+desiredWindowWidth)/2;
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08001281 if (DEBUG_DIALOG) Log.v(mTag, "Window " + mView + ": next baseSize="
Dianne Hackborn3a3a6cf2012-03-26 10:24:04 -07001282 + baseSize);
1283 childWidthMeasureSpec = getRootMeasureSpec(baseSize, lp.width);
Jeff Brownc8d2668b2012-05-16 17:23:59 -07001284 performMeasure(childWidthMeasureSpec, childHeightMeasureSpec);
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08001285 if (DEBUG_DIALOG) Log.v(mTag, "Window " + mView + ": measured ("
Dianne Hackborn3a3a6cf2012-03-26 10:24:04 -07001286 + host.getMeasuredWidth() + "," + host.getMeasuredHeight() + ")");
1287 if ((host.getMeasuredWidthAndState()&View.MEASURED_STATE_TOO_SMALL) == 0) {
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08001288 if (DEBUG_DIALOG) Log.v(mTag, "Good!");
Dianne Hackborn3a3a6cf2012-03-26 10:24:04 -07001289 goodMeasure = true;
1290 }
1291 }
1292 }
1293 }
1294
1295 if (!goodMeasure) {
1296 childWidthMeasureSpec = getRootMeasureSpec(desiredWindowWidth, lp.width);
1297 childHeightMeasureSpec = getRootMeasureSpec(desiredWindowHeight, lp.height);
Jeff Brownc8d2668b2012-05-16 17:23:59 -07001298 performMeasure(childWidthMeasureSpec, childHeightMeasureSpec);
Dianne Hackborn3a3a6cf2012-03-26 10:24:04 -07001299 if (mWidth != host.getMeasuredWidth() || mHeight != host.getMeasuredHeight()) {
1300 windowSizeMayChange = true;
1301 }
1302 }
1303
1304 if (DBG) {
1305 System.out.println("======================================");
1306 System.out.println("performTraversals -- after measure");
1307 host.debug();
1308 }
1309
1310 return windowSizeMayChange;
1311 }
1312
Alan Viverettefed3f722013-11-14 14:48:20 -08001313 /**
1314 * Modifies the input matrix such that it maps view-local coordinates to
1315 * on-screen coordinates.
1316 *
1317 * @param m input matrix to modify
1318 */
1319 void transformMatrixToGlobal(Matrix m) {
Dake Gu7bf379c2014-07-15 16:29:38 -07001320 m.preTranslate(mAttachInfo.mWindowLeft, mAttachInfo.mWindowTop);
Alan Viverettefed3f722013-11-14 14:48:20 -08001321 }
1322
1323 /**
1324 * Modifies the input matrix such that it maps on-screen coordinates to
1325 * view-local coordinates.
1326 *
1327 * @param m input matrix to modify
1328 */
1329 void transformMatrixToLocal(Matrix m) {
Dake Gu7bf379c2014-07-15 16:29:38 -07001330 m.postTranslate(-mAttachInfo.mWindowLeft, -mAttachInfo.mWindowTop);
Alan Viverettefed3f722013-11-14 14:48:20 -08001331 }
1332
Filip Gruszczynski954289d2015-02-26 15:46:47 -08001333 /* package */ WindowInsets getWindowInsets(boolean forceConstruct) {
1334 if (mLastWindowInsets == null || forceConstruct) {
1335 mDispatchContentInsets.set(mAttachInfo.mContentInsets);
1336 mDispatchStableInsets.set(mAttachInfo.mStableInsets);
1337 Rect contentInsets = mDispatchContentInsets;
1338 Rect stableInsets = mDispatchStableInsets;
1339 // For dispatch we preserve old logic, but for direct requests from Views we allow to
1340 // immediately use pending insets.
1341 if (!forceConstruct
1342 && (!mPendingContentInsets.equals(contentInsets) ||
1343 !mPendingStableInsets.equals(stableInsets))) {
1344 contentInsets = mPendingContentInsets;
1345 stableInsets = mPendingStableInsets;
1346 }
Filip Gruszczynski2217f612015-05-26 11:32:08 -07001347 Rect outsets = mAttachInfo.mOutsets;
1348 if (outsets.left > 0 || outsets.top > 0 || outsets.right > 0 || outsets.bottom > 0) {
1349 contentInsets = new Rect(contentInsets.left + outsets.left,
1350 contentInsets.top + outsets.top, contentInsets.right + outsets.right,
1351 contentInsets.bottom + outsets.bottom);
1352 }
Filip Gruszczynski954289d2015-02-26 15:46:47 -08001353 mLastWindowInsets = new WindowInsets(contentInsets,
Adam Powell01f280d2015-05-18 16:07:42 -07001354 null /* windowDecorInsets */, stableInsets,
Jorim Jaggi0ffd49c2016-02-12 15:04:21 -08001355 mContext.getResources().getConfiguration().isScreenRound(),
1356 mAttachInfo.mAlwaysConsumeNavBar);
Filip Gruszczynski954289d2015-02-26 15:46:47 -08001357 }
1358 return mLastWindowInsets;
1359 }
1360
Adam Powell2accbf92014-04-16 23:14:57 +00001361 void dispatchApplyInsets(View host) {
Filip Gruszczynski954289d2015-02-26 15:46:47 -08001362 host.dispatchApplyWindowInsets(getWindowInsets(true /* forceConstruct */));
Adam Powell2accbf92014-04-16 23:14:57 +00001363 }
1364
Chong Zhangf6525ce2016-01-14 17:09:56 -08001365 private static boolean shouldUseDisplaySize(final WindowManager.LayoutParams lp) {
1366 return lp.type == TYPE_STATUS_BAR_PANEL
1367 || lp.type == TYPE_INPUT_METHOD
1368 || lp.type == TYPE_VOLUME_OVERLAY;
1369 }
1370
1371 private int dipToPx(int dip) {
1372 final DisplayMetrics displayMetrics = mContext.getResources().getDisplayMetrics();
1373 return (int) (displayMetrics.density * dip + 0.5f);
1374 }
1375
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001376 private void performTraversals() {
1377 // cache mView since it is used so much below...
1378 final View host = mView;
1379
1380 if (DBG) {
1381 System.out.println("======================================");
1382 System.out.println("performTraversals");
1383 host.debug();
1384 }
1385
1386 if (host == null || !mAdded)
1387 return;
1388
Craig Mautnerdf2390a2012-08-29 20:59:22 -07001389 mIsInTraversal = true;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001390 mWillDrawSoon = true;
Dianne Hackborn711e62a2010-11-29 16:38:22 -08001391 boolean windowSizeMayChange = false;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001392 boolean newSurface = false;
Dianne Hackborndc8a7f62010-05-10 11:29:34 -07001393 boolean surfaceChanged = false;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001394 WindowManager.LayoutParams lp = mWindowAttributes;
1395
1396 int desiredWindowWidth;
1397 int desiredWindowHeight;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001398
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001399 final int viewVisibility = getHostVisibility();
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08001400 final boolean viewVisibilityChanged = !mFirst
1401 && (mViewVisibility != viewVisibility || mNewSurfaceNeeded);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001402
1403 WindowManager.LayoutParams params = null;
1404 if (mWindowAttributesChanged) {
1405 mWindowAttributesChanged = false;
Dianne Hackborndc8a7f62010-05-10 11:29:34 -07001406 surfaceChanged = true;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001407 params = lp;
1408 }
Craig Mautner48d0d182013-06-11 07:53:06 -07001409 CompatibilityInfo compatibilityInfo = mDisplayAdjustments.getCompatibilityInfo();
Dianne Hackborn5fd21692011-06-07 14:09:47 -07001410 if (compatibilityInfo.supportsScreen() == mLastInCompatMode) {
1411 params = lp;
Jeff Brown96e942d2011-11-30 19:55:01 -08001412 mFullRedrawNeeded = true;
Dianne Hackborn5fd21692011-06-07 14:09:47 -07001413 mLayoutRequested = true;
1414 if (mLastInCompatMode) {
Adam Lesinski95c42972013-10-02 10:13:27 -07001415 params.privateFlags &= ~WindowManager.LayoutParams.PRIVATE_FLAG_COMPATIBLE_WINDOW;
Dianne Hackborn5fd21692011-06-07 14:09:47 -07001416 mLastInCompatMode = false;
1417 } else {
Adam Lesinski95c42972013-10-02 10:13:27 -07001418 params.privateFlags |= WindowManager.LayoutParams.PRIVATE_FLAG_COMPATIBLE_WINDOW;
Dianne Hackborn5fd21692011-06-07 14:09:47 -07001419 mLastInCompatMode = true;
1420 }
1421 }
Igor Murashkina86ab6402013-08-30 12:58:36 -07001422
Romain Guyf21c9b02011-09-06 16:56:54 -07001423 mWindowAttributesChangesFlag = 0;
Igor Murashkina86ab6402013-08-30 12:58:36 -07001424
Mitsuru Oshima64f59342009-06-21 00:03:11 -07001425 Rect frame = mWinFrame;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001426 if (mFirst) {
Jeff Brown96e942d2011-11-30 19:55:01 -08001427 mFullRedrawNeeded = true;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001428 mLayoutRequested = true;
1429
Chong Zhangf6525ce2016-01-14 17:09:56 -08001430 if (shouldUseDisplaySize(lp)) {
Dianne Hackborna239c842011-06-01 12:28:20 -07001431 // NOTE -- system code, won't try to do compat mode.
Jeff Brownbc68a592011-07-25 12:58:12 -07001432 Point size = new Point();
Jeff Brown98365d72012-08-19 20:30:52 -07001433 mDisplay.getRealSize(size);
Jeff Brownbc68a592011-07-25 12:58:12 -07001434 desiredWindowWidth = size.x;
1435 desiredWindowHeight = size.y;
Dianne Hackborna239c842011-06-01 12:28:20 -07001436 } else {
Chong Zhangf6525ce2016-01-14 17:09:56 -08001437 Configuration config = mContext.getResources().getConfiguration();
1438 desiredWindowWidth = dipToPx(config.screenWidthDp);
1439 desiredWindowHeight = dipToPx(config.screenHeightDp);
Dianne Hackborna239c842011-06-01 12:28:20 -07001440 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001441
Romain Guyc5d55862011-01-21 19:01:46 -08001442 // We used to use the following condition to choose 32 bits drawing caches:
1443 // PixelFormat.hasAlpha(lp.format) || lp.format == PixelFormat.RGBX_8888
1444 // However, windows are now always 32 bits by default, so choose 32 bits
Chris Craikd36a81f2014-07-17 10:16:51 -07001445 mAttachInfo.mUse32BitDrawingCache = true;
1446 mAttachInfo.mHasWindowFocus = false;
1447 mAttachInfo.mWindowVisibility = viewVisibility;
1448 mAttachInfo.mRecomputeGlobalAttributes = false;
Dianne Hackborn694f79b2010-03-17 19:44:59 -07001449 mLastConfiguration.setTo(host.getResources().getConfiguration());
Dianne Hackborn3a3a6cf2012-03-26 10:24:04 -07001450 mLastSystemUiVisibility = mAttachInfo.mSystemUiVisibility;
Fabrice Di Megliob003e282012-10-17 17:20:19 -07001451 // Set the layout direction if it has not been set before (inherit is the default)
1452 if (mViewLayoutDirectionInitial == View.LAYOUT_DIRECTION_INHERIT) {
1453 host.setLayoutDirection(mLastConfiguration.getLayoutDirection());
1454 }
Chris Craikd36a81f2014-07-17 10:16:51 -07001455 host.dispatchAttachedToWindow(mAttachInfo, 0);
1456 mAttachInfo.mTreeObserver.dispatchOnWindowAttachedChange(true);
Adam Powell2accbf92014-04-16 23:14:57 +00001457 dispatchApplyInsets(host);
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08001458 //Log.i(mTag, "Screen on initialized: " + attachInfo.mKeepScreenOn);
svetoslavganov75986cf2009-05-14 22:28:01 -07001459
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001460 } else {
Mitsuru Oshima64f59342009-06-21 00:03:11 -07001461 desiredWindowWidth = frame.width();
1462 desiredWindowHeight = frame.height();
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001463 if (desiredWindowWidth != mWidth || desiredWindowHeight != mHeight) {
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08001464 if (DEBUG_ORIENTATION) Log.v(mTag, "View " + host + " resized to: " + frame);
Jeff Brown96e942d2011-11-30 19:55:01 -08001465 mFullRedrawNeeded = true;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001466 mLayoutRequested = true;
Dianne Hackborn711e62a2010-11-29 16:38:22 -08001467 windowSizeMayChange = true;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001468 }
1469 }
1470
1471 if (viewVisibilityChanged) {
Chris Craikd36a81f2014-07-17 10:16:51 -07001472 mAttachInfo.mWindowVisibility = viewVisibility;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001473 host.dispatchWindowVisibilityChanged(viewVisibility);
Adam Powell9c146bf2016-03-15 17:35:00 -07001474 host.dispatchVisibilityAggregated(viewVisibility == View.VISIBLE);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001475 if (viewVisibility != View.VISIBLE || mNewSurfaceNeeded) {
Skuhneb8160872015-09-22 09:51:39 -07001476 endDragResizing();
Romain Guy65b345f2011-07-27 18:51:50 -07001477 destroyHardwareResources();
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001478 }
1479 if (viewVisibility == View.GONE) {
1480 // After making a window gone, we will count it as being
1481 // shown for the first time the next time it gets focus.
1482 mHasHadWindowFocus = false;
1483 }
1484 }
1485
Alan Viverette7dbc3bf2015-01-28 16:14:36 -08001486 // Non-visible windows can't hold accessibility focus.
1487 if (mAttachInfo.mWindowVisibility != View.VISIBLE) {
1488 host.clearAccessibilityFocus();
1489 }
1490
Chet Haaseb78c2842012-04-19 13:39:50 -07001491 // Execute enqueued actions on every traversal in case a detached view enqueued an action
Chris Craikd36a81f2014-07-17 10:16:51 -07001492 getRunQueue().executeActions(mAttachInfo.mHandler);
Chet Haaseb78c2842012-04-19 13:39:50 -07001493
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001494 boolean insetsChanged = false;
Romain Guy8506ab42009-06-11 17:35:47 -07001495
Craig Mautner72d6f212015-02-19 16:33:09 -08001496 boolean layoutRequested = mLayoutRequested && (!mStopped || mReportNextDraw);
Dianne Hackborn3a3a6cf2012-03-26 10:24:04 -07001497 if (layoutRequested) {
Romain Guy15df6702009-08-17 20:17:30 -07001498
Dianne Hackborn711e62a2010-11-29 16:38:22 -08001499 final Resources res = mView.getContext().getResources();
1500
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001501 if (mFirst) {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001502 // make sure touch mode code executes by setting cached value
1503 // to opposite of the added touch mode.
1504 mAttachInfo.mInTouchMode = !mAddedTouchMode;
Romain Guy2d4cff62010-04-09 15:39:00 -07001505 ensureTouchModeLocally(mAddedTouchMode);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001506 } else {
Dianne Hackbornc4aad012013-02-22 15:05:25 -08001507 if (!mPendingOverscanInsets.equals(mAttachInfo.mOverscanInsets)) {
1508 insetsChanged = true;
1509 }
Dianne Hackbornfa6b35b2011-08-24 17:03:54 -07001510 if (!mPendingContentInsets.equals(mAttachInfo.mContentInsets)) {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001511 insetsChanged = true;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001512 }
Adrian Roosfa104232014-06-20 16:10:14 -07001513 if (!mPendingStableInsets.equals(mAttachInfo.mStableInsets)) {
1514 insetsChanged = true;
1515 }
Dianne Hackbornfa6b35b2011-08-24 17:03:54 -07001516 if (!mPendingVisibleInsets.equals(mAttachInfo.mVisibleInsets)) {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001517 mAttachInfo.mVisibleInsets.set(mPendingVisibleInsets);
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08001518 if (DEBUG_LAYOUT) Log.v(mTag, "Visible insets changing to: "
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001519 + mAttachInfo.mVisibleInsets);
1520 }
Filip Gruszczynski2217f612015-05-26 11:32:08 -07001521 if (!mPendingOutsets.equals(mAttachInfo.mOutsets)) {
1522 insetsChanged = true;
1523 }
Jorim Jaggi0ffd49c2016-02-12 15:04:21 -08001524 if (mPendingAlwaysConsumeNavBar != mAttachInfo.mAlwaysConsumeNavBar) {
1525 insetsChanged = true;
1526 }
Chong Zhangf6525ce2016-01-14 17:09:56 -08001527 if (lp.width == ViewGroup.LayoutParams.WRAP_CONTENT
1528 || lp.height == ViewGroup.LayoutParams.WRAP_CONTENT) {
Dianne Hackborn711e62a2010-11-29 16:38:22 -08001529 windowSizeMayChange = true;
Chong Zhangf6525ce2016-01-14 17:09:56 -08001530
1531 if (shouldUseDisplaySize(lp)) {
1532 // NOTE -- system code, won't try to do compat mode.
1533 Point size = new Point();
1534 mDisplay.getRealSize(size);
1535 desiredWindowWidth = size.x;
1536 desiredWindowHeight = size.y;
1537 } else {
1538 Configuration config = res.getConfiguration();
1539 desiredWindowWidth = dipToPx(config.screenWidthDp);
1540 desiredWindowHeight = dipToPx(config.screenHeightDp);
1541 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001542 }
1543 }
1544
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001545 // Ask host how big it wants to be
Dianne Hackborn3a3a6cf2012-03-26 10:24:04 -07001546 windowSizeMayChange |= measureHierarchy(host, lp, res,
1547 desiredWindowWidth, desiredWindowHeight);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001548 }
1549
Dianne Hackborn3a3a6cf2012-03-26 10:24:04 -07001550 if (collectViewAttributes()) {
1551 params = lp;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001552 }
Chris Craikd36a81f2014-07-17 10:16:51 -07001553 if (mAttachInfo.mForceReportNewAttributes) {
1554 mAttachInfo.mForceReportNewAttributes = false;
Dianne Hackborn9a230e02011-10-06 11:51:27 -07001555 params = lp;
1556 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001557
Chris Craikd36a81f2014-07-17 10:16:51 -07001558 if (mFirst || mAttachInfo.mViewVisibilityChanged) {
1559 mAttachInfo.mViewVisibilityChanged = false;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001560 int resizeMode = mSoftInputMode &
1561 WindowManager.LayoutParams.SOFT_INPUT_MASK_ADJUST;
1562 // If we are in auto resize mode, then we need to determine
1563 // what mode to use now.
1564 if (resizeMode == WindowManager.LayoutParams.SOFT_INPUT_ADJUST_UNSPECIFIED) {
Chris Craikd36a81f2014-07-17 10:16:51 -07001565 final int N = mAttachInfo.mScrollContainers.size();
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001566 for (int i=0; i<N; i++) {
Chris Craikd36a81f2014-07-17 10:16:51 -07001567 if (mAttachInfo.mScrollContainers.get(i).isShown()) {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001568 resizeMode = WindowManager.LayoutParams.SOFT_INPUT_ADJUST_RESIZE;
1569 }
1570 }
1571 if (resizeMode == 0) {
1572 resizeMode = WindowManager.LayoutParams.SOFT_INPUT_ADJUST_PAN;
1573 }
1574 if ((lp.softInputMode &
1575 WindowManager.LayoutParams.SOFT_INPUT_MASK_ADJUST) != resizeMode) {
1576 lp.softInputMode = (lp.softInputMode &
1577 ~WindowManager.LayoutParams.SOFT_INPUT_MASK_ADJUST) |
1578 resizeMode;
1579 params = lp;
1580 }
1581 }
1582 }
Romain Guy8506ab42009-06-11 17:35:47 -07001583
Dianne Hackbornc4aad012013-02-22 15:05:25 -08001584 if (params != null) {
1585 if ((host.mPrivateFlags & View.PFLAG_REQUEST_TRANSPARENT_REGIONS) != 0) {
1586 if (!PixelFormat.formatHasAlpha(params.format)) {
1587 params.format = PixelFormat.TRANSLUCENT;
1588 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001589 }
Dianne Hackbornc4aad012013-02-22 15:05:25 -08001590 mAttachInfo.mOverscanRequested = (params.flags
1591 & WindowManager.LayoutParams.FLAG_LAYOUT_IN_OVERSCAN) != 0;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001592 }
1593
Adrian Roosfa104232014-06-20 16:10:14 -07001594 if (mApplyInsetsRequested) {
1595 mApplyInsetsRequested = false;
Dianne Hackbornc4aad012013-02-22 15:05:25 -08001596 mLastOverscanRequested = mAttachInfo.mOverscanRequested;
Adam Powell2accbf92014-04-16 23:14:57 +00001597 dispatchApplyInsets(host);
Dianne Hackborn3a3a6cf2012-03-26 10:24:04 -07001598 if (mLayoutRequested) {
1599 // Short-circuit catching a new layout request here, so
1600 // we don't need to go through two layout passes when things
1601 // change due to fitting system windows, which can happen a lot.
1602 windowSizeMayChange |= measureHierarchy(host, lp,
1603 mView.getContext().getResources(),
1604 desiredWindowWidth, desiredWindowHeight);
1605 }
1606 }
1607
1608 if (layoutRequested) {
1609 // Clear this now, so that if anything requests a layout in the
1610 // rest of this function we will catch it and re-run a full
1611 // layout pass.
1612 mLayoutRequested = false;
1613 }
1614
1615 boolean windowShouldResize = layoutRequested && windowSizeMayChange
Dianne Hackborn189ee182010-12-02 21:48:53 -08001616 && ((mWidth != host.getMeasuredWidth() || mHeight != host.getMeasuredHeight())
Romain Guy2e4f4262010-04-06 11:07:52 -07001617 || (lp.width == ViewGroup.LayoutParams.WRAP_CONTENT &&
1618 frame.width() < desiredWindowWidth && frame.width() != mWidth)
1619 || (lp.height == ViewGroup.LayoutParams.WRAP_CONTENT &&
1620 frame.height() < desiredWindowHeight && frame.height() != mHeight));
Jorim Jaggi4846ee32016-01-07 17:39:12 +01001621 windowShouldResize |= mDragResizing && mResizeMode == RESIZE_MODE_FREEFORM;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001622
Jorim Jaggi4846ee32016-01-07 17:39:12 +01001623 // If the activity was just relaunched, it might have unfrozen the task bounds (while
1624 // relaunching), so we need to force a call into window manager to pick up the latest
1625 // bounds.
1626 windowShouldResize |= mActivityRelaunched;
1627
Jeff Brown2e05ec32013-09-30 15:57:43 -07001628 // Determine whether to compute insets.
1629 // If there are no inset listeners remaining then we may still need to compute
1630 // insets in case the old insets were non-empty and must be reset.
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001631 final boolean computesInternalInsets =
Chris Craikd36a81f2014-07-17 10:16:51 -07001632 mAttachInfo.mTreeObserver.hasComputeInternalInsetsListeners()
1633 || mAttachInfo.mHasNonEmptyGivenInternalInsets;
Romain Guy812ccbe2010-06-01 14:07:24 -07001634
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001635 boolean insetsPending = false;
1636 int relayoutResult = 0;
Romain Guy812ccbe2010-06-01 14:07:24 -07001637
Filip Gruszczynski1a4dfe52015-11-15 10:58:57 -08001638 final boolean isViewVisible = viewVisibility == View.VISIBLE;
Romain Guy812ccbe2010-06-01 14:07:24 -07001639 if (mFirst || windowShouldResize || insetsChanged ||
Jorim Jaggia4a58ef2016-01-27 02:10:08 -08001640 viewVisibilityChanged || params != null || mForceNextWindowRelayout) {
1641 mForceNextWindowRelayout = false;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001642
Alan Viverette64bf97a2015-09-18 16:42:00 -04001643 if (isViewVisible) {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001644 // If this window is giving internal insets to the window
1645 // manager, and it is being added or changing its visibility,
1646 // then we want to first give the window manager "fake"
1647 // insets to cause it to effectively ignore the content of
1648 // the window during layout. This avoids it briefly causing
1649 // other windows to resize/move based on the raw frame of the
1650 // window, waiting until we can finish laying out this window
1651 // and get back to the window manager with the ultimately
1652 // computed insets.
Romain Guy812ccbe2010-06-01 14:07:24 -07001653 insetsPending = computesInternalInsets && (mFirst || viewVisibilityChanged);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001654 }
1655
Dianne Hackborndc8a7f62010-05-10 11:29:34 -07001656 if (mSurfaceHolder != null) {
1657 mSurfaceHolder.mSurfaceLock.lock();
1658 mDrawingAllowed = true;
Dianne Hackborndc8a7f62010-05-10 11:29:34 -07001659 }
Romain Guy812ccbe2010-06-01 14:07:24 -07001660
Romain Guyc361da82010-10-25 15:29:10 -07001661 boolean hwInitialized = false;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001662 boolean contentInsetsChanged = false;
Dianne Hackborndc8a7f62010-05-10 11:29:34 -07001663 boolean hadSurface = mSurface.isValid();
Romain Guy812ccbe2010-06-01 14:07:24 -07001664
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001665 try {
Mitsuru Oshima8169dae2009-04-28 18:12:09 -07001666 if (DEBUG_LAYOUT) {
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08001667 Log.i(mTag, "host=w:" + host.getMeasuredWidth() + ", h:" +
Dianne Hackborn189ee182010-12-02 21:48:53 -08001668 host.getMeasuredHeight() + ", params=" + params);
Mitsuru Oshima8169dae2009-04-28 18:12:09 -07001669 }
Romain Guy2a83f002011-01-18 18:28:21 -08001670
John Reckf7d9c1d2014-04-09 10:01:03 -07001671 if (mAttachInfo.mHardwareRenderer != null) {
1672 // relayoutWindow may decide to destroy mSurface. As that decision
1673 // happens in WindowManager service, we need to be defensive here
1674 // and stop using the surface in case it gets destroyed.
John Reck01a5ea32014-12-03 13:01:07 -08001675 if (mAttachInfo.mHardwareRenderer.pauseSurface(mSurface)) {
1676 // Animations were running so we need to push a frame
1677 // to resume them
1678 mDirty.set(0, 0, mWidth, mHeight);
1679 }
John Reckba6adf62015-02-19 14:36:50 -08001680 mChoreographer.mFrameInfo.addFlags(FrameInfo.FLAG_WINDOW_LAYOUT_CHANGED);
John Reckf7d9c1d2014-04-09 10:01:03 -07001681 }
Romain Guy2a83f002011-01-18 18:28:21 -08001682 final int surfaceGenerationId = mSurface.getGenerationId();
Mitsuru Oshima8169dae2009-04-28 18:12:09 -07001683 relayoutResult = relayoutWindow(params, viewVisibility, insetsPending);
1684
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08001685 if (DEBUG_LAYOUT) Log.v(mTag, "relayout: frame=" + frame.toShortString()
Dianne Hackbornc4aad012013-02-22 15:05:25 -08001686 + " overscan=" + mPendingOverscanInsets.toShortString()
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001687 + " content=" + mPendingContentInsets.toShortString()
1688 + " visible=" + mPendingVisibleInsets.toShortString()
Adrian Roosfa104232014-06-20 16:10:14 -07001689 + " visible=" + mPendingStableInsets.toShortString()
Filip Gruszczynski0ec13282015-06-25 11:26:01 -07001690 + " outsets=" + mPendingOutsets.toShortString()
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001691 + " surface=" + mSurface);
Romain Guy8506ab42009-06-11 17:35:47 -07001692
Dianne Hackborn694f79b2010-03-17 19:44:59 -07001693 if (mPendingConfiguration.seq != 0) {
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08001694 if (DEBUG_CONFIGURATION) Log.v(mTag, "Visible with new config: "
Dianne Hackborn694f79b2010-03-17 19:44:59 -07001695 + mPendingConfiguration);
riddle_hsu164725c2015-11-12 14:07:12 +08001696 updateConfiguration(new Configuration(mPendingConfiguration), !mFirst);
Dianne Hackborn694f79b2010-03-17 19:44:59 -07001697 mPendingConfiguration.seq = 0;
1698 }
Dianne Hackborn3556c9a2012-05-04 16:31:25 -07001699
Dianne Hackbornc4aad012013-02-22 15:05:25 -08001700 final boolean overscanInsetsChanged = !mPendingOverscanInsets.equals(
1701 mAttachInfo.mOverscanInsets);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001702 contentInsetsChanged = !mPendingContentInsets.equals(
1703 mAttachInfo.mContentInsets);
Dianne Hackbornc4aad012013-02-22 15:05:25 -08001704 final boolean visibleInsetsChanged = !mPendingVisibleInsets.equals(
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001705 mAttachInfo.mVisibleInsets);
Adrian Roosfa104232014-06-20 16:10:14 -07001706 final boolean stableInsetsChanged = !mPendingStableInsets.equals(
1707 mAttachInfo.mStableInsets);
Filip Gruszczynski2217f612015-05-26 11:32:08 -07001708 final boolean outsetsChanged = !mPendingOutsets.equals(mAttachInfo.mOutsets);
Chong Zhangf4abc2b2015-11-12 23:40:58 -08001709 final boolean surfaceSizeChanged = (relayoutResult
1710 & WindowManagerGlobal.RELAYOUT_RES_SURFACE_RESIZED) != 0;
Jorim Jaggi0ffd49c2016-02-12 15:04:21 -08001711 final boolean alwaysConsumeNavBarChanged =
1712 mPendingAlwaysConsumeNavBar != mAttachInfo.mAlwaysConsumeNavBar;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001713 if (contentInsetsChanged) {
1714 mAttachInfo.mContentInsets.set(mPendingContentInsets);
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08001715 if (DEBUG_LAYOUT) Log.v(mTag, "Content insets changing to: "
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001716 + mAttachInfo.mContentInsets);
1717 }
Dianne Hackbornc4aad012013-02-22 15:05:25 -08001718 if (overscanInsetsChanged) {
1719 mAttachInfo.mOverscanInsets.set(mPendingOverscanInsets);
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08001720 if (DEBUG_LAYOUT) Log.v(mTag, "Overscan insets changing to: "
Dianne Hackbornc4aad012013-02-22 15:05:25 -08001721 + mAttachInfo.mOverscanInsets);
1722 // Need to relayout with content insets.
1723 contentInsetsChanged = true;
1724 }
Adrian Roosfa104232014-06-20 16:10:14 -07001725 if (stableInsetsChanged) {
1726 mAttachInfo.mStableInsets.set(mPendingStableInsets);
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08001727 if (DEBUG_LAYOUT) Log.v(mTag, "Decor insets changing to: "
Adrian Roosfa104232014-06-20 16:10:14 -07001728 + mAttachInfo.mStableInsets);
1729 // Need to relayout with content insets.
1730 contentInsetsChanged = true;
1731 }
Jorim Jaggi0ffd49c2016-02-12 15:04:21 -08001732 if (alwaysConsumeNavBarChanged) {
1733 mAttachInfo.mAlwaysConsumeNavBar = mPendingAlwaysConsumeNavBar;
1734 contentInsetsChanged = true;
1735 }
Dianne Hackborn3a3a6cf2012-03-26 10:24:04 -07001736 if (contentInsetsChanged || mLastSystemUiVisibility !=
Adrian Roosfa104232014-06-20 16:10:14 -07001737 mAttachInfo.mSystemUiVisibility || mApplyInsetsRequested
Filip Gruszczynski2217f612015-05-26 11:32:08 -07001738 || mLastOverscanRequested != mAttachInfo.mOverscanRequested
1739 || outsetsChanged) {
Dianne Hackborn3a3a6cf2012-03-26 10:24:04 -07001740 mLastSystemUiVisibility = mAttachInfo.mSystemUiVisibility;
Dianne Hackbornc4aad012013-02-22 15:05:25 -08001741 mLastOverscanRequested = mAttachInfo.mOverscanRequested;
Filip Gruszczynski2217f612015-05-26 11:32:08 -07001742 mAttachInfo.mOutsets.set(mPendingOutsets);
Adrian Roosfa104232014-06-20 16:10:14 -07001743 mApplyInsetsRequested = false;
Adam Powell2accbf92014-04-16 23:14:57 +00001744 dispatchApplyInsets(host);
Dianne Hackborn3a3a6cf2012-03-26 10:24:04 -07001745 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001746 if (visibleInsetsChanged) {
1747 mAttachInfo.mVisibleInsets.set(mPendingVisibleInsets);
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08001748 if (DEBUG_LAYOUT) Log.v(mTag, "Visible insets changing to: "
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001749 + mAttachInfo.mVisibleInsets);
1750 }
1751
1752 if (!hadSurface) {
1753 if (mSurface.isValid()) {
1754 // If we are creating a new surface, then we need to
1755 // completely redraw it. Also, when we get to the
1756 // point of drawing it we will hold off and schedule
1757 // a new traversal instead. This is so we can tell the
1758 // window manager about all of the windows being displayed
1759 // before actually drawing them, so it can display then
1760 // all at once.
1761 newSurface = true;
Jeff Brown96e942d2011-11-30 19:55:01 -08001762 mFullRedrawNeeded = true;
Jack Palevich61a6e682009-10-09 17:37:50 -07001763 mPreviousTransparentRegion.setEmpty();
Romain Guy8506ab42009-06-11 17:35:47 -07001764
John Reck63005e62015-05-19 15:00:13 -07001765 // Only initialize up-front if transparent regions are not
1766 // requested, otherwise defer to see if the entire window
1767 // will be transparent
Romain Guyb051e892010-09-28 19:09:36 -07001768 if (mAttachInfo.mHardwareRenderer != null) {
Dianne Hackborn64825172011-03-02 21:32:58 -08001769 try {
Romain Guy786fc932012-07-24 16:24:56 -07001770 hwInitialized = mAttachInfo.mHardwareRenderer.initialize(
John Reck79d81e62013-11-05 13:26:57 -08001771 mSurface);
John Reck63005e62015-05-19 15:00:13 -07001772 if (hwInitialized && (host.mPrivateFlags
1773 & View.PFLAG_REQUEST_TRANSPARENT_REGIONS) == 0) {
1774 // Don't pre-allocate if transparent regions
1775 // are requested as they may not be needed
1776 mSurface.allocateBuffers();
1777 }
Igor Murashkina86ab6402013-08-30 12:58:36 -07001778 } catch (OutOfResourcesException e) {
Romain Guy3696779b2013-01-28 14:04:07 -08001779 handleOutOfResourcesException(e);
Dianne Hackborn64825172011-03-02 21:32:58 -08001780 return;
1781 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001782 }
1783 }
1784 } else if (!mSurface.isValid()) {
1785 // If the surface has been removed, then reset the scroll
1786 // positions.
Svetoslav Ganov149567f2013-01-08 15:23:34 -08001787 if (mLastScrolledFocus != null) {
1788 mLastScrolledFocus.clear();
1789 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001790 mScrollY = mCurScrollY = 0;
Adrian Roos0e7ae4e2014-10-01 15:33:22 +02001791 if (mView instanceof RootViewSurfaceTaker) {
1792 ((RootViewSurfaceTaker) mView).onRootViewScrollYChanged(mCurScrollY);
1793 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001794 if (mScroller != null) {
1795 mScroller.abortAnimation();
1796 }
Romain Guy1d0c7082011-08-03 16:22:24 -07001797 // Our surface is gone
1798 if (mAttachInfo.mHardwareRenderer != null &&
1799 mAttachInfo.mHardwareRenderer.isEnabled()) {
John Reckf47a5942014-06-30 16:20:04 -07001800 mAttachInfo.mHardwareRenderer.destroy();
Romain Guy1d0c7082011-08-03 16:22:24 -07001801 }
Chong Zhangf4abc2b2015-11-12 23:40:58 -08001802 } else if ((surfaceGenerationId != mSurface.getGenerationId()
1803 || surfaceSizeChanged)
1804 && mSurfaceHolder == null
1805 && mAttachInfo.mHardwareRenderer != null) {
Jeff Brown96e942d2011-11-30 19:55:01 -08001806 mFullRedrawNeeded = true;
Dianne Hackborn64825172011-03-02 21:32:58 -08001807 try {
Chong Zhangf4abc2b2015-11-12 23:40:58 -08001808 // Need to do updateSurface (which leads to CanvasContext::setSurface and
1809 // re-create the EGLSurface) if either the Surface changed (as indicated by
1810 // generation id), or WindowManager changed the surface size. The latter is
1811 // because on some chips, changing the consumer side's BufferQueue size may
1812 // not take effect immediately unless we create a new EGLSurface.
1813 // Note that frame size change doesn't always imply surface size change (eg.
1814 // drag resizing uses fullscreen surface), need to check surfaceSizeChanged
1815 // flag from WindowManager.
John Reck79d81e62013-11-05 13:26:57 -08001816 mAttachInfo.mHardwareRenderer.updateSurface(mSurface);
Igor Murashkina86ab6402013-08-30 12:58:36 -07001817 } catch (OutOfResourcesException e) {
Romain Guy3696779b2013-01-28 14:04:07 -08001818 handleOutOfResourcesException(e);
Dianne Hackborn64825172011-03-02 21:32:58 -08001819 return;
1820 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001821 }
Chong Zhang0275e392015-09-17 10:41:44 -07001822
Jorim Jaggi4846ee32016-01-07 17:39:12 +01001823 final boolean freeformResizing = (relayoutResult
1824 & WindowManagerGlobal.RELAYOUT_RES_DRAG_RESIZING_FREEFORM) != 0;
1825 final boolean dockedResizing = (relayoutResult
1826 & WindowManagerGlobal.RELAYOUT_RES_DRAG_RESIZING_DOCKED) != 0;
1827 final boolean dragResizing = freeformResizing || dockedResizing;
Chong Zhang0275e392015-09-17 10:41:44 -07001828 if (mDragResizing != dragResizing) {
Skuhneb8160872015-09-22 09:51:39 -07001829 if (dragResizing) {
Jorim Jaggi4846ee32016-01-07 17:39:12 +01001830 mResizeMode = freeformResizing
1831 ? RESIZE_MODE_FREEFORM
1832 : RESIZE_MODE_DOCKED_DIVIDER;
Jorim Jaggic39c7b02016-03-24 10:47:07 -07001833 startDragResizing(mPendingBackDropFrame,
1834 mWinFrame.equals(mPendingBackDropFrame), mPendingVisibleInsets,
1835 mPendingStableInsets, mResizeMode);
Skuhneb8160872015-09-22 09:51:39 -07001836 } else {
1837 // We shouldn't come here, but if we come we should end the resize.
1838 endDragResizing();
1839 }
Chong Zhang0275e392015-09-17 10:41:44 -07001840 }
Skuhneb8160872015-09-22 09:51:39 -07001841 if (!USE_MT_RENDERER) {
1842 if (dragResizing) {
1843 mCanvasOffsetX = mWinFrame.left;
1844 mCanvasOffsetY = mWinFrame.top;
1845 } else {
1846 mCanvasOffsetX = mCanvasOffsetY = 0;
1847 }
Chong Zhang0275e392015-09-17 10:41:44 -07001848 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001849 } catch (RemoteException e) {
1850 }
Romain Guy1d0c7082011-08-03 16:22:24 -07001851
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001852 if (DEBUG_ORIENTATION) Log.v(
Jeff Brownc5ed5912010-07-14 18:48:53 -07001853 TAG, "Relayout returned: frame=" + frame + ", surface=" + mSurface);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001854
Chris Craikd36a81f2014-07-17 10:16:51 -07001855 mAttachInfo.mWindowLeft = frame.left;
1856 mAttachInfo.mWindowTop = frame.top;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001857
1858 // !!FIXME!! This next section handles the case where we did not get the
1859 // window size we asked for. We should avoid this by getting a maximum size from
1860 // the window session beforehand.
Dianne Hackborn3556c9a2012-05-04 16:31:25 -07001861 if (mWidth != frame.width() || mHeight != frame.height()) {
Dianne Hackborn3556c9a2012-05-04 16:31:25 -07001862 mWidth = frame.width();
1863 mHeight = frame.height();
1864 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001865
Dianne Hackborndc8a7f62010-05-10 11:29:34 -07001866 if (mSurfaceHolder != null) {
1867 // The app owns the surface; tell it about what is going on.
1868 if (mSurface.isValid()) {
1869 // XXX .copyFrom() doesn't work!
1870 //mSurfaceHolder.mSurface.copyFrom(mSurface);
1871 mSurfaceHolder.mSurface = mSurface;
1872 }
Jeff Brown30bc34f2011-01-25 12:56:56 -08001873 mSurfaceHolder.setSurfaceFrameSize(mWidth, mHeight);
Dianne Hackborndc8a7f62010-05-10 11:29:34 -07001874 mSurfaceHolder.mSurfaceLock.unlock();
1875 if (mSurface.isValid()) {
1876 if (!hadSurface) {
1877 mSurfaceHolder.ungetCallbacks();
1878
1879 mIsCreating = true;
1880 mSurfaceHolderCallback.surfaceCreated(mSurfaceHolder);
1881 SurfaceHolder.Callback callbacks[] = mSurfaceHolder.getCallbacks();
1882 if (callbacks != null) {
1883 for (SurfaceHolder.Callback c : callbacks) {
1884 c.surfaceCreated(mSurfaceHolder);
1885 }
1886 }
1887 surfaceChanged = true;
1888 }
1889 if (surfaceChanged) {
1890 mSurfaceHolderCallback.surfaceChanged(mSurfaceHolder,
1891 lp.format, mWidth, mHeight);
1892 SurfaceHolder.Callback callbacks[] = mSurfaceHolder.getCallbacks();
1893 if (callbacks != null) {
1894 for (SurfaceHolder.Callback c : callbacks) {
1895 c.surfaceChanged(mSurfaceHolder, lp.format,
1896 mWidth, mHeight);
1897 }
1898 }
1899 }
1900 mIsCreating = false;
1901 } else if (hadSurface) {
1902 mSurfaceHolder.ungetCallbacks();
1903 SurfaceHolder.Callback callbacks[] = mSurfaceHolder.getCallbacks();
1904 mSurfaceHolderCallback.surfaceDestroyed(mSurfaceHolder);
1905 if (callbacks != null) {
1906 for (SurfaceHolder.Callback c : callbacks) {
1907 c.surfaceDestroyed(mSurfaceHolder);
1908 }
1909 }
1910 mSurfaceHolder.mSurfaceLock.lock();
Jozef BABJAK93c5b6a2011-02-22 09:33:19 +01001911 try {
1912 mSurfaceHolder.mSurface = new Surface();
1913 } finally {
1914 mSurfaceHolder.mSurfaceLock.unlock();
1915 }
Dianne Hackborndc8a7f62010-05-10 11:29:34 -07001916 }
1917 }
Romain Guy53389bd2010-09-07 17:16:32 -07001918
John Reck51aaf902015-12-02 15:08:07 -08001919 final ThreadedRenderer hardwareRenderer = mAttachInfo.mHardwareRenderer;
Alan Viverette50210d92015-05-14 18:05:36 -07001920 if (hardwareRenderer != null && hardwareRenderer.isEnabled()) {
1921 if (hwInitialized
1922 || mWidth != hardwareRenderer.getWidth()
1923 || mHeight != hardwareRenderer.getHeight()) {
1924 hardwareRenderer.setup(mWidth, mHeight, mAttachInfo,
1925 mWindowAttributes.surfaceInsets);
Romain Guy03985752011-07-11 15:33:51 -07001926 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001927 }
1928
Craig Mautner72d6f212015-02-19 16:33:09 -08001929 if (!mStopped || mReportNextDraw) {
Dianne Hackbornce418e62011-03-01 14:31:38 -08001930 boolean focusChangedDueToTouchMode = ensureTouchModeLocally(
Jeff Brown98365d72012-08-19 20:30:52 -07001931 (relayoutResult&WindowManagerGlobal.RELAYOUT_RES_IN_TOUCH_MODE) != 0);
Dianne Hackbornce418e62011-03-01 14:31:38 -08001932 if (focusChangedDueToTouchMode || mWidth != host.getMeasuredWidth()
1933 || mHeight != host.getMeasuredHeight() || contentInsetsChanged) {
Dianne Hackborn3a3a6cf2012-03-26 10:24:04 -07001934 int childWidthMeasureSpec = getRootMeasureSpec(mWidth, lp.width);
1935 int childHeightMeasureSpec = getRootMeasureSpec(mHeight, lp.height);
Igor Murashkina86ab6402013-08-30 12:58:36 -07001936
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08001937 if (DEBUG_LAYOUT) Log.v(mTag, "Ooops, something changed! mWidth="
Dianne Hackbornce418e62011-03-01 14:31:38 -08001938 + mWidth + " measuredWidth=" + host.getMeasuredWidth()
1939 + " mHeight=" + mHeight
1940 + " measuredHeight=" + host.getMeasuredHeight()
1941 + " coveredInsetsChanged=" + contentInsetsChanged);
Igor Murashkina86ab6402013-08-30 12:58:36 -07001942
Dianne Hackbornce418e62011-03-01 14:31:38 -08001943 // Ask host how big it wants to be
Jeff Brownc8d2668b2012-05-16 17:23:59 -07001944 performMeasure(childWidthMeasureSpec, childHeightMeasureSpec);
Igor Murashkina86ab6402013-08-30 12:58:36 -07001945
Dianne Hackbornce418e62011-03-01 14:31:38 -08001946 // Implementation of weights from WindowManager.LayoutParams
1947 // We just grow the dimensions as needed and re-measure if
1948 // needs be
1949 int width = host.getMeasuredWidth();
1950 int height = host.getMeasuredHeight();
1951 boolean measureAgain = false;
Igor Murashkina86ab6402013-08-30 12:58:36 -07001952
Dianne Hackbornce418e62011-03-01 14:31:38 -08001953 if (lp.horizontalWeight > 0.0f) {
1954 width += (int) ((mWidth - width) * lp.horizontalWeight);
1955 childWidthMeasureSpec = MeasureSpec.makeMeasureSpec(width,
1956 MeasureSpec.EXACTLY);
1957 measureAgain = true;
1958 }
1959 if (lp.verticalWeight > 0.0f) {
1960 height += (int) ((mHeight - height) * lp.verticalWeight);
1961 childHeightMeasureSpec = MeasureSpec.makeMeasureSpec(height,
1962 MeasureSpec.EXACTLY);
1963 measureAgain = true;
1964 }
Igor Murashkina86ab6402013-08-30 12:58:36 -07001965
Dianne Hackbornce418e62011-03-01 14:31:38 -08001966 if (measureAgain) {
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08001967 if (DEBUG_LAYOUT) Log.v(mTag,
Dianne Hackbornce418e62011-03-01 14:31:38 -08001968 "And hey let's measure once more: width=" + width
1969 + " height=" + height);
Jeff Brownc8d2668b2012-05-16 17:23:59 -07001970 performMeasure(childWidthMeasureSpec, childHeightMeasureSpec);
Dianne Hackbornce418e62011-03-01 14:31:38 -08001971 }
Igor Murashkina86ab6402013-08-30 12:58:36 -07001972
Dianne Hackborn3a3a6cf2012-03-26 10:24:04 -07001973 layoutRequested = true;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001974 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001975 }
Svetoslav Ganovc9c9a482012-07-16 08:46:07 -07001976 } else {
1977 // Not the first pass and no window/insets/visibility change but the window
1978 // may have moved and we need check that and if so to update the left and right
1979 // in the attach info. We translate only the window frame since on window move
1980 // the window manager tells us only for the new frame but the insets are the
1981 // same and we do not want to translate them more than once.
Jorim Jaggi2e95a482016-01-14 17:36:55 -08001982 maybeHandleWindowMove(frame);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001983 }
1984
Craig Mautner72d6f212015-02-19 16:33:09 -08001985 final boolean didLayout = layoutRequested && (!mStopped || mReportNextDraw);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001986 boolean triggerGlobalLayoutListener = didLayout
Chris Craikd36a81f2014-07-17 10:16:51 -07001987 || mAttachInfo.mRecomputeGlobalAttributes;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001988 if (didLayout) {
Jorim Jaggi26d02d22016-02-24 18:37:51 -05001989 performLayout(lp, mWidth, mHeight);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001990
Mathias Agopian54e3d3842013-04-12 15:13:12 -07001991 // By this point all views have been sized and positioned
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001992 // We can compute the transparent area
1993
Dianne Hackborn4702a852012-08-17 15:18:29 -07001994 if ((host.mPrivateFlags & View.PFLAG_REQUEST_TRANSPARENT_REGIONS) != 0) {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001995 // start out transparent
1996 // TODO: AVOID THAT CALL BY CACHING THE RESULT?
1997 host.getLocationInWindow(mTmpLocation);
1998 mTransparentRegion.set(mTmpLocation[0], mTmpLocation[1],
1999 mTmpLocation[0] + host.mRight - host.mLeft,
2000 mTmpLocation[1] + host.mBottom - host.mTop);
2001
2002 host.gatherTransparentRegion(mTransparentRegion);
Mitsuru Oshima64f59342009-06-21 00:03:11 -07002003 if (mTranslator != null) {
2004 mTranslator.translateRegionInWindowToScreen(mTransparentRegion);
2005 }
2006
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08002007 if (!mTransparentRegion.equals(mPreviousTransparentRegion)) {
2008 mPreviousTransparentRegion.set(mTransparentRegion);
Mathias Agopian54e3d3842013-04-12 15:13:12 -07002009 mFullRedrawNeeded = true;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08002010 // reconfigure window manager
2011 try {
Jeff Brown98365d72012-08-19 20:30:52 -07002012 mWindowSession.setTransparentRegion(mWindow, mTransparentRegion);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08002013 } catch (RemoteException e) {
2014 }
2015 }
2016 }
2017
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08002018 if (DBG) {
2019 System.out.println("======================================");
Chet Haase4610eef2015-12-03 07:38:11 -08002020 System.out.println("performTraversals -- after setFrame");
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08002021 host.debug();
2022 }
2023 }
2024
2025 if (triggerGlobalLayoutListener) {
Chris Craikd36a81f2014-07-17 10:16:51 -07002026 mAttachInfo.mRecomputeGlobalAttributes = false;
2027 mAttachInfo.mTreeObserver.dispatchOnGlobalLayout();
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08002028 }
2029
2030 if (computesInternalInsets) {
Jeff Brownfbf09772011-01-16 14:06:57 -08002031 // Clear the original insets.
Chris Craikd36a81f2014-07-17 10:16:51 -07002032 final ViewTreeObserver.InternalInsetsInfo insets = mAttachInfo.mGivenInternalInsets;
Jeff Brownfbf09772011-01-16 14:06:57 -08002033 insets.reset();
2034
2035 // Compute new insets in place.
Chris Craikd36a81f2014-07-17 10:16:51 -07002036 mAttachInfo.mTreeObserver.dispatchOnComputeInternalInsets(insets);
2037 mAttachInfo.mHasNonEmptyGivenInternalInsets = !insets.isEmpty();
Jeff Brownfbf09772011-01-16 14:06:57 -08002038
2039 // Tell the window manager.
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08002040 if (insetsPending || !mLastGivenInsets.equals(insets)) {
2041 mLastGivenInsets.set(insets);
Jeff Brownfbf09772011-01-16 14:06:57 -08002042
2043 // Translate insets to screen coordinates if needed.
2044 final Rect contentInsets;
2045 final Rect visibleInsets;
2046 final Region touchableRegion;
2047 if (mTranslator != null) {
2048 contentInsets = mTranslator.getTranslatedContentInsets(insets.contentInsets);
2049 visibleInsets = mTranslator.getTranslatedVisibleInsets(insets.visibleInsets);
2050 touchableRegion = mTranslator.getTranslatedTouchableArea(insets.touchableRegion);
2051 } else {
2052 contentInsets = insets.contentInsets;
2053 visibleInsets = insets.visibleInsets;
2054 touchableRegion = insets.touchableRegion;
2055 }
2056
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08002057 try {
Jeff Brown98365d72012-08-19 20:30:52 -07002058 mWindowSession.setInsets(mWindow, insets.mTouchableInsets,
Jeff Brownfbf09772011-01-16 14:06:57 -08002059 contentInsets, visibleInsets, touchableRegion);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08002060 } catch (RemoteException e) {
2061 }
2062 }
2063 }
Romain Guy8506ab42009-06-11 17:35:47 -07002064
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08002065 if (mFirst) {
2066 // handle first focus request
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08002067 if (DEBUG_INPUT_RESIZE) Log.v(mTag, "First: mView.hasFocus()="
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08002068 + mView.hasFocus());
2069 if (mView != null) {
2070 if (!mView.hasFocus()) {
2071 mView.requestFocus(View.FOCUS_FORWARD);
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08002072 if (DEBUG_INPUT_RESIZE) Log.v(mTag, "First: requested focused view="
Svetoslav Ganov149567f2013-01-08 15:23:34 -08002073 + mView.findFocus());
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08002074 } else {
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08002075 if (DEBUG_INPUT_RESIZE) Log.v(mTag, "First: existing focused view="
Svetoslav Ganov149567f2013-01-08 15:23:34 -08002076 + mView.findFocus());
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08002077 }
2078 }
2079 }
2080
Alan Viverette64bf97a2015-09-18 16:42:00 -04002081 final boolean changedVisibility = (viewVisibilityChanged || mFirst) && isViewVisible;
2082 final boolean hasWindowFocus = mAttachInfo.mHasWindowFocus && isViewVisible;
2083 final boolean regainedFocus = hasWindowFocus && mLostWindowFocus;
2084 if (regainedFocus) {
2085 mLostWindowFocus = false;
2086 } else if (!hasWindowFocus && mHadWindowFocus) {
2087 mLostWindowFocus = true;
2088 }
2089
2090 if (changedVisibility || regainedFocus) {
2091 host.sendAccessibilityEvent(AccessibilityEvent.TYPE_WINDOW_STATE_CHANGED);
2092 }
2093
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08002094 mFirst = false;
2095 mWillDrawSoon = false;
2096 mNewSurfaceNeeded = false;
Jorim Jaggi4846ee32016-01-07 17:39:12 +01002097 mActivityRelaunched = false;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08002098 mViewVisibility = viewVisibility;
Alan Viverette64bf97a2015-09-18 16:42:00 -04002099 mHadWindowFocus = hasWindowFocus;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08002100
Alan Viverette64bf97a2015-09-18 16:42:00 -04002101 if (hasWindowFocus && !isInLocalFocusMode()) {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08002102 final boolean imTarget = WindowManager.LayoutParams
2103 .mayUseInputMethod(mWindowAttributes.flags);
2104 if (imTarget != mLastWasImTarget) {
2105 mLastWasImTarget = imTarget;
2106 InputMethodManager imm = InputMethodManager.peekInstance();
2107 if (imm != null && imTarget) {
Alan Viverette64bf97a2015-09-18 16:42:00 -04002108 imm.onPreWindowFocus(mView, hasWindowFocus);
Yohei Yukawa5f059652015-05-14 22:16:41 -07002109 imm.onPostWindowFocus(mView, mView.findFocus(),
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08002110 mWindowAttributes.softInputMode,
2111 !mHasHadWindowFocus, mWindowAttributes.flags);
2112 }
2113 }
2114 }
Romain Guy8506ab42009-06-11 17:35:47 -07002115
Jeff Brown96e942d2011-11-30 19:55:01 -08002116 // Remember if we must report the next draw.
Jeff Brown98365d72012-08-19 20:30:52 -07002117 if ((relayoutResult & WindowManagerGlobal.RELAYOUT_RES_FIRST_TIME) != 0) {
Jeff Brown96e942d2011-11-30 19:55:01 -08002118 mReportNextDraw = true;
2119 }
2120
Alan Viverette64bf97a2015-09-18 16:42:00 -04002121 boolean cancelDraw = mAttachInfo.mTreeObserver.dispatchOnPreDraw() || !isViewVisible;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08002122
Jorim Jaggi3127c2a42016-01-26 18:48:44 -08002123 if (!cancelDraw) {
Chet Haase9c450412015-10-01 13:25:58 -07002124 if (mPendingTransitions != null && mPendingTransitions.size() > 0) {
2125 for (int i = 0; i < mPendingTransitions.size(); ++i) {
2126 mPendingTransitions.get(i).startChangingAnimations();
Chet Haased56c6952011-09-07 08:46:23 -07002127 }
Chet Haase9c450412015-10-01 13:25:58 -07002128 mPendingTransitions.clear();
Chet Haased56c6952011-09-07 08:46:23 -07002129 }
Chet Haase9c450412015-10-01 13:25:58 -07002130
2131 performDraw();
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08002132 } else {
Alan Viverette64bf97a2015-09-18 16:42:00 -04002133 if (isViewVisible) {
Chris Wren78cb7cf2012-05-15 12:36:44 -04002134 // Try again
2135 scheduleTraversals();
2136 } else if (mPendingTransitions != null && mPendingTransitions.size() > 0) {
Chet Haased56c6952011-09-07 08:46:23 -07002137 for (int i = 0; i < mPendingTransitions.size(); ++i) {
2138 mPendingTransitions.get(i).endChangingAnimations();
2139 }
2140 mPendingTransitions.clear();
2141 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08002142 }
Craig Mautnerdf2390a2012-08-29 20:59:22 -07002143
2144 mIsInTraversal = false;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08002145 }
2146
Jorim Jaggi2e95a482016-01-14 17:36:55 -08002147 private void maybeHandleWindowMove(Rect frame) {
2148
2149 // TODO: Well, we are checking whether the frame has changed similarly
2150 // to how this is done for the insets. This is however incorrect since
2151 // the insets and the frame are translated. For example, the old frame
2152 // was (1, 1 - 1, 1) and was translated to say (2, 2 - 2, 2), now the new
2153 // reported frame is (2, 2 - 2, 2) which implies no change but this is not
2154 // true since we are comparing a not translated value to a translated one.
2155 // This scenario is rare but we may want to fix that.
2156
2157 final boolean windowMoved = mAttachInfo.mWindowLeft != frame.left
2158 || mAttachInfo.mWindowTop != frame.top;
2159 if (windowMoved) {
2160 if (mTranslator != null) {
2161 mTranslator.translateRectInScreenToAppWinFrame(frame);
2162 }
2163 mAttachInfo.mWindowLeft = frame.left;
2164 mAttachInfo.mWindowTop = frame.top;
2165
2166 // Update the light position for the new window offsets.
2167 if (mAttachInfo.mHardwareRenderer != null) {
2168 mAttachInfo.mHardwareRenderer.setLightCenter(mAttachInfo);
2169 }
2170 }
2171 }
Jorim Jaggi3127c2a42016-01-26 18:48:44 -08002172
Romain Guy3696779b2013-01-28 14:04:07 -08002173 private void handleOutOfResourcesException(Surface.OutOfResourcesException e) {
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08002174 Log.e(mTag, "OutOfResourcesException initializing HW surface", e);
Romain Guy3696779b2013-01-28 14:04:07 -08002175 try {
2176 if (!mWindowSession.outOfMemory(mWindow) &&
2177 Process.myUid() != Process.SYSTEM_UID) {
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08002178 Slog.w(mTag, "No processes killed for memory; killing self");
Romain Guy3696779b2013-01-28 14:04:07 -08002179 Process.killProcess(Process.myPid());
2180 }
2181 } catch (RemoteException ex) {
2182 }
2183 mLayoutRequested = true; // ask wm for a new surface next time.
2184 }
2185
Jeff Brownc8d2668b2012-05-16 17:23:59 -07002186 private void performMeasure(int childWidthMeasureSpec, int childHeightMeasureSpec) {
2187 Trace.traceBegin(Trace.TRACE_TAG_VIEW, "measure");
2188 try {
2189 mView.measure(childWidthMeasureSpec, childHeightMeasureSpec);
2190 } finally {
2191 Trace.traceEnd(Trace.TRACE_TAG_VIEW);
2192 }
2193 }
2194
Chet Haase97140572012-09-13 14:56:47 -07002195 /**
2196 * Called by {@link android.view.View#isInLayout()} to determine whether the view hierarchy
2197 * is currently undergoing a layout pass.
2198 *
2199 * @return whether the view hierarchy is currently undergoing a layout pass
2200 */
2201 boolean isInLayout() {
2202 return mInLayout;
2203 }
2204
2205 /**
Chet Haasecc699b42012-12-13 09:06:55 -08002206 * Called by {@link android.view.View#requestLayout()} if the view hierarchy is currently
2207 * undergoing a layout pass. requestLayout() should not generally be called during layout,
2208 * unless the container hierarchy knows what it is doing (i.e., it is fine as long as
2209 * all children in that container hierarchy are measured and laid out at the end of the layout
2210 * pass for that container). If requestLayout() is called anyway, we handle it correctly
2211 * by registering all requesters during a frame as it proceeds. At the end of the frame,
2212 * we check all of those views to see if any still have pending layout requests, which
2213 * indicates that they were not correctly handled by their container hierarchy. If that is
2214 * the case, we clear all such flags in the tree, to remove the buggy flag state that leads
2215 * to blank containers, and force a second request/measure/layout pass in this frame. If
Chet Haase97140572012-09-13 14:56:47 -07002216 * more requestLayout() calls are received during that second layout pass, we post those
Chet Haasecc699b42012-12-13 09:06:55 -08002217 * requests to the next frame to avoid possible infinite loops.
2218 *
2219 * <p>The return value from this method indicates whether the request should proceed
2220 * (if it is a request during the first layout pass) or should be skipped and posted to the
2221 * next frame (if it is a request during the second layout pass).</p>
Chet Haase97140572012-09-13 14:56:47 -07002222 *
2223 * @param view the view that requested the layout.
Chet Haasecc699b42012-12-13 09:06:55 -08002224 *
2225 * @return true if request should proceed, false otherwise.
Chet Haase97140572012-09-13 14:56:47 -07002226 */
Chet Haasecc699b42012-12-13 09:06:55 -08002227 boolean requestLayoutDuringLayout(final View view) {
2228 if (view.mParent == null || view.mAttachInfo == null) {
2229 // Would not normally trigger another layout, so just let it pass through as usual
2230 return true;
2231 }
Chet Haase107a4822013-03-13 06:46:50 -07002232 if (!mLayoutRequesters.contains(view)) {
2233 mLayoutRequesters.add(view);
2234 }
Chet Haase97140572012-09-13 14:56:47 -07002235 if (!mHandlingLayoutInLayoutRequest) {
Chet Haase107a4822013-03-13 06:46:50 -07002236 // Let the request proceed normally; it will be processed in a second layout pass
2237 // if necessary
Chet Haasecc699b42012-12-13 09:06:55 -08002238 return true;
Chet Haase97140572012-09-13 14:56:47 -07002239 } else {
Chet Haase107a4822013-03-13 06:46:50 -07002240 // Don't let the request proceed during the second layout pass.
2241 // It will post to the next frame instead.
Chet Haasecc699b42012-12-13 09:06:55 -08002242 return false;
Chet Haase97140572012-09-13 14:56:47 -07002243 }
2244 }
2245
Chet Haase3efa7b52012-12-03 08:33:17 -08002246 private void performLayout(WindowManager.LayoutParams lp, int desiredWindowWidth,
2247 int desiredWindowHeight) {
Jeff Brownc8d2668b2012-05-16 17:23:59 -07002248 mLayoutRequested = false;
2249 mScrollMayChange = true;
Chet Haase97140572012-09-13 14:56:47 -07002250 mInLayout = true;
Jeff Brownc8d2668b2012-05-16 17:23:59 -07002251
2252 final View host = mView;
2253 if (DEBUG_ORIENTATION || DEBUG_LAYOUT) {
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08002254 Log.v(mTag, "Laying out " + host + " to (" +
Jeff Brownc8d2668b2012-05-16 17:23:59 -07002255 host.getMeasuredWidth() + ", " + host.getMeasuredHeight() + ")");
2256 }
2257
Jeff Brownc8d2668b2012-05-16 17:23:59 -07002258 Trace.traceBegin(Trace.TRACE_TAG_VIEW, "layout");
2259 try {
2260 host.layout(0, 0, host.getMeasuredWidth(), host.getMeasuredHeight());
Chet Haasecc699b42012-12-13 09:06:55 -08002261
Chet Haased5a83522012-11-21 16:24:44 -08002262 mInLayout = false;
Chet Haase97140572012-09-13 14:56:47 -07002263 int numViewsRequestingLayout = mLayoutRequesters.size();
2264 if (numViewsRequestingLayout > 0) {
Chet Haasecc699b42012-12-13 09:06:55 -08002265 // requestLayout() was called during layout.
2266 // If no layout-request flags are set on the requesting views, there is no problem.
2267 // If some requests are still pending, then we need to clear those flags and do
2268 // a full request/measure/layout pass to handle this situation.
Chet Haase107a4822013-03-13 06:46:50 -07002269 ArrayList<View> validLayoutRequesters = getValidLayoutRequesters(mLayoutRequesters,
2270 false);
2271 if (validLayoutRequesters != null) {
2272 // Set this flag to indicate that any further requests are happening during
2273 // the second pass, which may result in posting those requests to the next
2274 // frame instead
Chet Haasecc699b42012-12-13 09:06:55 -08002275 mHandlingLayoutInLayoutRequest = true;
Chet Haase107a4822013-03-13 06:46:50 -07002276
2277 // Process fresh layout requests, then measure and layout
2278 int numValidRequests = validLayoutRequesters.size();
Chet Haasecc699b42012-12-13 09:06:55 -08002279 for (int i = 0; i < numValidRequests; ++i) {
Chet Haase107a4822013-03-13 06:46:50 -07002280 final View view = validLayoutRequesters.get(i);
2281 Log.w("View", "requestLayout() improperly called by " + view +
2282 " during layout: running second layout pass");
2283 view.requestLayout();
Chet Haasecc699b42012-12-13 09:06:55 -08002284 }
2285 measureHierarchy(host, lp, mView.getContext().getResources(),
2286 desiredWindowWidth, desiredWindowHeight);
2287 mInLayout = true;
2288 host.layout(0, 0, host.getMeasuredWidth(), host.getMeasuredHeight());
Chet Haase107a4822013-03-13 06:46:50 -07002289
Chet Haasecc699b42012-12-13 09:06:55 -08002290 mHandlingLayoutInLayoutRequest = false;
Chet Haase107a4822013-03-13 06:46:50 -07002291
2292 // Check the valid requests again, this time without checking/clearing the
2293 // layout flags, since requests happening during the second pass get noop'd
2294 validLayoutRequesters = getValidLayoutRequesters(mLayoutRequesters, true);
2295 if (validLayoutRequesters != null) {
2296 final ArrayList<View> finalRequesters = validLayoutRequesters;
2297 // Post second-pass requests to the next frame
2298 getRunQueue().post(new Runnable() {
2299 @Override
2300 public void run() {
2301 int numValidRequests = finalRequesters.size();
2302 for (int i = 0; i < numValidRequests; ++i) {
2303 final View view = finalRequesters.get(i);
2304 Log.w("View", "requestLayout() improperly called by " + view +
2305 " during second layout pass: posting in next frame");
2306 view.requestLayout();
2307 }
2308 }
2309 });
2310 }
Chet Haasecc699b42012-12-13 09:06:55 -08002311 }
Chet Haase107a4822013-03-13 06:46:50 -07002312
Chet Haase97140572012-09-13 14:56:47 -07002313 }
Jeff Brownc8d2668b2012-05-16 17:23:59 -07002314 } finally {
2315 Trace.traceEnd(Trace.TRACE_TAG_VIEW);
2316 }
Chet Haase97140572012-09-13 14:56:47 -07002317 mInLayout = false;
Jeff Brownc8d2668b2012-05-16 17:23:59 -07002318 }
2319
Chet Haase107a4822013-03-13 06:46:50 -07002320 /**
2321 * This method is called during layout when there have been calls to requestLayout() during
2322 * layout. It walks through the list of views that requested layout to determine which ones
2323 * still need it, based on visibility in the hierarchy and whether they have already been
2324 * handled (as is usually the case with ListView children).
2325 *
2326 * @param layoutRequesters The list of views that requested layout during layout
2327 * @param secondLayoutRequests Whether the requests were issued during the second layout pass.
2328 * If so, the FORCE_LAYOUT flag was not set on requesters.
2329 * @return A list of the actual views that still need to be laid out.
2330 */
2331 private ArrayList<View> getValidLayoutRequesters(ArrayList<View> layoutRequesters,
2332 boolean secondLayoutRequests) {
2333
2334 int numViewsRequestingLayout = layoutRequesters.size();
2335 ArrayList<View> validLayoutRequesters = null;
2336 for (int i = 0; i < numViewsRequestingLayout; ++i) {
2337 View view = layoutRequesters.get(i);
2338 if (view != null && view.mAttachInfo != null && view.mParent != null &&
2339 (secondLayoutRequests || (view.mPrivateFlags & View.PFLAG_FORCE_LAYOUT) ==
2340 View.PFLAG_FORCE_LAYOUT)) {
2341 boolean gone = false;
2342 View parent = view;
2343 // Only trigger new requests for views in a non-GONE hierarchy
2344 while (parent != null) {
2345 if ((parent.mViewFlags & View.VISIBILITY_MASK) == View.GONE) {
2346 gone = true;
2347 break;
2348 }
2349 if (parent.mParent instanceof View) {
2350 parent = (View) parent.mParent;
2351 } else {
2352 parent = null;
2353 }
2354 }
2355 if (!gone) {
2356 if (validLayoutRequesters == null) {
2357 validLayoutRequesters = new ArrayList<View>();
2358 }
2359 validLayoutRequesters.add(view);
2360 }
2361 }
2362 }
2363 if (!secondLayoutRequests) {
2364 // If we're checking the layout flags, then we need to clean them up also
2365 for (int i = 0; i < numViewsRequestingLayout; ++i) {
2366 View view = layoutRequesters.get(i);
2367 while (view != null &&
2368 (view.mPrivateFlags & View.PFLAG_FORCE_LAYOUT) != 0) {
2369 view.mPrivateFlags &= ~View.PFLAG_FORCE_LAYOUT;
2370 if (view.mParent instanceof View) {
2371 view = (View) view.mParent;
2372 } else {
2373 view = null;
2374 }
2375 }
2376 }
2377 }
2378 layoutRequesters.clear();
2379 return validLayoutRequesters;
2380 }
2381
Igor Murashkina86ab6402013-08-30 12:58:36 -07002382 @Override
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08002383 public void requestTransparentRegion(View child) {
2384 // the test below should not fail unless someone is messing with us
2385 checkThread();
2386 if (mView == child) {
Dianne Hackborn4702a852012-08-17 15:18:29 -07002387 mView.mPrivateFlags |= View.PFLAG_REQUEST_TRANSPARENT_REGIONS;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08002388 // Need to make sure we re-evaluate the window attributes next
2389 // time around, to ensure the window has the correct format.
2390 mWindowAttributesChanged = true;
Romain Guyf21c9b02011-09-06 16:56:54 -07002391 mWindowAttributesChangesFlag = 0;
Mathias Agopian1bd80ad2010-11-04 17:13:39 -07002392 requestLayout();
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08002393 }
2394 }
2395
2396 /**
2397 * Figures out the measure spec for the root view in a window based on it's
2398 * layout params.
2399 *
2400 * @param windowSize
2401 * The available width or height of the window
2402 *
2403 * @param rootDimension
2404 * The layout params for one dimension (width or height) of the
2405 * window.
2406 *
2407 * @return The measure spec to use to measure the root view.
2408 */
Romain Guya998dff2012-03-23 18:58:36 -07002409 private static int getRootMeasureSpec(int windowSize, int rootDimension) {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08002410 int measureSpec;
2411 switch (rootDimension) {
2412
Romain Guy980a9382010-01-08 15:06:28 -08002413 case ViewGroup.LayoutParams.MATCH_PARENT:
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08002414 // Window can't resize. Force root view to be windowSize.
2415 measureSpec = MeasureSpec.makeMeasureSpec(windowSize, MeasureSpec.EXACTLY);
2416 break;
2417 case ViewGroup.LayoutParams.WRAP_CONTENT:
2418 // Window can resize. Set max size for root view.
2419 measureSpec = MeasureSpec.makeMeasureSpec(windowSize, MeasureSpec.AT_MOST);
2420 break;
2421 default:
2422 // Window wants to be an exact size. Force root view to be that size.
2423 measureSpec = MeasureSpec.makeMeasureSpec(rootDimension, MeasureSpec.EXACTLY);
2424 break;
2425 }
2426 return measureSpec;
2427 }
2428
Alan Viveretteccb11e12014-07-08 16:04:02 -07002429 int mHardwareXOffset;
Dianne Hackborn0f761d62010-11-30 22:06:10 -08002430 int mHardwareYOffset;
Dianne Hackborn0f761d62010-11-30 22:06:10 -08002431
Igor Murashkina86ab6402013-08-30 12:58:36 -07002432 @Override
Chris Craikf6829a02015-03-10 10:28:59 -07002433 public void onHardwarePreDraw(DisplayListCanvas canvas) {
Alan Viveretteccb11e12014-07-08 16:04:02 -07002434 canvas.translate(-mHardwareXOffset, -mHardwareYOffset);
Dianne Hackborn0f761d62010-11-30 22:06:10 -08002435 }
2436
Igor Murashkina86ab6402013-08-30 12:58:36 -07002437 @Override
Chris Craikf6829a02015-03-10 10:28:59 -07002438 public void onHardwarePostDraw(DisplayListCanvas canvas) {
Alan Viverette632af842014-10-28 13:45:11 -07002439 drawAccessibilityFocusedDrawableIfNeeded(canvas);
Jorim Jaggic39c7b02016-03-24 10:47:07 -07002440 synchronized (mWindowCallbacks) {
2441 for (int i = mWindowCallbacks.size() - 1; i >= 0; i--) {
2442 mWindowCallbacks.get(i).onPostDraw(canvas);
2443 }
2444 }
Dianne Hackborn0f761d62010-11-30 22:06:10 -08002445 }
2446
Chet Haaseed30fd82011-04-22 16:18:45 -07002447 /**
2448 * @hide
2449 */
2450 void outputDisplayList(View view) {
Chris Craik356b5fe2015-07-07 10:39:36 -07002451 view.mRenderNode.output();
John Recke248bd12015-08-05 13:53:53 -07002452 if (mAttachInfo.mHardwareRenderer != null) {
2453 ((ThreadedRenderer)mAttachInfo.mHardwareRenderer).serializeDisplayListTree();
2454 }
Romain Guy59a12ca2011-06-09 17:48:21 -07002455 }
2456
2457 /**
2458 * @see #PROPERTY_PROFILE_RENDERING
2459 */
2460 private void profileRendering(boolean enabled) {
2461 if (mProfileRendering) {
2462 mRenderProfilingEnabled = enabled;
Chris Craikae4f32042013-02-07 12:57:10 -08002463
2464 if (mRenderProfiler != null) {
2465 mChoreographer.removeFrameCallback(mRenderProfiler);
2466 }
2467 if (mRenderProfilingEnabled) {
2468 if (mRenderProfiler == null) {
2469 mRenderProfiler = new Choreographer.FrameCallback() {
2470 @Override
2471 public void doFrame(long frameTimeNanos) {
2472 mDirty.set(0, 0, mWidth, mHeight);
2473 scheduleTraversals();
2474 if (mRenderProfilingEnabled) {
2475 mChoreographer.postFrameCallback(mRenderProfiler);
2476 }
Romain Guy59a12ca2011-06-09 17:48:21 -07002477 }
Chris Craikae4f32042013-02-07 12:57:10 -08002478 };
2479 }
Romain Guy5bb3c732012-11-29 17:52:58 -08002480 mChoreographer.postFrameCallback(mRenderProfiler);
Romain Guy59a12ca2011-06-09 17:48:21 -07002481 } else {
Romain Guy59a12ca2011-06-09 17:48:21 -07002482 mRenderProfiler = null;
Chet Haaseed30fd82011-04-22 16:18:45 -07002483 }
2484 }
2485 }
2486
Chet Haase2f2022a2011-10-11 06:41:59 -07002487 /**
2488 * Called from draw() when DEBUG_FPS is enabled
2489 */
2490 private void trackFPS() {
2491 // Tracks frames per second drawn. First value in a series of draws may be bogus
2492 // because it down not account for the intervening idle time
2493 long nowTime = System.currentTimeMillis();
2494 if (mFpsStartTime < 0) {
2495 mFpsStartTime = mFpsPrevTime = nowTime;
2496 mFpsNumFrames = 0;
2497 } else {
2498 ++mFpsNumFrames;
2499 String thisHash = Integer.toHexString(System.identityHashCode(this));
2500 long frameTime = nowTime - mFpsPrevTime;
2501 long totalTime = nowTime - mFpsStartTime;
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08002502 Log.v(mTag, "0x" + thisHash + "\tFrame time:\t" + frameTime);
Chet Haase2f2022a2011-10-11 06:41:59 -07002503 mFpsPrevTime = nowTime;
2504 if (totalTime > 1000) {
2505 float fps = (float) mFpsNumFrames * 1000 / totalTime;
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08002506 Log.v(mTag, "0x" + thisHash + "\tFPS:\t" + fps);
Chet Haase2f2022a2011-10-11 06:41:59 -07002507 mFpsStartTime = nowTime;
2508 mFpsNumFrames = 0;
2509 }
2510 }
2511 }
2512
Jeff Brown96e942d2011-11-30 19:55:01 -08002513 private void performDraw() {
Jeff Brownd912e1f2014-04-11 18:46:22 -07002514 if (mAttachInfo.mDisplayState == Display.STATE_OFF && !mReportNextDraw) {
Craig Mautner006f0e42012-03-21 11:00:32 -07002515 return;
2516 }
Romain Guy7e4e5612012-03-05 14:37:29 -08002517
Jeff Brown96e942d2011-11-30 19:55:01 -08002518 final boolean fullRedrawNeeded = mFullRedrawNeeded;
2519 mFullRedrawNeeded = false;
Jeff Brown481c1572012-03-09 14:41:15 -08002520
Romain Guy1f59e5c2012-05-06 14:11:16 -07002521 mIsDrawing = true;
Jeff Brown481c1572012-03-09 14:41:15 -08002522 Trace.traceBegin(Trace.TRACE_TAG_VIEW, "draw");
2523 try {
2524 draw(fullRedrawNeeded);
2525 } finally {
Romain Guy1f59e5c2012-05-06 14:11:16 -07002526 mIsDrawing = false;
Jeff Brown481c1572012-03-09 14:41:15 -08002527 Trace.traceEnd(Trace.TRACE_TAG_VIEW);
2528 }
Jeff Brown96e942d2011-11-30 19:55:01 -08002529
John Reck119907c2014-08-14 09:02:01 -07002530 // For whatever reason we didn't create a HardwareRenderer, end any
2531 // hardware animations that are now dangling
2532 if (mAttachInfo.mPendingAnimatingRenderNodes != null) {
2533 final int count = mAttachInfo.mPendingAnimatingRenderNodes.size();
2534 for (int i = 0; i < count; i++) {
2535 mAttachInfo.mPendingAnimatingRenderNodes.get(i).endAllAnimators();
2536 }
2537 mAttachInfo.mPendingAnimatingRenderNodes.clear();
2538 }
2539
Jeff Brown96e942d2011-11-30 19:55:01 -08002540 if (mReportNextDraw) {
2541 mReportNextDraw = false;
Chong Zhang8dbd9ad2015-10-09 10:06:11 -07002542
2543 // if we're using multi-thread renderer, wait for the window frame draws
2544 if (mWindowDrawCountDown != null) {
2545 try {
2546 mWindowDrawCountDown.await();
2547 } catch (InterruptedException e) {
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08002548 Log.e(mTag, "Window redraw count down interruped!");
Chong Zhang8dbd9ad2015-10-09 10:06:11 -07002549 }
2550 mWindowDrawCountDown = null;
2551 }
2552
John Reck28ad7b52014-04-07 16:59:25 -07002553 if (mAttachInfo.mHardwareRenderer != null) {
2554 mAttachInfo.mHardwareRenderer.fence();
2555 }
Jeff Brown96e942d2011-11-30 19:55:01 -08002556
2557 if (LOCAL_LOGV) {
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08002558 Log.v(mTag, "FINISHED DRAWING: " + mWindowAttributes.getTitle());
Jeff Brown96e942d2011-11-30 19:55:01 -08002559 }
2560 if (mSurfaceHolder != null && mSurface.isValid()) {
2561 mSurfaceHolderCallback.surfaceRedrawNeeded(mSurfaceHolder);
2562 SurfaceHolder.Callback callbacks[] = mSurfaceHolder.getCallbacks();
2563 if (callbacks != null) {
2564 for (SurfaceHolder.Callback c : callbacks) {
2565 if (c instanceof SurfaceHolder.Callback2) {
Filip Gruszczynski1a4dfe52015-11-15 10:58:57 -08002566 ((SurfaceHolder.Callback2)c).surfaceRedrawNeeded(mSurfaceHolder);
Jeff Brown96e942d2011-11-30 19:55:01 -08002567 }
2568 }
2569 }
2570 }
2571 try {
Jeff Brown98365d72012-08-19 20:30:52 -07002572 mWindowSession.finishDrawing(mWindow);
Jeff Brown96e942d2011-11-30 19:55:01 -08002573 } catch (RemoteException e) {
2574 }
2575 }
2576 }
2577
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08002578 private void draw(boolean fullRedrawNeeded) {
2579 Surface surface = mSurface;
Romain Guyfbb93fa2012-12-03 18:50:35 -08002580 if (!surface.isValid()) {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08002581 return;
2582 }
2583
Chet Haase2f2022a2011-10-11 06:41:59 -07002584 if (DEBUG_FPS) {
2585 trackFPS();
2586 }
2587
Dianne Hackborn2a9094d2010-02-03 19:20:09 -08002588 if (!sFirstDrawComplete) {
2589 synchronized (sFirstDrawHandlers) {
2590 sFirstDrawComplete = true;
Romain Guy812ccbe2010-06-01 14:07:24 -07002591 final int count = sFirstDrawHandlers.size();
2592 for (int i = 0; i< count; i++) {
Jeff Browna175a5b2012-02-15 19:18:31 -08002593 mHandler.post(sFirstDrawHandlers.get(i));
Dianne Hackborn2a9094d2010-02-03 19:20:09 -08002594 }
2595 }
2596 }
Romain Guy59a12ca2011-06-09 17:48:21 -07002597
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08002598 scrollToRectOrFocus(null, false);
2599
Chris Craikd36a81f2014-07-17 10:16:51 -07002600 if (mAttachInfo.mViewScrollChanged) {
2601 mAttachInfo.mViewScrollChanged = false;
2602 mAttachInfo.mTreeObserver.dispatchOnScrollChanged();
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08002603 }
Romain Guy8506ab42009-06-11 17:35:47 -07002604
Dianne Hackborn0f761d62010-11-30 22:06:10 -08002605 boolean animating = mScroller != null && mScroller.computeScrollOffset();
Alan Viveretteccb11e12014-07-08 16:04:02 -07002606 final int curScrollY;
Dianne Hackborn0f761d62010-11-30 22:06:10 -08002607 if (animating) {
Alan Viveretteccb11e12014-07-08 16:04:02 -07002608 curScrollY = mScroller.getCurrY();
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08002609 } else {
Alan Viveretteccb11e12014-07-08 16:04:02 -07002610 curScrollY = mScrollY;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08002611 }
Alan Viveretteccb11e12014-07-08 16:04:02 -07002612 if (mCurScrollY != curScrollY) {
2613 mCurScrollY = curScrollY;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08002614 fullRedrawNeeded = true;
Adrian Roos0e7ae4e2014-10-01 15:33:22 +02002615 if (mView instanceof RootViewSurfaceTaker) {
2616 ((RootViewSurfaceTaker) mView).onRootViewScrollYChanged(mCurScrollY);
2617 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08002618 }
Jeff Brown96e942d2011-11-30 19:55:01 -08002619
Chris Craikd36a81f2014-07-17 10:16:51 -07002620 final float appScale = mAttachInfo.mApplicationScale;
2621 final boolean scalingRequired = mAttachInfo.mScalingRequired;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08002622
Dianne Hackborn0f761d62010-11-30 22:06:10 -08002623 int resizeAlpha = 0;
Dianne Hackborn0f761d62010-11-30 22:06:10 -08002624
Jeff Brown96e942d2011-11-30 19:55:01 -08002625 final Rect dirty = mDirty;
Dianne Hackborndc8a7f62010-05-10 11:29:34 -07002626 if (mSurfaceHolder != null) {
2627 // The app owns the surface, we won't draw.
2628 dirty.setEmpty();
Derek Sollenberger8d948352015-07-16 09:27:59 -04002629 if (animating && mScroller != null) {
2630 mScroller.abortAnimation();
Dianne Hackborn0f761d62010-11-30 22:06:10 -08002631 }
Dianne Hackborndc8a7f62010-05-10 11:29:34 -07002632 return;
2633 }
Romain Guy58ef7fb2010-09-13 12:52:37 -07002634
2635 if (fullRedrawNeeded) {
Chris Craikd36a81f2014-07-17 10:16:51 -07002636 mAttachInfo.mIgnoreDirtyState = true;
Romain Guyc3166e12011-08-08 15:00:03 -07002637 dirty.set(0, 0, (int) (mWidth * appScale + 0.5f), (int) (mHeight * appScale + 0.5f));
Romain Guy58ef7fb2010-09-13 12:52:37 -07002638 }
Chet Haasead4f7032011-06-22 09:18:31 -07002639
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08002640 if (DEBUG_ORIENTATION || DEBUG_DRAW) {
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08002641 Log.v(mTag, "Draw " + mView + "/"
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08002642 + mWindowAttributes.getTitle()
2643 + ": dirty={" + dirty.left + "," + dirty.top
2644 + "," + dirty.right + "," + dirty.bottom + "} surface="
Mitsuru Oshima9189cab2009-06-03 11:19:12 -07002645 + surface + " surface.isValid()=" + surface.isValid() + ", appScale:" +
2646 appScale + ", width=" + mWidth + ", height=" + mHeight);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08002647 }
2648
Chris Craikd36a81f2014-07-17 10:16:51 -07002649 mAttachInfo.mTreeObserver.dispatchOnDraw();
Romain Guy25eba5c2012-04-04 17:29:03 -07002650
Chong Zhang0275e392015-09-17 10:41:44 -07002651 int xOffset = -mCanvasOffsetX;
2652 int yOffset = -mCanvasOffsetY + curScrollY;
Alan Viverettea51cab92014-07-16 15:15:49 -07002653 final WindowManager.LayoutParams params = mWindowAttributes;
2654 final Rect surfaceInsets = params != null ? params.surfaceInsets : null;
2655 if (surfaceInsets != null) {
2656 xOffset -= surfaceInsets.left;
2657 yOffset -= surfaceInsets.top;
2658
2659 // Offset dirty rect for surface insets.
2660 dirty.offset(surfaceInsets.left, surfaceInsets.right);
2661 }
2662
Alan Viverette632af842014-10-28 13:45:11 -07002663 boolean accessibilityFocusDirty = false;
2664 final Drawable drawable = mAttachInfo.mAccessibilityFocusDrawable;
2665 if (drawable != null) {
2666 final Rect bounds = mAttachInfo.mTmpInvalRect;
2667 final boolean hasFocus = getAccessibilityFocusedRect(bounds);
2668 if (!hasFocus) {
2669 bounds.setEmpty();
2670 }
2671 if (!bounds.equals(drawable.getBounds())) {
2672 accessibilityFocusDirty = true;
2673 }
2674 }
2675
John Reckba6adf62015-02-19 14:36:50 -08002676 mAttachInfo.mDrawingTime =
2677 mChoreographer.getFrameTimeNanos() / TimeUtils.NANOS_PER_MS;
2678
Alan Viverette632af842014-10-28 13:45:11 -07002679 if (!dirty.isEmpty() || mIsAnimating || accessibilityFocusDirty) {
Chris Craikd36a81f2014-07-17 10:16:51 -07002680 if (mAttachInfo.mHardwareRenderer != null && mAttachInfo.mHardwareRenderer.isEnabled()) {
Alan Viverette632af842014-10-28 13:45:11 -07002681 // If accessibility focus moved, always invalidate the root.
Jorim Jaggic39c7b02016-03-24 10:47:07 -07002682 boolean invalidateRoot = accessibilityFocusDirty || mInvalidateRootRequested;
2683 mInvalidateRootRequested = false;
Alan Viverette632af842014-10-28 13:45:11 -07002684
Jeff Brown96e942d2011-11-30 19:55:01 -08002685 // Draw with hardware renderer.
2686 mIsAnimating = false;
Alan Viverette632af842014-10-28 13:45:11 -07002687
John Reck0a973302014-07-16 13:29:45 -07002688 if (mHardwareYOffset != yOffset || mHardwareXOffset != xOffset) {
2689 mHardwareYOffset = yOffset;
2690 mHardwareXOffset = xOffset;
Alan Viverette632af842014-10-28 13:45:11 -07002691 invalidateRoot = true;
Alan Viverettef6cf1a02014-08-11 14:13:02 -07002692 }
2693
Alan Viverette632af842014-10-28 13:45:11 -07002694 if (invalidateRoot) {
2695 mAttachInfo.mHardwareRenderer.invalidateRoot();
2696 }
2697
Jeff Brown96e942d2011-11-30 19:55:01 -08002698 dirty.setEmpty();
2699
Skuhne980ee472015-10-06 11:31:31 -07002700 // Stage the content drawn size now. It will be transferred to the renderer
2701 // shortly before the draw commands get send to the renderer.
Chong Zhang8dbd9ad2015-10-09 10:06:11 -07002702 final boolean updated = updateContentDrawBounds();
2703
John Reck61375a82014-09-18 19:27:48 +00002704 mAttachInfo.mHardwareRenderer.draw(mView, mAttachInfo, this);
Chong Zhang8dbd9ad2015-10-09 10:06:11 -07002705
2706 if (updated) {
2707 requestDrawWindow();
2708 }
Romain Guy3696779b2013-01-28 14:04:07 -08002709 } else {
2710 // If we get here with a disabled & requested hardware renderer, something went
2711 // wrong (an invalidate posted right before we destroyed the hardware surface
2712 // for instance) so we should just bail out. Locking the surface with software
2713 // rendering at this point would lock it forever and prevent hardware renderer
2714 // from doing its job when it comes back.
2715 // Before we request a new frame we must however attempt to reinitiliaze the
2716 // hardware renderer if it's in requested state. This would happen after an
2717 // eglTerminate() for instance.
Chris Craikd36a81f2014-07-17 10:16:51 -07002718 if (mAttachInfo.mHardwareRenderer != null &&
2719 !mAttachInfo.mHardwareRenderer.isEnabled() &&
2720 mAttachInfo.mHardwareRenderer.isRequested()) {
Romain Guy3696779b2013-01-28 14:04:07 -08002721
2722 try {
Chris Craikd36a81f2014-07-17 10:16:51 -07002723 mAttachInfo.mHardwareRenderer.initializeIfNeeded(
Alan Viverette50210d92015-05-14 18:05:36 -07002724 mWidth, mHeight, mAttachInfo, mSurface, surfaceInsets);
Igor Murashkina86ab6402013-08-30 12:58:36 -07002725 } catch (OutOfResourcesException e) {
Romain Guy3696779b2013-01-28 14:04:07 -08002726 handleOutOfResourcesException(e);
2727 return;
2728 }
2729
2730 mFullRedrawNeeded = true;
2731 scheduleTraversals();
2732 return;
2733 }
2734
Chris Craikd36a81f2014-07-17 10:16:51 -07002735 if (!drawSoftware(surface, mAttachInfo, xOffset, yOffset, scalingRequired, dirty)) {
Romain Guy3696779b2013-01-28 14:04:07 -08002736 return;
2737 }
Jeff Brown95db2b22011-11-30 19:54:41 -08002738 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08002739 }
Romain Guy8506ab42009-06-11 17:35:47 -07002740
Dianne Hackborn0f761d62010-11-30 22:06:10 -08002741 if (animating) {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08002742 mFullRedrawNeeded = true;
2743 scheduleTraversals();
2744 }
2745 }
2746
Romain Guy25eba5c2012-04-04 17:29:03 -07002747 /**
Alan Viveretteccb11e12014-07-08 16:04:02 -07002748 * @return true if drawing was successful, false if an error occurred
Romain Guy25eba5c2012-04-04 17:29:03 -07002749 */
Alan Viveretteccb11e12014-07-08 16:04:02 -07002750 private boolean drawSoftware(Surface surface, AttachInfo attachInfo, int xoff, int yoff,
Romain Guy25eba5c2012-04-04 17:29:03 -07002751 boolean scalingRequired, Rect dirty) {
2752
2753 // Draw with software renderer.
Alan Viverettea51cab92014-07-16 15:15:49 -07002754 final Canvas canvas;
Romain Guy25eba5c2012-04-04 17:29:03 -07002755 try {
Alan Viverettea51cab92014-07-16 15:15:49 -07002756 final int left = dirty.left;
2757 final int top = dirty.top;
2758 final int right = dirty.right;
2759 final int bottom = dirty.bottom;
Romain Guy25eba5c2012-04-04 17:29:03 -07002760
Romain Guy25eba5c2012-04-04 17:29:03 -07002761 canvas = mSurface.lockCanvas(dirty);
2762
Romain Guye55945e2013-04-04 15:26:04 -07002763 // The dirty rectangle can be modified by Surface.lockCanvas()
2764 //noinspection ConstantConditions
Alan Viverettea51cab92014-07-16 15:15:49 -07002765 if (left != dirty.left || top != dirty.top || right != dirty.right
2766 || bottom != dirty.bottom) {
Romain Guy25eba5c2012-04-04 17:29:03 -07002767 attachInfo.mIgnoreDirtyState = true;
2768 }
2769
2770 // TODO: Do this in native
2771 canvas.setDensity(mDensity);
2772 } catch (Surface.OutOfResourcesException e) {
Romain Guy3696779b2013-01-28 14:04:07 -08002773 handleOutOfResourcesException(e);
Romain Guy25eba5c2012-04-04 17:29:03 -07002774 return false;
2775 } catch (IllegalArgumentException e) {
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08002776 Log.e(mTag, "Could not lock surface", e);
Romain Guy25eba5c2012-04-04 17:29:03 -07002777 // Don't assume this is due to out of memory, it could be
2778 // something else, and if it is something else then we could
2779 // kill stuff (or ourself) for no reason.
2780 mLayoutRequested = true; // ask wm for a new surface next time.
2781 return false;
2782 }
2783
2784 try {
2785 if (DEBUG_ORIENTATION || DEBUG_DRAW) {
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08002786 Log.v(mTag, "Surface " + surface + " drawing to bitmap w="
Romain Guy25eba5c2012-04-04 17:29:03 -07002787 + canvas.getWidth() + ", h=" + canvas.getHeight());
2788 //canvas.drawARGB(255, 255, 0, 0);
2789 }
2790
Romain Guy25eba5c2012-04-04 17:29:03 -07002791 // If this bitmap's format includes an alpha channel, we
2792 // need to clear it before drawing so that the child will
2793 // properly re-composite its drawing on a transparent
2794 // background. This automatically respects the clip/dirty region
2795 // or
2796 // If we are applying an offset, we need to clear the area
2797 // where the offset doesn't appear to avoid having garbage
2798 // left in the blank areas.
Alan Viveretteccb11e12014-07-08 16:04:02 -07002799 if (!canvas.isOpaque() || yoff != 0 || xoff != 0) {
Romain Guy25eba5c2012-04-04 17:29:03 -07002800 canvas.drawColor(0, PorterDuff.Mode.CLEAR);
2801 }
2802
2803 dirty.setEmpty();
2804 mIsAnimating = false;
Dianne Hackborn4702a852012-08-17 15:18:29 -07002805 mView.mPrivateFlags |= View.PFLAG_DRAWN;
Romain Guy25eba5c2012-04-04 17:29:03 -07002806
2807 if (DEBUG_DRAW) {
2808 Context cxt = mView.getContext();
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08002809 Log.i(mTag, "Drawing: package:" + cxt.getPackageName() +
Romain Guy25eba5c2012-04-04 17:29:03 -07002810 ", metrics=" + cxt.getResources().getDisplayMetrics() +
2811 ", compatibilityInfo=" + cxt.getResources().getCompatibilityInfo());
2812 }
2813 try {
Alan Viveretteccb11e12014-07-08 16:04:02 -07002814 canvas.translate(-xoff, -yoff);
Romain Guy25eba5c2012-04-04 17:29:03 -07002815 if (mTranslator != null) {
2816 mTranslator.translateCanvas(canvas);
2817 }
Dianne Hackborn908aecc2012-07-31 16:37:34 -07002818 canvas.setScreenDensity(scalingRequired ? mNoncompatDensity : 0);
Romain Guy25eba5c2012-04-04 17:29:03 -07002819 attachInfo.mSetIgnoreDirtyState = false;
2820
Romain Guy25eba5c2012-04-04 17:29:03 -07002821 mView.draw(canvas);
Alan Viverette632af842014-10-28 13:45:11 -07002822
2823 drawAccessibilityFocusedDrawableIfNeeded(canvas);
Romain Guy25eba5c2012-04-04 17:29:03 -07002824 } finally {
2825 if (!attachInfo.mSetIgnoreDirtyState) {
2826 // Only clear the flag if it was not set during the mView.draw() call
2827 attachInfo.mIgnoreDirtyState = false;
2828 }
2829 }
Romain Guy25eba5c2012-04-04 17:29:03 -07002830 } finally {
Romain Guydddcd222012-05-18 15:33:57 -07002831 try {
2832 surface.unlockCanvasAndPost(canvas);
2833 } catch (IllegalArgumentException e) {
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08002834 Log.e(mTag, "Could not unlock surface", e);
Romain Guydddcd222012-05-18 15:33:57 -07002835 mLayoutRequested = true; // ask wm for a new surface next time.
2836 //noinspection ReturnInsideFinallyBlock
2837 return false;
2838 }
Romain Guy25eba5c2012-04-04 17:29:03 -07002839
Romain Guy25eba5c2012-04-04 17:29:03 -07002840 if (LOCAL_LOGV) {
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08002841 Log.v(mTag, "Surface " + surface + " unlockCanvasAndPost");
Romain Guy25eba5c2012-04-04 17:29:03 -07002842 }
2843 }
2844 return true;
2845 }
2846
Alan Viverette632af842014-10-28 13:45:11 -07002847 /**
2848 * We want to draw a highlight around the current accessibility focused.
2849 * Since adding a style for all possible view is not a viable option we
2850 * have this specialized drawing method.
2851 *
2852 * Note: We are doing this here to be able to draw the highlight for
2853 * virtual views in addition to real ones.
2854 *
2855 * @param canvas The canvas on which to draw.
2856 */
2857 private void drawAccessibilityFocusedDrawableIfNeeded(Canvas canvas) {
2858 final Rect bounds = mAttachInfo.mTmpInvalRect;
2859 if (getAccessibilityFocusedRect(bounds)) {
2860 final Drawable drawable = getAccessibilityFocusedDrawable();
2861 if (drawable != null) {
2862 drawable.setBounds(bounds);
2863 drawable.draw(canvas);
2864 }
2865 } else if (mAttachInfo.mAccessibilityFocusDrawable != null) {
2866 mAttachInfo.mAccessibilityFocusDrawable.setBounds(0, 0, 0, 0);
2867 }
2868 }
2869
2870 private boolean getAccessibilityFocusedRect(Rect bounds) {
2871 final AccessibilityManager manager = AccessibilityManager.getInstance(mView.mContext);
2872 if (!manager.isEnabled() || !manager.isTouchExplorationEnabled()) {
2873 return false;
2874 }
2875
2876 final View host = mAccessibilityFocusedHost;
2877 if (host == null || host.mAttachInfo == null) {
2878 return false;
2879 }
2880
2881 final AccessibilityNodeProvider provider = host.getAccessibilityNodeProvider();
2882 if (provider == null) {
Svetoslavded133c2015-01-30 20:28:41 -08002883 host.getBoundsOnScreen(bounds, true);
Alan Viverette632af842014-10-28 13:45:11 -07002884 } else if (mAccessibilityFocusedVirtualView != null) {
2885 mAccessibilityFocusedVirtualView.getBoundsInScreen(bounds);
2886 } else {
2887 return false;
2888 }
2889
Alan Viverette2232add2015-05-26 15:24:18 -07002890 // Transform the rect into window-relative coordinates.
Alan Viverette632af842014-10-28 13:45:11 -07002891 final AttachInfo attachInfo = mAttachInfo;
Alan Viverette2232add2015-05-26 15:24:18 -07002892 bounds.offset(0, attachInfo.mViewRootImpl.mScrollY);
Alan Viverette632af842014-10-28 13:45:11 -07002893 bounds.offset(-attachInfo.mWindowLeft, -attachInfo.mWindowTop);
Doris Liu9607fbe2015-05-28 17:17:28 -07002894 if (!bounds.intersect(0, 0, attachInfo.mViewRootImpl.mWidth,
2895 attachInfo.mViewRootImpl.mHeight)) {
2896 // If no intersection, set bounds to empty.
2897 bounds.setEmpty();
2898 }
Alan Viverette632af842014-10-28 13:45:11 -07002899 return !bounds.isEmpty();
2900 }
2901
2902 private Drawable getAccessibilityFocusedDrawable() {
Chris Craikd36a81f2014-07-17 10:16:51 -07002903 // Lazily load the accessibility focus drawable.
2904 if (mAttachInfo.mAccessibilityFocusDrawable == null) {
Alan Viverettef6cf1a02014-08-11 14:13:02 -07002905 final TypedValue value = new TypedValue();
Chris Craikd36a81f2014-07-17 10:16:51 -07002906 final boolean resolved = mView.mContext.getTheme().resolveAttribute(
2907 R.attr.accessibilityFocusedDrawable, value, true);
2908 if (resolved) {
2909 mAttachInfo.mAccessibilityFocusDrawable =
Alan Viverette8eea3ea2014-02-03 18:40:20 -08002910 mView.mContext.getDrawable(value.resourceId);
Svetoslav Ganov42138042012-03-20 11:51:39 -07002911 }
Svetoslav Ganov42138042012-03-20 11:51:39 -07002912 }
Chris Craikd36a81f2014-07-17 10:16:51 -07002913 return mAttachInfo.mAccessibilityFocusDrawable;
Svetoslav Ganov42138042012-03-20 11:51:39 -07002914 }
2915
Jorim Jaggic39c7b02016-03-24 10:47:07 -07002916 /**
2917 * Requests that the root render node is invalidated next time we perform a draw, such that
2918 * {@link WindowCallbacks#onPostDraw} gets called.
2919 */
2920 public void requestInvalidateRootRenderNode() {
2921 mInvalidateRootRequested = true;
2922 }
2923
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08002924 boolean scrollToRectOrFocus(Rect rectangle, boolean immediate) {
Chris Craikd36a81f2014-07-17 10:16:51 -07002925 final Rect ci = mAttachInfo.mContentInsets;
2926 final Rect vi = mAttachInfo.mVisibleInsets;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08002927 int scrollY = 0;
2928 boolean handled = false;
Romain Guy8506ab42009-06-11 17:35:47 -07002929
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08002930 if (vi.left > ci.left || vi.top > ci.top
2931 || vi.right > ci.right || vi.bottom > ci.bottom) {
2932 // We'll assume that we aren't going to change the scroll
2933 // offset, since we want to avoid that unless it is actually
2934 // going to make the focus visible... otherwise we scroll
2935 // all over the place.
2936 scrollY = mScrollY;
2937 // We can be called for two different situations: during a draw,
2938 // to update the scroll position if the focus has changed (in which
2939 // case 'rectangle' is null), or in response to a
2940 // requestChildRectangleOnScreen() call (in which case 'rectangle'
2941 // is non-null and we just want to scroll to whatever that
2942 // rectangle is).
Craig Mautner26a84df2013-04-11 19:09:05 -07002943 final View focus = mView.findFocus();
Svetoslav Ganov149567f2013-01-08 15:23:34 -08002944 if (focus == null) {
Romain Guye8b16522009-07-14 13:06:42 -07002945 return false;
2946 }
Svetoslav Ganov149567f2013-01-08 15:23:34 -08002947 View lastScrolledFocus = (mLastScrolledFocus != null) ? mLastScrolledFocus.get() : null;
Craig Mautner26a84df2013-04-11 19:09:05 -07002948 if (focus != lastScrolledFocus) {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08002949 // If the focus has changed, then ignore any requests to scroll
2950 // to a rectangle; first we want to make sure the entire focus
2951 // view is visible.
2952 rectangle = null;
2953 }
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08002954 if (DEBUG_INPUT_RESIZE) Log.v(mTag, "Eval scroll: focus=" + focus
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08002955 + " rectangle=" + rectangle + " ci=" + ci
2956 + " vi=" + vi);
Svetoslav Ganov149567f2013-01-08 15:23:34 -08002957 if (focus == lastScrolledFocus && !mScrollMayChange && rectangle == null) {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08002958 // Optimization: if the focus hasn't changed since last
2959 // time, and no layout has happened, then just leave things
2960 // as they are.
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08002961 if (DEBUG_INPUT_RESIZE) Log.v(mTag, "Keeping scroll y="
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08002962 + mScrollY + " vi=" + vi.toShortString());
Craig Mautner26a84df2013-04-11 19:09:05 -07002963 } else {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08002964 // We need to determine if the currently focused view is
2965 // within the visible part of the window and, if not, apply
2966 // a pan so it can be seen.
Svetoslav Ganov149567f2013-01-08 15:23:34 -08002967 mLastScrolledFocus = new WeakReference<View>(focus);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08002968 mScrollMayChange = false;
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08002969 if (DEBUG_INPUT_RESIZE) Log.v(mTag, "Need to scroll?");
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08002970 // Try to find the rectangle from the focus view.
2971 if (focus.getGlobalVisibleRect(mVisRect, null)) {
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08002972 if (DEBUG_INPUT_RESIZE) Log.v(mTag, "Root w="
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08002973 + mView.getWidth() + " h=" + mView.getHeight()
2974 + " ci=" + ci.toShortString()
2975 + " vi=" + vi.toShortString());
2976 if (rectangle == null) {
2977 focus.getFocusedRect(mTempRect);
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08002978 if (DEBUG_INPUT_RESIZE) Log.v(mTag, "Focus " + focus
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08002979 + ": focusRect=" + mTempRect.toShortString());
Dianne Hackborn1c6a8942010-03-23 16:34:20 -07002980 if (mView instanceof ViewGroup) {
2981 ((ViewGroup) mView).offsetDescendantRectToMyCoords(
2982 focus, mTempRect);
2983 }
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08002984 if (DEBUG_INPUT_RESIZE) Log.v(mTag,
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08002985 "Focus in window: focusRect="
2986 + mTempRect.toShortString()
2987 + " visRect=" + mVisRect.toShortString());
2988 } else {
2989 mTempRect.set(rectangle);
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08002990 if (DEBUG_INPUT_RESIZE) Log.v(mTag,
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08002991 "Request scroll to rect: "
2992 + mTempRect.toShortString()
2993 + " visRect=" + mVisRect.toShortString());
2994 }
2995 if (mTempRect.intersect(mVisRect)) {
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08002996 if (DEBUG_INPUT_RESIZE) Log.v(mTag,
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08002997 "Focus window visible rect: "
2998 + mTempRect.toShortString());
2999 if (mTempRect.height() >
3000 (mView.getHeight()-vi.top-vi.bottom)) {
3001 // If the focus simply is not going to fit, then
3002 // best is probably just to leave things as-is.
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08003003 if (DEBUG_INPUT_RESIZE) Log.v(mTag,
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08003004 "Too tall; leaving scrollY=" + scrollY);
3005 } else if ((mTempRect.top-scrollY) < vi.top) {
3006 scrollY -= vi.top - (mTempRect.top-scrollY);
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08003007 if (DEBUG_INPUT_RESIZE) Log.v(mTag,
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08003008 "Top covered; scrollY=" + scrollY);
3009 } else if ((mTempRect.bottom-scrollY)
3010 > (mView.getHeight()-vi.bottom)) {
3011 scrollY += (mTempRect.bottom-scrollY)
3012 - (mView.getHeight()-vi.bottom);
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08003013 if (DEBUG_INPUT_RESIZE) Log.v(mTag,
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08003014 "Bottom covered; scrollY=" + scrollY);
3015 }
3016 handled = true;
3017 }
3018 }
3019 }
3020 }
Romain Guy8506ab42009-06-11 17:35:47 -07003021
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08003022 if (scrollY != mScrollY) {
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08003023 if (DEBUG_INPUT_RESIZE) Log.v(mTag, "Pan scroll changed: old="
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08003024 + mScrollY + " , new=" + scrollY);
Derek Sollenberger8d948352015-07-16 09:27:59 -04003025 if (!immediate) {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08003026 if (mScroller == null) {
3027 mScroller = new Scroller(mView.getContext());
3028 }
3029 mScroller.startScroll(0, mScrollY, 0, scrollY-mScrollY);
3030 } else if (mScroller != null) {
3031 mScroller.abortAnimation();
3032 }
3033 mScrollY = scrollY;
3034 }
Romain Guy8506ab42009-06-11 17:35:47 -07003035
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08003036 return handled;
3037 }
Romain Guy8506ab42009-06-11 17:35:47 -07003038
Svetoslav Ganove5dfa47d2012-05-08 15:58:32 -07003039 /**
3040 * @hide
3041 */
3042 public View getAccessibilityFocusedHost() {
3043 return mAccessibilityFocusedHost;
3044 }
3045
3046 /**
3047 * @hide
3048 */
3049 public AccessibilityNodeInfo getAccessibilityFocusedVirtualView() {
3050 return mAccessibilityFocusedVirtualView;
3051 }
3052
Svetoslav Ganov45a02e02012-06-17 15:07:29 -07003053 void setAccessibilityFocus(View view, AccessibilityNodeInfo node) {
Svetoslav Ganov791fd312012-05-14 15:12:30 -07003054 // If we have a virtual view with accessibility focus we need
3055 // to clear the focus and invalidate the virtual view bounds.
3056 if (mAccessibilityFocusedVirtualView != null) {
3057
3058 AccessibilityNodeInfo focusNode = mAccessibilityFocusedVirtualView;
3059 View focusHost = mAccessibilityFocusedHost;
Svetoslav Ganov791fd312012-05-14 15:12:30 -07003060
3061 // Wipe the state of the current accessibility focus since
3062 // the call into the provider to clear accessibility focus
3063 // will fire an accessibility event which will end up calling
3064 // this method and we want to have clean state when this
3065 // invocation happens.
3066 mAccessibilityFocusedHost = null;
3067 mAccessibilityFocusedVirtualView = null;
3068
Alan Viverette239a0c02013-05-07 17:17:35 -07003069 // Clear accessibility focus on the host after clearing state since
3070 // this method may be reentrant.
3071 focusHost.clearAccessibilityFocusNoCallbacks();
3072
Svetoslav Ganov791fd312012-05-14 15:12:30 -07003073 AccessibilityNodeProvider provider = focusHost.getAccessibilityNodeProvider();
3074 if (provider != null) {
3075 // Invalidate the area of the cleared accessibility focus.
3076 focusNode.getBoundsInParent(mTempRect);
3077 focusHost.invalidate(mTempRect);
3078 // Clear accessibility focus in the virtual node.
3079 final int virtualNodeId = AccessibilityNodeInfo.getVirtualDescendantId(
3080 focusNode.getSourceNodeId());
3081 provider.performAction(virtualNodeId,
3082 AccessibilityNodeInfo.ACTION_CLEAR_ACCESSIBILITY_FOCUS, null);
3083 }
Svetoslav Ganov45a02e02012-06-17 15:07:29 -07003084 focusNode.recycle();
Svetoslav Ganov791fd312012-05-14 15:12:30 -07003085 }
3086 if (mAccessibilityFocusedHost != null) {
3087 // Clear accessibility focus in the view.
Svetoslav Ganov42138042012-03-20 11:51:39 -07003088 mAccessibilityFocusedHost.clearAccessibilityFocusNoCallbacks();
3089 }
Svetoslav Ganov791fd312012-05-14 15:12:30 -07003090
Svetoslav Ganov45a02e02012-06-17 15:07:29 -07003091 // Set the new focus host and node.
3092 mAccessibilityFocusedHost = view;
3093 mAccessibilityFocusedVirtualView = node;
John Reck0a973302014-07-16 13:29:45 -07003094
3095 if (mAttachInfo.mHardwareRenderer != null) {
3096 mAttachInfo.mHardwareRenderer.invalidateRoot();
3097 }
Svetoslav Ganov42138042012-03-20 11:51:39 -07003098 }
3099
Jun Mukai347e5d42015-12-03 01:13:31 -08003100 void setPointerCapture(View view) {
3101 if (!mAttachInfo.mHasWindowFocus) {
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08003102 Log.w(mTag, "Can't set capture if it's not focused.");
Jun Mukai347e5d42015-12-03 01:13:31 -08003103 return;
3104 }
3105 if (mCapturingView == view) {
3106 return;
3107 }
3108 mCapturingView = view;
3109 InputManager.getInstance().setPointerIconDetached(true);
3110 }
3111
3112 void releasePointerCapture(View view) {
3113 if (mCapturingView != view || mCapturingView == null) {
3114 return;
3115 }
3116
3117 mCapturingView = null;
3118 InputManager.getInstance().setPointerIconDetached(false);
3119 }
3120
3121 boolean hasPointerCapture(View view) {
3122 return view != null && mCapturingView == view;
3123 }
3124
Igor Murashkina86ab6402013-08-30 12:58:36 -07003125 @Override
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08003126 public void requestChildFocus(View child, View focused) {
Svetoslav Ganovb36a0ac2012-02-14 17:46:47 -08003127 if (DEBUG_INPUT_RESIZE) {
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08003128 Log.v(mTag, "Request child focus: focus now " + focused);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08003129 }
Svetoslav Ganov149567f2013-01-08 15:23:34 -08003130 checkThread();
Svetoslav Ganovb36a0ac2012-02-14 17:46:47 -08003131 scheduleTraversals();
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08003132 }
3133
Igor Murashkina86ab6402013-08-30 12:58:36 -07003134 @Override
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08003135 public void clearChildFocus(View child) {
Svetoslav Ganovb36a0ac2012-02-14 17:46:47 -08003136 if (DEBUG_INPUT_RESIZE) {
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08003137 Log.v(mTag, "Clearing child focus");
Amith Yamasani73eb97f2012-02-14 15:45:15 -08003138 }
Svetoslav Ganov149567f2013-01-08 15:23:34 -08003139 checkThread();
3140 scheduleTraversals();
Svetoslav Ganovb36a0ac2012-02-14 17:46:47 -08003141 }
Amith Yamasani73eb97f2012-02-14 15:45:15 -08003142
Svetoslav Ganov42138042012-03-20 11:51:39 -07003143 @Override
3144 public ViewParent getParentForAccessibility() {
3145 return null;
3146 }
3147
Igor Murashkina86ab6402013-08-30 12:58:36 -07003148 @Override
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08003149 public void focusableViewAvailable(View v) {
3150 checkThread();
Romain Guy1c90f032011-05-24 14:59:50 -07003151 if (mView != null) {
3152 if (!mView.hasFocus()) {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08003153 v.requestFocus();
Romain Guy1c90f032011-05-24 14:59:50 -07003154 } else {
3155 // the one case where will transfer focus away from the current one
3156 // is if the current view is a view group that prefers to give focus
3157 // to its children first AND the view is a descendant of it.
Svetoslav Ganov149567f2013-01-08 15:23:34 -08003158 View focused = mView.findFocus();
3159 if (focused instanceof ViewGroup) {
3160 ViewGroup group = (ViewGroup) focused;
3161 if (group.getDescendantFocusability() == ViewGroup.FOCUS_AFTER_DESCENDANTS
3162 && isViewDescendantOf(v, focused)) {
3163 v.requestFocus();
3164 }
Romain Guy1c90f032011-05-24 14:59:50 -07003165 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08003166 }
3167 }
3168 }
3169
Igor Murashkina86ab6402013-08-30 12:58:36 -07003170 @Override
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08003171 public void recomputeViewAttributes(View child) {
3172 checkThread();
3173 if (mView == child) {
3174 mAttachInfo.mRecomputeGlobalAttributes = true;
3175 if (!mWillDrawSoon) {
3176 scheduleTraversals();
3177 }
3178 }
3179 }
3180
3181 void dispatchDetachedFromWindow() {
Romain Guy90fc03b2011-01-16 13:07:15 -08003182 if (mView != null && mView.mAttachInfo != null) {
Dianne Hackborn961cae92013-03-20 14:59:43 -07003183 mAttachInfo.mTreeObserver.dispatchOnWindowAttachedChange(false);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08003184 mView.dispatchDetachedFromWindow();
3185 }
3186
Svetoslav Ganov8643aa02011-04-20 12:12:33 -07003187 mAccessibilityInteractionConnectionManager.ensureNoConnection();
3188 mAccessibilityManager.removeAccessibilityStateChangeListener(
3189 mAccessibilityInteractionConnectionManager);
Chris Craikcce47eb2014-07-16 15:12:15 -07003190 mAccessibilityManager.removeHighTextContrastStateChangeListener(
3191 mHighContrastTextManager);
Svetoslav Ganoveeee4d22011-06-10 20:51:30 -07003192 removeSendWindowContentChangedCallback();
Svetoslav Ganov8643aa02011-04-20 12:12:33 -07003193
Romain Guya998dff2012-03-23 18:58:36 -07003194 destroyHardwareRenderer();
3195
Svetoslav Ganov45a02e02012-06-17 15:07:29 -07003196 setAccessibilityFocus(null, null);
Svetoslav Ganov791fd312012-05-14 15:12:30 -07003197
Craig Mautner8f303ad2013-06-14 11:32:22 -07003198 mView.assignParent(null);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08003199 mView = null;
3200 mAttachInfo.mRootView = null;
3201
Jun Mukai347e5d42015-12-03 01:13:31 -08003202 if (mCapturingView != null) {
3203 releasePointerCapture(mCapturingView);
3204 }
3205
Dianne Hackborn0586a1b2009-09-06 21:08:27 -07003206 mSurface.release();
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08003207
Jeff Browncc4f7db2011-08-30 20:34:48 -07003208 if (mInputQueueCallback != null && mInputQueue != null) {
3209 mInputQueueCallback.onInputQueueDestroyed(mInputQueue);
Michael Wrighta44dd262013-04-10 21:12:00 -07003210 mInputQueue.dispose();
Jeff Browncc4f7db2011-08-30 20:34:48 -07003211 mInputQueueCallback = null;
3212 mInputQueue = null;
Michael Wrighta44dd262013-04-10 21:12:00 -07003213 }
3214 if (mInputEventReceiver != null) {
Jeff Brown32cbc38552011-12-01 14:01:49 -08003215 mInputEventReceiver.dispose();
3216 mInputEventReceiver = null;
Jeff Brown46b9ac02010-04-22 18:58:52 -07003217 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08003218 try {
Jeff Brown98365d72012-08-19 20:30:52 -07003219 mWindowSession.remove(mWindow);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08003220 } catch (RemoteException e) {
3221 }
Jeff Brownf9e989d2013-04-04 23:04:03 -07003222
Jeff Brown00fa7bd2010-07-02 15:37:36 -07003223 // Dispose the input channel after removing the window so the Window Manager
3224 // doesn't interpret the input channel being closed as an abnormal termination.
3225 if (mInputChannel != null) {
3226 mInputChannel.dispose();
3227 mInputChannel = null;
Jeff Brown349703e2010-06-22 01:27:15 -07003228 }
Jeff Brown96e942d2011-11-30 19:55:01 -08003229
Jeff Brownd912e1f2014-04-11 18:46:22 -07003230 mDisplayManager.unregisterDisplayListener(mDisplayListener);
3231
Jeff Brownebb2d8d2012-03-23 17:14:34 -07003232 unscheduleTraversals();
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08003233 }
Romain Guy8506ab42009-06-11 17:35:47 -07003234
Dianne Hackborn694f79b2010-03-17 19:44:59 -07003235 void updateConfiguration(Configuration config, boolean force) {
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08003236 if (DEBUG_CONFIGURATION) Log.v(mTag,
Dianne Hackborn694f79b2010-03-17 19:44:59 -07003237 "Applying new config to window "
3238 + mWindowAttributes.getTitle()
3239 + ": " + config);
Dianne Hackborn5fd21692011-06-07 14:09:47 -07003240
Craig Mautner48d0d182013-06-11 07:53:06 -07003241 CompatibilityInfo ci = mDisplayAdjustments.getCompatibilityInfo();
3242 if (!ci.equals(CompatibilityInfo.DEFAULT_COMPATIBILITY_INFO)) {
Dianne Hackborn5fd21692011-06-07 14:09:47 -07003243 config = new Configuration(config);
Dianne Hackborn908aecc2012-07-31 16:37:34 -07003244 ci.applyToConfiguration(mNoncompatDensity, config);
Dianne Hackborn5fd21692011-06-07 14:09:47 -07003245 }
3246
Dianne Hackborn694f79b2010-03-17 19:44:59 -07003247 synchronized (sConfigCallbacks) {
3248 for (int i=sConfigCallbacks.size()-1; i>=0; i--) {
3249 sConfigCallbacks.get(i).onConfigurationChanged(config);
3250 }
3251 }
3252 if (mView != null) {
3253 // At this point the resources have been updated to
3254 // have the most recent config, whatever that is. Use
Dianne Hackborn908aecc2012-07-31 16:37:34 -07003255 // the one in them which may be newer.
Romain Guy1c90f032011-05-24 14:59:50 -07003256 config = mView.getResources().getConfiguration();
Dianne Hackborn694f79b2010-03-17 19:44:59 -07003257 if (force || mLastConfiguration.diff(config) != 0) {
Fabrice Di Megliocf128972012-10-16 20:51:12 -07003258 final int lastLayoutDirection = mLastConfiguration.getLayoutDirection();
3259 final int currentLayoutDirection = config.getLayoutDirection();
Dianne Hackborn694f79b2010-03-17 19:44:59 -07003260 mLastConfiguration.setTo(config);
Fabrice Di Megliob003e282012-10-17 17:20:19 -07003261 if (lastLayoutDirection != currentLayoutDirection &&
3262 mViewLayoutDirectionInitial == View.LAYOUT_DIRECTION_INHERIT) {
Fabrice Di Megliocf128972012-10-16 20:51:12 -07003263 mView.setLayoutDirection(currentLayoutDirection);
3264 }
Dianne Hackborn694f79b2010-03-17 19:44:59 -07003265 mView.dispatchConfigurationChanged(config);
3266 }
3267 }
3268 }
John Reck05e85842014-04-23 14:48:28 -07003269
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08003270 /**
3271 * Return true if child is an ancestor of parent, (or equal to the parent).
3272 */
Svetoslav Ganove5dfa47d2012-05-08 15:58:32 -07003273 public static boolean isViewDescendantOf(View child, View parent) {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08003274 if (child == parent) {
3275 return true;
3276 }
3277
3278 final ViewParent theParent = child.getParent();
3279 return (theParent instanceof ViewGroup) && isViewDescendantOf((View) theParent, parent);
3280 }
3281
Yohei Yukawad2e56472015-07-28 17:00:33 -07003282 private static void forceLayout(View view) {
3283 view.forceLayout();
3284 if (view instanceof ViewGroup) {
3285 ViewGroup group = (ViewGroup) view;
3286 final int count = group.getChildCount();
3287 for (int i = 0; i < count; i++) {
3288 forceLayout(group.getChildAt(i));
3289 }
3290 }
3291 }
3292
Jeff Browna175a5b2012-02-15 19:18:31 -08003293 private final static int MSG_INVALIDATE = 1;
3294 private final static int MSG_INVALIDATE_RECT = 2;
3295 private final static int MSG_DIE = 3;
3296 private final static int MSG_RESIZED = 4;
3297 private final static int MSG_RESIZED_REPORT = 5;
3298 private final static int MSG_WINDOW_FOCUS_CHANGED = 6;
keunyoung30f420f2013-08-02 14:23:10 -07003299 private final static int MSG_DISPATCH_INPUT_EVENT = 7;
Jeff Browna175a5b2012-02-15 19:18:31 -08003300 private final static int MSG_DISPATCH_APP_VISIBILITY = 8;
3301 private final static int MSG_DISPATCH_GET_NEW_SURFACE = 9;
Jeff Browna175a5b2012-02-15 19:18:31 -08003302 private final static int MSG_DISPATCH_KEY_FROM_IME = 11;
Yohei Yukawa1fa5f592016-03-24 10:24:23 -07003303 private final static int MSG_FINISH_INPUT_CONNECTION = 12;
Jeff Browna175a5b2012-02-15 19:18:31 -08003304 private final static int MSG_CHECK_FOCUS = 13;
3305 private final static int MSG_CLOSE_SYSTEM_DIALOGS = 14;
3306 private final static int MSG_DISPATCH_DRAG_EVENT = 15;
3307 private final static int MSG_DISPATCH_DRAG_LOCATION_EVENT = 16;
3308 private final static int MSG_DISPATCH_SYSTEM_UI_VISIBILITY = 17;
3309 private final static int MSG_UPDATE_CONFIGURATION = 18;
Svetoslav Ganov42138042012-03-20 11:51:39 -07003310 private final static int MSG_PROCESS_INPUT_EVENTS = 19;
Romain Guyfbb93fa2012-12-03 18:50:35 -08003311 private final static int MSG_CLEAR_ACCESSIBILITY_FOCUS_HOST = 21;
Jorim Jaggi827e0fa2015-05-07 11:41:41 -07003312 private final static int MSG_INVALIDATE_WORLD = 22;
3313 private final static int MSG_WINDOW_MOVED = 23;
3314 private final static int MSG_SYNTHESIZE_INPUT_EVENT = 24;
3315 private final static int MSG_DISPATCH_WINDOW_SHOWN = 25;
Clara Bayarri75e09792015-07-29 16:20:40 +01003316 private final static int MSG_REQUEST_KEYBOARD_SHORTCUTS = 26;
Vladislav Kaznacheevec6a4472016-01-22 12:21:52 -08003317 private final static int MSG_UPDATE_POINTER_ICON = 27;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08003318
Jeff Browna175a5b2012-02-15 19:18:31 -08003319 final class ViewRootHandler extends Handler {
3320 @Override
3321 public String getMessageName(Message message) {
3322 switch (message.what) {
3323 case MSG_INVALIDATE:
3324 return "MSG_INVALIDATE";
3325 case MSG_INVALIDATE_RECT:
3326 return "MSG_INVALIDATE_RECT";
3327 case MSG_DIE:
3328 return "MSG_DIE";
3329 case MSG_RESIZED:
3330 return "MSG_RESIZED";
3331 case MSG_RESIZED_REPORT:
3332 return "MSG_RESIZED_REPORT";
3333 case MSG_WINDOW_FOCUS_CHANGED:
3334 return "MSG_WINDOW_FOCUS_CHANGED";
keunyoung30f420f2013-08-02 14:23:10 -07003335 case MSG_DISPATCH_INPUT_EVENT:
3336 return "MSG_DISPATCH_INPUT_EVENT";
Jeff Browna175a5b2012-02-15 19:18:31 -08003337 case MSG_DISPATCH_APP_VISIBILITY:
3338 return "MSG_DISPATCH_APP_VISIBILITY";
3339 case MSG_DISPATCH_GET_NEW_SURFACE:
3340 return "MSG_DISPATCH_GET_NEW_SURFACE";
Jeff Browna175a5b2012-02-15 19:18:31 -08003341 case MSG_DISPATCH_KEY_FROM_IME:
3342 return "MSG_DISPATCH_KEY_FROM_IME";
Yohei Yukawa1fa5f592016-03-24 10:24:23 -07003343 case MSG_FINISH_INPUT_CONNECTION:
3344 return "MSG_FINISH_INPUT_CONNECTION";
Jeff Browna175a5b2012-02-15 19:18:31 -08003345 case MSG_CHECK_FOCUS:
3346 return "MSG_CHECK_FOCUS";
3347 case MSG_CLOSE_SYSTEM_DIALOGS:
3348 return "MSG_CLOSE_SYSTEM_DIALOGS";
3349 case MSG_DISPATCH_DRAG_EVENT:
3350 return "MSG_DISPATCH_DRAG_EVENT";
3351 case MSG_DISPATCH_DRAG_LOCATION_EVENT:
3352 return "MSG_DISPATCH_DRAG_LOCATION_EVENT";
3353 case MSG_DISPATCH_SYSTEM_UI_VISIBILITY:
3354 return "MSG_DISPATCH_SYSTEM_UI_VISIBILITY";
3355 case MSG_UPDATE_CONFIGURATION:
3356 return "MSG_UPDATE_CONFIGURATION";
Jeff Browna175a5b2012-02-15 19:18:31 -08003357 case MSG_PROCESS_INPUT_EVENTS:
3358 return "MSG_PROCESS_INPUT_EVENTS";
Svetoslav Ganov005b83b2012-04-16 18:17:17 -07003359 case MSG_CLEAR_ACCESSIBILITY_FOCUS_HOST:
3360 return "MSG_CLEAR_ACCESSIBILITY_FOCUS_HOST";
Craig Mautner5702d4d2012-06-30 14:10:16 -07003361 case MSG_WINDOW_MOVED:
3362 return "MSG_WINDOW_MOVED";
Michael Wright899d7052014-04-23 17:23:39 -07003363 case MSG_SYNTHESIZE_INPUT_EVENT:
3364 return "MSG_SYNTHESIZE_INPUT_EVENT";
Craig Mautner9c795042014-10-28 19:59:59 -07003365 case MSG_DISPATCH_WINDOW_SHOWN:
3366 return "MSG_DISPATCH_WINDOW_SHOWN";
Vladislav Kaznacheevec6a4472016-01-22 12:21:52 -08003367 case MSG_UPDATE_POINTER_ICON:
3368 return "MSG_UPDATE_POINTER_ICON";
Jeff Browna175a5b2012-02-15 19:18:31 -08003369 }
3370 return super.getMessageName(message);
Romain Guyf9284692011-07-13 18:46:21 -07003371 }
Romain Guyf9284692011-07-13 18:46:21 -07003372
Jeff Browna175a5b2012-02-15 19:18:31 -08003373 @Override
3374 public void handleMessage(Message msg) {
3375 switch (msg.what) {
3376 case MSG_INVALIDATE:
3377 ((View) msg.obj).invalidate();
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08003378 break;
Jeff Browna175a5b2012-02-15 19:18:31 -08003379 case MSG_INVALIDATE_RECT:
3380 final View.AttachInfo.InvalidateInfo info = (View.AttachInfo.InvalidateInfo) msg.obj;
3381 info.target.invalidate(info.left, info.top, info.right, info.bottom);
Svetoslav Ganovabae2a12012-11-27 16:59:37 -08003382 info.recycle();
Jeff Browna175a5b2012-02-15 19:18:31 -08003383 break;
Jeff Browna175a5b2012-02-15 19:18:31 -08003384 case MSG_PROCESS_INPUT_EVENTS:
3385 mProcessInputEventsScheduled = false;
3386 doProcessInputEvents();
3387 break;
3388 case MSG_DISPATCH_APP_VISIBILITY:
3389 handleAppVisibility(msg.arg1 != 0);
3390 break;
3391 case MSG_DISPATCH_GET_NEW_SURFACE:
3392 handleGetNewSurface();
3393 break;
Svetoslav Ganov758143e2012-08-06 16:40:27 -07003394 case MSG_RESIZED: {
3395 // Recycled in the fall through...
3396 SomeArgs args = (SomeArgs) msg.obj;
Romain Guydfab3632012-10-03 14:53:25 -07003397 if (mWinFrame.equals(args.arg1)
Dianne Hackbornc4aad012013-02-22 15:05:25 -08003398 && mPendingOverscanInsets.equals(args.arg5)
Romain Guydfab3632012-10-03 14:53:25 -07003399 && mPendingContentInsets.equals(args.arg2)
Adrian Roosfa104232014-06-20 16:10:14 -07003400 && mPendingStableInsets.equals(args.arg6)
Romain Guydfab3632012-10-03 14:53:25 -07003401 && mPendingVisibleInsets.equals(args.arg3)
Filip Gruszczynski2217f612015-05-26 11:32:08 -07003402 && mPendingOutsets.equals(args.arg7)
Jorim Jaggi0fe356e2016-01-05 14:43:25 +01003403 && mPendingBackDropFrame.equals(args.arg8)
Jorim Jaggia4a58ef2016-01-27 02:10:08 -08003404 && args.arg4 == null
3405 && args.argi1 == 0) {
Jeff Browna175a5b2012-02-15 19:18:31 -08003406 break;
Romain Guycdb86672010-03-18 18:54:50 -07003407 }
Svetoslav Ganov758143e2012-08-06 16:40:27 -07003408 } // fall through...
Jeff Browna175a5b2012-02-15 19:18:31 -08003409 case MSG_RESIZED_REPORT:
3410 if (mAdded) {
Svetoslav Ganov758143e2012-08-06 16:40:27 -07003411 SomeArgs args = (SomeArgs) msg.obj;
3412
3413 Configuration config = (Configuration) args.arg4;
Jeff Browna175a5b2012-02-15 19:18:31 -08003414 if (config != null) {
3415 updateConfiguration(config, false);
3416 }
Svetoslav Ganov758143e2012-08-06 16:40:27 -07003417
3418 mWinFrame.set((Rect) args.arg1);
Dianne Hackbornc4aad012013-02-22 15:05:25 -08003419 mPendingOverscanInsets.set((Rect) args.arg5);
Svetoslav Ganov758143e2012-08-06 16:40:27 -07003420 mPendingContentInsets.set((Rect) args.arg2);
Adrian Roosfa104232014-06-20 16:10:14 -07003421 mPendingStableInsets.set((Rect) args.arg6);
Svetoslav Ganov758143e2012-08-06 16:40:27 -07003422 mPendingVisibleInsets.set((Rect) args.arg3);
Filip Gruszczynski2217f612015-05-26 11:32:08 -07003423 mPendingOutsets.set((Rect) args.arg7);
Jorim Jaggia7262a82015-11-03 15:15:40 +01003424 mPendingBackDropFrame.set((Rect) args.arg8);
Jorim Jaggia4a58ef2016-01-27 02:10:08 -08003425 mForceNextWindowRelayout = args.argi1 != 0;
Jorim Jaggi0ffd49c2016-02-12 15:04:21 -08003426 mPendingAlwaysConsumeNavBar = args.argi2 != 0;
Svetoslav Ganov758143e2012-08-06 16:40:27 -07003427
3428 args.recycle();
3429
Jeff Browna175a5b2012-02-15 19:18:31 -08003430 if (msg.what == MSG_RESIZED_REPORT) {
3431 mReportNextDraw = true;
3432 }
Romain Guy59a12ca2011-06-09 17:48:21 -07003433
Yohei Yukawad2e56472015-07-28 17:00:33 -07003434 if (mView != null) {
3435 forceLayout(mView);
3436 }
3437
Jeff Browna175a5b2012-02-15 19:18:31 -08003438 requestLayout();
3439 }
3440 break;
Craig Mautner5702d4d2012-06-30 14:10:16 -07003441 case MSG_WINDOW_MOVED:
3442 if (mAdded) {
3443 final int w = mWinFrame.width();
3444 final int h = mWinFrame.height();
3445 final int l = msg.arg1;
3446 final int t = msg.arg2;
3447 mWinFrame.left = l;
3448 mWinFrame.right = l + w;
3449 mWinFrame.top = t;
3450 mWinFrame.bottom = t + h;
3451
Jorim Jaggia7262a82015-11-03 15:15:40 +01003452 mPendingBackDropFrame.set(mWinFrame);
3453
Jorim Jaggi844e1712016-01-13 17:39:25 -08003454 // Suppress layouts during resizing - a correct layout will happen when resizing
3455 // is done, and this just increases system load.
Jorim Jaggi2e95a482016-01-14 17:36:55 -08003456 boolean isDockedDivider = mWindowAttributes.type == TYPE_DOCK_DIVIDER;
3457 boolean suppress = (mDragResizing && mResizeMode == RESIZE_MODE_DOCKED_DIVIDER)
3458 || isDockedDivider;
Jorim Jaggi844e1712016-01-13 17:39:25 -08003459 if (!suppress) {
3460 if (mView != null) {
3461 forceLayout(mView);
3462 }
3463 requestLayout();
Jorim Jaggi2e95a482016-01-14 17:36:55 -08003464 } else {
3465 maybeHandleWindowMove(mWinFrame);
Yohei Yukawad2e56472015-07-28 17:00:33 -07003466 }
Craig Mautner5702d4d2012-06-30 14:10:16 -07003467 }
3468 break;
Jeff Browna175a5b2012-02-15 19:18:31 -08003469 case MSG_WINDOW_FOCUS_CHANGED: {
3470 if (mAdded) {
3471 boolean hasWindowFocus = msg.arg1 != 0;
3472 mAttachInfo.mHasWindowFocus = hasWindowFocus;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08003473
Jeff Browna175a5b2012-02-15 19:18:31 -08003474 profileRendering(hasWindowFocus);
3475
3476 if (hasWindowFocus) {
3477 boolean inTouchMode = msg.arg2 != 0;
3478 ensureTouchModeLocally(inTouchMode);
3479
Romain Guye55945e2013-04-04 15:26:04 -07003480 if (mAttachInfo.mHardwareRenderer != null && mSurface.isValid()){
Jeff Browna175a5b2012-02-15 19:18:31 -08003481 mFullRedrawNeeded = true;
Dianne Hackborn64825172011-03-02 21:32:58 -08003482 try {
Alan Viveretteccb11e12014-07-08 16:04:02 -07003483 final WindowManager.LayoutParams lp = mWindowAttributes;
Alan Viverette49a22e82014-07-12 20:01:27 -07003484 final Rect surfaceInsets = lp != null ? lp.surfaceInsets : null;
Romain Guy3696779b2013-01-28 14:04:07 -08003485 mAttachInfo.mHardwareRenderer.initializeIfNeeded(
Alan Viverette50210d92015-05-14 18:05:36 -07003486 mWidth, mHeight, mAttachInfo, mSurface, surfaceInsets);
Igor Murashkina86ab6402013-08-30 12:58:36 -07003487 } catch (OutOfResourcesException e) {
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08003488 Log.e(mTag, "OutOfResourcesException locking surface", e);
Jeff Browna175a5b2012-02-15 19:18:31 -08003489 try {
Jeff Brown98365d72012-08-19 20:30:52 -07003490 if (!mWindowSession.outOfMemory(mWindow)) {
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08003491 Slog.w(mTag, "No processes killed for memory; killing self");
Jeff Browna175a5b2012-02-15 19:18:31 -08003492 Process.killProcess(Process.myPid());
3493 }
3494 } catch (RemoteException ex) {
Dianne Hackborn64825172011-03-02 21:32:58 -08003495 }
Jeff Browna175a5b2012-02-15 19:18:31 -08003496 // Retry in a bit.
3497 sendMessageDelayed(obtainMessage(msg.what, msg.arg1, msg.arg2), 500);
3498 return;
Dianne Hackborn64825172011-03-02 21:32:58 -08003499 }
Dianne Hackborn64825172011-03-02 21:32:58 -08003500 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08003501 }
Romain Guy8506ab42009-06-11 17:35:47 -07003502
Jeff Browna175a5b2012-02-15 19:18:31 -08003503 mLastWasImTarget = WindowManager.LayoutParams
3504 .mayUseInputMethod(mWindowAttributes.flags);
Romain Guy8506ab42009-06-11 17:35:47 -07003505
Jeff Browna175a5b2012-02-15 19:18:31 -08003506 InputMethodManager imm = InputMethodManager.peekInstance();
Yohei Yukawa5f059652015-05-14 22:16:41 -07003507 if (imm != null && mLastWasImTarget && !isInLocalFocusMode()) {
3508 imm.onPreWindowFocus(mView, hasWindowFocus);
3509 }
Jeff Browna175a5b2012-02-15 19:18:31 -08003510 if (mView != null) {
Jeff Browna175a5b2012-02-15 19:18:31 -08003511 mAttachInfo.mKeyDispatchState.reset();
3512 mView.dispatchWindowFocusChanged(hasWindowFocus);
Dianne Hackborn961cae92013-03-20 14:59:43 -07003513 mAttachInfo.mTreeObserver.dispatchOnWindowFocusChange(hasWindowFocus);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08003514 }
svetoslavganov75986cf2009-05-14 22:28:01 -07003515
Jeff Browna175a5b2012-02-15 19:18:31 -08003516 // Note: must be done after the focus change callbacks,
3517 // so all of the view state is set up correctly.
3518 if (hasWindowFocus) {
keunyoung30f420f2013-08-02 14:23:10 -07003519 if (imm != null && mLastWasImTarget && !isInLocalFocusMode()) {
Yohei Yukawa5f059652015-05-14 22:16:41 -07003520 imm.onPostWindowFocus(mView, mView.findFocus(),
Jeff Browna175a5b2012-02-15 19:18:31 -08003521 mWindowAttributes.softInputMode,
3522 !mHasHadWindowFocus, mWindowAttributes.flags);
3523 }
3524 // Clear the forward bit. We can just do this directly, since
3525 // the window manager doesn't care about it.
3526 mWindowAttributes.softInputMode &=
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08003527 ~WindowManager.LayoutParams.SOFT_INPUT_IS_FORWARD_NAVIGATION;
Jeff Browna175a5b2012-02-15 19:18:31 -08003528 ((WindowManager.LayoutParams)mView.getLayoutParams())
3529 .softInputMode &=
3530 ~WindowManager.LayoutParams.SOFT_INPUT_IS_FORWARD_NAVIGATION;
3531 mHasHadWindowFocus = true;
Jun Mukai347e5d42015-12-03 01:13:31 -08003532 } else if (mCapturingView != null) {
3533 releasePointerCapture(mCapturingView);
Jeff Browna175a5b2012-02-15 19:18:31 -08003534 }
svetoslavganov75986cf2009-05-14 22:28:01 -07003535 }
Jeff Browna175a5b2012-02-15 19:18:31 -08003536 } break;
3537 case MSG_DIE:
3538 doDie();
3539 break;
keunyoung30f420f2013-08-02 14:23:10 -07003540 case MSG_DISPATCH_INPUT_EVENT: {
Jae Seo6a6059a2014-04-17 21:35:29 -07003541 SomeArgs args = (SomeArgs)msg.obj;
3542 InputEvent event = (InputEvent)args.arg1;
3543 InputEventReceiver receiver = (InputEventReceiver)args.arg2;
3544 enqueueInputEvent(event, receiver, 0, true);
3545 args.recycle();
Jeff Browna175a5b2012-02-15 19:18:31 -08003546 } break;
Michael Wright899d7052014-04-23 17:23:39 -07003547 case MSG_SYNTHESIZE_INPUT_EVENT: {
3548 InputEvent event = (InputEvent)msg.obj;
3549 enqueueInputEvent(event, null, QueuedInputEvent.FLAG_UNHANDLED, true);
3550 } break;
Jeff Browna175a5b2012-02-15 19:18:31 -08003551 case MSG_DISPATCH_KEY_FROM_IME: {
3552 if (LOCAL_LOGV) Log.v(
3553 TAG, "Dispatching key "
3554 + msg.obj + " from IME to " + mView);
3555 KeyEvent event = (KeyEvent)msg.obj;
3556 if ((event.getFlags()&KeyEvent.FLAG_FROM_SYSTEM) != 0) {
3557 // The IME is trying to say this event is from the
3558 // system! Bad bad bad!
3559 //noinspection UnusedAssignment
Michael Wright899d7052014-04-23 17:23:39 -07003560 event = KeyEvent.changeFlags(event, event.getFlags() &
3561 ~KeyEvent.FLAG_FROM_SYSTEM);
Jeff Browna175a5b2012-02-15 19:18:31 -08003562 }
3563 enqueueInputEvent(event, null, QueuedInputEvent.FLAG_DELIVER_POST_IME, true);
3564 } break;
Yohei Yukawa1fa5f592016-03-24 10:24:23 -07003565 case MSG_FINISH_INPUT_CONNECTION: {
3566 InputMethodManager imm = InputMethodManager.peekInstance();
3567 if (imm != null) {
3568 imm.reportFinishInputConnection((InputConnection)msg.obj);
3569 }
3570 } break;
Jeff Browna175a5b2012-02-15 19:18:31 -08003571 case MSG_CHECK_FOCUS: {
3572 InputMethodManager imm = InputMethodManager.peekInstance();
3573 if (imm != null) {
3574 imm.checkFocus();
3575 }
3576 } break;
3577 case MSG_CLOSE_SYSTEM_DIALOGS: {
3578 if (mView != null) {
3579 mView.onCloseSystemDialogs((String)msg.obj);
3580 }
3581 } break;
3582 case MSG_DISPATCH_DRAG_EVENT:
3583 case MSG_DISPATCH_DRAG_LOCATION_EVENT: {
3584 DragEvent event = (DragEvent)msg.obj;
3585 event.mLocalState = mLocalDragState; // only present when this app called startDrag()
3586 handleDragEvent(event);
3587 } break;
3588 case MSG_DISPATCH_SYSTEM_UI_VISIBILITY: {
Romain Guyfbb93fa2012-12-03 18:50:35 -08003589 handleDispatchSystemUiVisibilityChanged((SystemUiVisibilityInfo) msg.obj);
Jeff Browna175a5b2012-02-15 19:18:31 -08003590 } break;
3591 case MSG_UPDATE_CONFIGURATION: {
3592 Configuration config = (Configuration)msg.obj;
3593 if (config.isOtherSeqNewer(mLastConfiguration)) {
3594 config = mLastConfiguration;
3595 }
3596 updateConfiguration(config, false);
3597 } break;
Svetoslav Ganov005b83b2012-04-16 18:17:17 -07003598 case MSG_CLEAR_ACCESSIBILITY_FOCUS_HOST: {
Svetoslav Ganov45a02e02012-06-17 15:07:29 -07003599 setAccessibilityFocus(null, null);
Svetoslav Ganov005b83b2012-04-16 18:17:17 -07003600 } break;
Dianne Hackborna53de062012-05-08 18:53:51 -07003601 case MSG_INVALIDATE_WORLD: {
Romain Guyf84208f2012-09-13 22:50:18 -07003602 if (mView != null) {
3603 invalidateWorld(mView);
3604 }
Dianne Hackborna53de062012-05-08 18:53:51 -07003605 } break;
Craig Mautner9c795042014-10-28 19:59:59 -07003606 case MSG_DISPATCH_WINDOW_SHOWN: {
3607 handleDispatchWindowShown();
Clara Bayarri75e09792015-07-29 16:20:40 +01003608 } break;
3609 case MSG_REQUEST_KEYBOARD_SHORTCUTS: {
Clara Bayarrifcd7e802016-03-10 12:58:18 +00003610 final IResultReceiver receiver = (IResultReceiver) msg.obj;
3611 final int deviceId = msg.arg1;
3612 handleRequestKeyboardShortcuts(receiver, deviceId);
Clara Bayarri75e09792015-07-29 16:20:40 +01003613 } break;
Vladislav Kaznacheevec6a4472016-01-22 12:21:52 -08003614 case MSG_UPDATE_POINTER_ICON: {
3615 MotionEvent event = (MotionEvent) msg.obj;
3616 resetPointerIcon(event);
3617 } break;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08003618 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08003619 }
3620 }
Romain Guy51e4d4d2012-03-15 18:30:47 -07003621
Jeff Browna175a5b2012-02-15 19:18:31 -08003622 final ViewRootHandler mHandler = new ViewRootHandler();
Svetoslav Ganov8643aa02011-04-20 12:12:33 -07003623
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08003624 /**
3625 * Something in the current window tells us we need to change the touch mode. For
3626 * example, we are not in touch mode, and the user touches the screen.
3627 *
3628 * If the touch mode has changed, tell the window manager, and handle it locally.
3629 *
3630 * @param inTouchMode Whether we want to be in touch mode.
3631 * @return True if the touch mode changed and focus changed was changed as a result
3632 */
3633 boolean ensureTouchMode(boolean inTouchMode) {
3634 if (DBG) Log.d("touchmode", "ensureTouchMode(" + inTouchMode + "), current "
3635 + "touch mode is " + mAttachInfo.mInTouchMode);
3636 if (mAttachInfo.mInTouchMode == inTouchMode) return false;
3637
3638 // tell the window manager
3639 try {
Matt Wud6bc96d2016-01-14 12:59:24 -08003640 mWindowSession.setInTouchMode(inTouchMode);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08003641 } catch (RemoteException e) {
3642 throw new RuntimeException(e);
3643 }
3644
3645 // handle the change
Romain Guy2d4cff62010-04-09 15:39:00 -07003646 return ensureTouchModeLocally(inTouchMode);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08003647 }
3648
3649 /**
3650 * Ensure that the touch mode for this window is set, and if it is changing,
3651 * take the appropriate action.
3652 * @param inTouchMode Whether we want to be in touch mode.
3653 * @return True if the touch mode changed and focus changed was changed as a result
3654 */
Romain Guy2d4cff62010-04-09 15:39:00 -07003655 private boolean ensureTouchModeLocally(boolean inTouchMode) {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08003656 if (DBG) Log.d("touchmode", "ensureTouchModeLocally(" + inTouchMode + "), current "
3657 + "touch mode is " + mAttachInfo.mInTouchMode);
3658
3659 if (mAttachInfo.mInTouchMode == inTouchMode) return false;
3660
3661 mAttachInfo.mInTouchMode = inTouchMode;
3662 mAttachInfo.mTreeObserver.dispatchOnTouchModeChanged(inTouchMode);
3663
Romain Guy2d4cff62010-04-09 15:39:00 -07003664 return (inTouchMode) ? enterTouchMode() : leaveTouchMode();
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08003665 }
3666
3667 private boolean enterTouchMode() {
Alan Viveretteed7821a2013-07-24 15:49:23 -07003668 if (mView != null && mView.hasFocus()) {
3669 // note: not relying on mFocusedView here because this could
3670 // be when the window is first being added, and mFocused isn't
3671 // set yet.
3672 final View focused = mView.findFocus();
3673 if (focused != null && !focused.isFocusableInTouchMode()) {
3674 final ViewGroup ancestorToTakeFocus = findAncestorToTakeFocusInTouchMode(focused);
3675 if (ancestorToTakeFocus != null) {
3676 // there is an ancestor that wants focus after its
3677 // descendants that is focusable in touch mode.. give it
3678 // focus
3679 return ancestorToTakeFocus.requestFocus();
3680 } else {
Alan Viverette973f3b42013-08-13 16:57:28 -07003681 // There's nothing to focus. Clear and propagate through the
3682 // hierarchy, but don't attempt to place new focus.
Alan Viverette223622a2013-12-17 13:29:02 -08003683 focused.clearFocusInternal(null, true, false);
Alan Viveretteed7821a2013-07-24 15:49:23 -07003684 return true;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08003685 }
3686 }
3687 }
3688 return false;
3689 }
3690
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08003691 /**
3692 * Find an ancestor of focused that wants focus after its descendants and is
3693 * focusable in touch mode.
3694 * @param focused The currently focused view.
3695 * @return An appropriate view, or null if no such view exists.
3696 */
Romain Guya998dff2012-03-23 18:58:36 -07003697 private static ViewGroup findAncestorToTakeFocusInTouchMode(View focused) {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08003698 ViewParent parent = focused.getParent();
3699 while (parent instanceof ViewGroup) {
3700 final ViewGroup vgParent = (ViewGroup) parent;
3701 if (vgParent.getDescendantFocusability() == ViewGroup.FOCUS_AFTER_DESCENDANTS
3702 && vgParent.isFocusableInTouchMode()) {
3703 return vgParent;
3704 }
3705 if (vgParent.isRootNamespace()) {
3706 return null;
3707 } else {
3708 parent = vgParent.getParent();
3709 }
3710 }
3711 return null;
3712 }
3713
3714 private boolean leaveTouchMode() {
3715 if (mView != null) {
3716 if (mView.hasFocus()) {
Svetoslav Ganov149567f2013-01-08 15:23:34 -08003717 View focusedView = mView.findFocus();
3718 if (!(focusedView instanceof ViewGroup)) {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08003719 // some view has focus, let it keep it
Svetoslav Ganovcf8a3b82012-04-30 16:49:59 -07003720 return false;
Svetoslav Ganov149567f2013-01-08 15:23:34 -08003721 } else if (((ViewGroup) focusedView).getDescendantFocusability() !=
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08003722 ViewGroup.FOCUS_AFTER_DESCENDANTS) {
3723 // some view group has focus, and doesn't prefer its children
3724 // over itself for focus, so let them keep it.
Svetoslav Ganovcf8a3b82012-04-30 16:49:59 -07003725 return false;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08003726 }
3727 }
Svetoslav Ganovcf8a3b82012-04-30 16:49:59 -07003728
3729 // find the best view to give focus to in this brave new non-touch-mode
3730 // world
3731 final View focused = focusSearch(null, View.FOCUS_DOWN);
3732 if (focused != null) {
3733 return focused.requestFocus(View.FOCUS_DOWN);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08003734 }
3735 }
3736 return false;
3737 }
3738
Jeff Brownf9e989d2013-04-04 23:04:03 -07003739 /**
3740 * Base class for implementing a stage in the chain of responsibility
3741 * for processing input events.
3742 * <p>
3743 * Events are delivered to the stage by the {@link #deliver} method. The stage
3744 * then has the choice of finishing the event or forwarding it to the next stage.
3745 * </p>
3746 */
3747 abstract class InputStage {
3748 private final InputStage mNext;
3749
3750 protected static final int FORWARD = 0;
3751 protected static final int FINISH_HANDLED = 1;
3752 protected static final int FINISH_NOT_HANDLED = 2;
3753
3754 /**
3755 * Creates an input stage.
3756 * @param next The next stage to which events should be forwarded.
3757 */
3758 public InputStage(InputStage next) {
3759 mNext = next;
3760 }
3761
3762 /**
3763 * Delivers an event to be processed.
3764 */
3765 public final void deliver(QueuedInputEvent q) {
3766 if ((q.mFlags & QueuedInputEvent.FLAG_FINISHED) != 0) {
3767 forward(q);
Michael Wright17d28ca2013-10-31 17:47:45 -07003768 } else if (shouldDropInputEvent(q)) {
Jeff Brownf9e989d2013-04-04 23:04:03 -07003769 finish(q, false);
3770 } else {
3771 apply(q, onProcess(q));
3772 }
3773 }
3774
3775 /**
3776 * Marks the the input event as finished then forwards it to the next stage.
3777 */
3778 protected void finish(QueuedInputEvent q, boolean handled) {
3779 q.mFlags |= QueuedInputEvent.FLAG_FINISHED;
3780 if (handled) {
3781 q.mFlags |= QueuedInputEvent.FLAG_FINISHED_HANDLED;
3782 }
3783 forward(q);
3784 }
3785
3786 /**
3787 * Forwards the event to the next stage.
3788 */
3789 protected void forward(QueuedInputEvent q) {
3790 onDeliverToNext(q);
3791 }
3792
3793 /**
3794 * Applies a result code from {@link #onProcess} to the specified event.
3795 */
3796 protected void apply(QueuedInputEvent q, int result) {
3797 if (result == FORWARD) {
3798 forward(q);
3799 } else if (result == FINISH_HANDLED) {
3800 finish(q, true);
3801 } else if (result == FINISH_NOT_HANDLED) {
3802 finish(q, false);
3803 } else {
3804 throw new IllegalArgumentException("Invalid result: " + result);
3805 }
3806 }
3807
3808 /**
3809 * Called when an event is ready to be processed.
3810 * @return A result code indicating how the event was handled.
3811 */
3812 protected int onProcess(QueuedInputEvent q) {
3813 return FORWARD;
3814 }
3815
3816 /**
3817 * Called when an event is being delivered to the next stage.
3818 */
3819 protected void onDeliverToNext(QueuedInputEvent q) {
Michael Wright06a79252014-05-05 17:45:29 -07003820 if (DEBUG_INPUT_STAGES) {
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08003821 Log.v(mTag, "Done with " + getClass().getSimpleName() + ". " + q);
Michael Wright06a79252014-05-05 17:45:29 -07003822 }
Jeff Brownf9e989d2013-04-04 23:04:03 -07003823 if (mNext != null) {
3824 mNext.deliver(q);
3825 } else {
3826 finishInputEvent(q);
3827 }
3828 }
Jeff Brown5182c782013-10-15 20:31:52 -07003829
Michael Wright17d28ca2013-10-31 17:47:45 -07003830 protected boolean shouldDropInputEvent(QueuedInputEvent q) {
3831 if (mView == null || !mAdded) {
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08003832 Slog.w(mTag, "Dropping event due to root view being removed: " + q.mEvent);
Michael Wright17d28ca2013-10-31 17:47:45 -07003833 return true;
George Mount41725de2015-04-09 08:23:05 -07003834 } else if ((!mAttachInfo.mHasWindowFocus
3835 && !q.mEvent.isFromSource(InputDevice.SOURCE_CLASS_POINTER)) || mStopped
Joe LaPenna90776de2016-01-22 07:11:49 -08003836 || (mIsAmbientMode && !q.mEvent.isFromSource(InputDevice.SOURCE_CLASS_BUTTON))
3837 || (mPausedForTransition && !isBack(q.mEvent))) {
Wale Ogunwalec3672cd2014-11-05 15:17:35 -08003838 // This is a focus event and the window doesn't currently have input focus or
3839 // has stopped. This could be an event that came back from the previous stage
3840 // but the window has lost focus or stopped in the meantime.
3841 if (isTerminalInputEvent(q.mEvent)) {
3842 // Don't drop terminal input events, however mark them as canceled.
3843 q.mEvent.cancel();
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08003844 Slog.w(mTag, "Cancelling event due to no window focus: " + q.mEvent);
Wale Ogunwalec3672cd2014-11-05 15:17:35 -08003845 return false;
3846 }
3847
3848 // Drop non-terminal input events.
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08003849 Slog.w(mTag, "Dropping event due to no window focus: " + q.mEvent);
Michael Wright17d28ca2013-10-31 17:47:45 -07003850 return true;
3851 }
3852 return false;
3853 }
3854
Jeff Brown5182c782013-10-15 20:31:52 -07003855 void dump(String prefix, PrintWriter writer) {
3856 if (mNext != null) {
3857 mNext.dump(prefix, writer);
3858 }
3859 }
George Mount41725de2015-04-09 08:23:05 -07003860
3861 private boolean isBack(InputEvent event) {
3862 if (event instanceof KeyEvent) {
3863 return ((KeyEvent) event).getKeyCode() == KeyEvent.KEYCODE_BACK;
3864 } else {
3865 return false;
3866 }
3867 }
Jeff Brownf9e989d2013-04-04 23:04:03 -07003868 }
3869
3870 /**
3871 * Base class for implementing an input pipeline stage that supports
3872 * asynchronous and out-of-order processing of input events.
3873 * <p>
3874 * In addition to what a normal input stage can do, an asynchronous
3875 * input stage may also defer an input event that has been delivered to it
3876 * and finish or forward it later.
3877 * </p>
3878 */
3879 abstract class AsyncInputStage extends InputStage {
3880 private final String mTraceCounter;
3881
3882 private QueuedInputEvent mQueueHead;
3883 private QueuedInputEvent mQueueTail;
3884 private int mQueueLength;
3885
3886 protected static final int DEFER = 3;
3887
3888 /**
3889 * Creates an asynchronous input stage.
3890 * @param next The next stage to which events should be forwarded.
3891 * @param traceCounter The name of a counter to record the size of
3892 * the queue of pending events.
3893 */
3894 public AsyncInputStage(InputStage next, String traceCounter) {
3895 super(next);
3896 mTraceCounter = traceCounter;
3897 }
3898
3899 /**
3900 * Marks the event as deferred, which is to say that it will be handled
3901 * asynchronously. The caller is responsible for calling {@link #forward}
3902 * or {@link #finish} later when it is done handling the event.
3903 */
3904 protected void defer(QueuedInputEvent q) {
3905 q.mFlags |= QueuedInputEvent.FLAG_DEFERRED;
3906 enqueue(q);
3907 }
3908
3909 @Override
3910 protected void forward(QueuedInputEvent q) {
3911 // Clear the deferred flag.
3912 q.mFlags &= ~QueuedInputEvent.FLAG_DEFERRED;
3913
3914 // Fast path if the queue is empty.
3915 QueuedInputEvent curr = mQueueHead;
3916 if (curr == null) {
3917 super.forward(q);
3918 return;
3919 }
3920
3921 // Determine whether the event must be serialized behind any others
3922 // before it can be delivered to the next stage. This is done because
3923 // deferred events might be handled out of order by the stage.
3924 final int deviceId = q.mEvent.getDeviceId();
3925 QueuedInputEvent prev = null;
3926 boolean blocked = false;
3927 while (curr != null && curr != q) {
3928 if (!blocked && deviceId == curr.mEvent.getDeviceId()) {
3929 blocked = true;
3930 }
3931 prev = curr;
3932 curr = curr.mNext;
3933 }
3934
3935 // If the event is blocked, then leave it in the queue to be delivered later.
3936 // Note that the event might not yet be in the queue if it was not previously
3937 // deferred so we will enqueue it if needed.
3938 if (blocked) {
3939 if (curr == null) {
3940 enqueue(q);
3941 }
3942 return;
3943 }
3944
3945 // The event is not blocked. Deliver it immediately.
3946 if (curr != null) {
3947 curr = curr.mNext;
3948 dequeue(q, prev);
3949 }
3950 super.forward(q);
3951
3952 // Dequeuing this event may have unblocked successors. Deliver them.
3953 while (curr != null) {
3954 if (deviceId == curr.mEvent.getDeviceId()) {
3955 if ((curr.mFlags & QueuedInputEvent.FLAG_DEFERRED) != 0) {
3956 break;
3957 }
3958 QueuedInputEvent next = curr.mNext;
3959 dequeue(curr, prev);
3960 super.forward(curr);
3961 curr = next;
3962 } else {
3963 prev = curr;
3964 curr = curr.mNext;
3965 }
3966 }
3967 }
3968
3969 @Override
3970 protected void apply(QueuedInputEvent q, int result) {
3971 if (result == DEFER) {
3972 defer(q);
3973 } else {
3974 super.apply(q, result);
3975 }
3976 }
3977
3978 private void enqueue(QueuedInputEvent q) {
3979 if (mQueueTail == null) {
3980 mQueueHead = q;
3981 mQueueTail = q;
3982 } else {
3983 mQueueTail.mNext = q;
3984 mQueueTail = q;
3985 }
3986
3987 mQueueLength += 1;
3988 Trace.traceCounter(Trace.TRACE_TAG_INPUT, mTraceCounter, mQueueLength);
3989 }
3990
3991 private void dequeue(QueuedInputEvent q, QueuedInputEvent prev) {
3992 if (prev == null) {
3993 mQueueHead = q.mNext;
3994 } else {
3995 prev.mNext = q.mNext;
3996 }
3997 if (mQueueTail == q) {
3998 mQueueTail = prev;
3999 }
4000 q.mNext = null;
4001
4002 mQueueLength -= 1;
4003 Trace.traceCounter(Trace.TRACE_TAG_INPUT, mTraceCounter, mQueueLength);
4004 }
Jeff Brown5182c782013-10-15 20:31:52 -07004005
4006 @Override
4007 void dump(String prefix, PrintWriter writer) {
4008 writer.print(prefix);
4009 writer.print(getClass().getName());
4010 writer.print(": mQueueLength=");
4011 writer.println(mQueueLength);
4012
4013 super.dump(prefix, writer);
4014 }
Jeff Brownf9e989d2013-04-04 23:04:03 -07004015 }
4016
4017 /**
4018 * Delivers pre-ime input events to a native activity.
4019 * Does not support pointer events.
4020 */
Michael Wrighta44dd262013-04-10 21:12:00 -07004021 final class NativePreImeInputStage extends AsyncInputStage
4022 implements InputQueue.FinishedInputEventCallback {
Jeff Brownf9e989d2013-04-04 23:04:03 -07004023 public NativePreImeInputStage(InputStage next, String traceCounter) {
4024 super(next, traceCounter);
4025 }
4026
4027 @Override
4028 protected int onProcess(QueuedInputEvent q) {
Michael Wrighta44dd262013-04-10 21:12:00 -07004029 if (mInputQueue != null && q.mEvent instanceof KeyEvent) {
4030 mInputQueue.sendInputEvent(q.mEvent, q, true, this);
4031 return DEFER;
4032 }
Jeff Brownf9e989d2013-04-04 23:04:03 -07004033 return FORWARD;
4034 }
Michael Wrighta44dd262013-04-10 21:12:00 -07004035
4036 @Override
4037 public void onFinishedInputEvent(Object token, boolean handled) {
4038 QueuedInputEvent q = (QueuedInputEvent)token;
4039 if (handled) {
4040 finish(q, true);
4041 return;
4042 }
4043 forward(q);
4044 }
Jeff Brownf9e989d2013-04-04 23:04:03 -07004045 }
4046
4047 /**
4048 * Delivers pre-ime input events to the view hierarchy.
4049 * Does not support pointer events.
4050 */
4051 final class ViewPreImeInputStage extends InputStage {
4052 public ViewPreImeInputStage(InputStage next) {
4053 super(next);
4054 }
4055
4056 @Override
4057 protected int onProcess(QueuedInputEvent q) {
Jeff Brown481c1572012-03-09 14:41:15 -08004058 if (q.mEvent instanceof KeyEvent) {
Jeff Brownf9e989d2013-04-04 23:04:03 -07004059 return processKeyEvent(q);
4060 }
4061 return FORWARD;
4062 }
4063
4064 private int processKeyEvent(QueuedInputEvent q) {
4065 final KeyEvent event = (KeyEvent)q.mEvent;
4066 if (mView.dispatchKeyEventPreIme(event)) {
4067 return FINISH_HANDLED;
4068 }
4069 return FORWARD;
4070 }
4071 }
4072
4073 /**
4074 * Delivers input events to the ime.
4075 * Does not support pointer events.
4076 */
4077 final class ImeInputStage extends AsyncInputStage
4078 implements InputMethodManager.FinishedInputEventCallback {
4079 public ImeInputStage(InputStage next, String traceCounter) {
4080 super(next, traceCounter);
4081 }
4082
4083 @Override
4084 protected int onProcess(QueuedInputEvent q) {
keunyoung30f420f2013-08-02 14:23:10 -07004085 if (mLastWasImTarget && !isInLocalFocusMode()) {
Jeff Brownf9e989d2013-04-04 23:04:03 -07004086 InputMethodManager imm = InputMethodManager.peekInstance();
4087 if (imm != null) {
4088 final InputEvent event = q.mEvent;
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08004089 if (DEBUG_IMF) Log.v(mTag, "Sending input event to IME: " + event);
Jeff Brownf9e989d2013-04-04 23:04:03 -07004090 int result = imm.dispatchInputEvent(event, q, this, mHandler);
4091 if (result == InputMethodManager.DISPATCH_HANDLED) {
4092 return FINISH_HANDLED;
4093 } else if (result == InputMethodManager.DISPATCH_NOT_HANDLED) {
Adam Lesinskic0f6eed2013-10-28 16:03:19 -07004094 // The IME could not handle it, so skip along to the next InputStage
4095 return FORWARD;
Jeff Brownf9e989d2013-04-04 23:04:03 -07004096 } else {
4097 return DEFER; // callback will be invoked later
4098 }
4099 }
4100 }
4101 return FORWARD;
4102 }
4103
4104 @Override
4105 public void onFinishedInputEvent(Object token, boolean handled) {
4106 QueuedInputEvent q = (QueuedInputEvent)token;
4107 if (handled) {
4108 finish(q, true);
4109 return;
4110 }
Jeff Brownf9e989d2013-04-04 23:04:03 -07004111 forward(q);
4112 }
4113 }
4114
4115 /**
4116 * Performs early processing of post-ime input events.
4117 */
4118 final class EarlyPostImeInputStage extends InputStage {
4119 public EarlyPostImeInputStage(InputStage next) {
4120 super(next);
4121 }
4122
4123 @Override
4124 protected int onProcess(QueuedInputEvent q) {
4125 if (q.mEvent instanceof KeyEvent) {
4126 return processKeyEvent(q);
Jeff Brown4952dfd2011-11-30 19:23:22 -08004127 } else {
Jeff Brown481c1572012-03-09 14:41:15 -08004128 final int source = q.mEvent.getSource();
4129 if ((source & InputDevice.SOURCE_CLASS_POINTER) != 0) {
Jeff Brownf9e989d2013-04-04 23:04:03 -07004130 return processPointerEvent(q);
Jeff Brown481c1572012-03-09 14:41:15 -08004131 }
Jeff Brown4952dfd2011-11-30 19:23:22 -08004132 }
Jeff Brownf9e989d2013-04-04 23:04:03 -07004133 return FORWARD;
4134 }
4135
4136 private int processKeyEvent(QueuedInputEvent q) {
4137 final KeyEvent event = (KeyEvent)q.mEvent;
4138
4139 // If the key's purpose is to exit touch mode then we consume it
4140 // and consider it handled.
4141 if (checkForLeavingTouchModeAndConsume(event)) {
4142 return FINISH_HANDLED;
4143 }
4144
4145 // Make sure the fallback event policy sees all keys that will be
4146 // delivered to the view hierarchy.
4147 mFallbackEventHandler.preDispatchKeyEvent(event);
4148 return FORWARD;
4149 }
4150
4151 private int processPointerEvent(QueuedInputEvent q) {
4152 final MotionEvent event = (MotionEvent)q.mEvent;
4153
4154 // Translate the pointer event for compatibility, if needed.
4155 if (mTranslator != null) {
4156 mTranslator.translateEventInScreenToAppWindow(event);
4157 }
4158
4159 // Enter touch mode on down or scroll.
4160 final int action = event.getAction();
4161 if (action == MotionEvent.ACTION_DOWN || action == MotionEvent.ACTION_SCROLL) {
4162 ensureTouchMode(true);
4163 }
4164
4165 // Offset the scroll position.
4166 if (mCurScrollY != 0) {
4167 event.offsetLocation(0, mCurScrollY);
4168 }
4169
4170 // Remember the touch position for possible drag-initiation.
4171 if (event.isTouchEvent()) {
4172 mLastTouchPoint.x = event.getRawX();
4173 mLastTouchPoint.y = event.getRawY();
Vladislav Kaznacheevba761122016-01-22 12:09:45 -08004174 mLastTouchSource = event.getSource();
Jeff Brownf9e989d2013-04-04 23:04:03 -07004175 }
4176 return FORWARD;
Jeff Brown4952dfd2011-11-30 19:23:22 -08004177 }
4178 }
4179
Jeff Brownf9e989d2013-04-04 23:04:03 -07004180 /**
4181 * Delivers post-ime input events to a native activity.
4182 */
Michael Wrighta44dd262013-04-10 21:12:00 -07004183 final class NativePostImeInputStage extends AsyncInputStage
4184 implements InputQueue.FinishedInputEventCallback {
Jeff Brownf9e989d2013-04-04 23:04:03 -07004185 public NativePostImeInputStage(InputStage next, String traceCounter) {
4186 super(next, traceCounter);
4187 }
4188
4189 @Override
4190 protected int onProcess(QueuedInputEvent q) {
Michael Wrighta44dd262013-04-10 21:12:00 -07004191 if (mInputQueue != null) {
4192 mInputQueue.sendInputEvent(q.mEvent, q, false, this);
4193 return DEFER;
4194 }
Jeff Brownf9e989d2013-04-04 23:04:03 -07004195 return FORWARD;
4196 }
Michael Wrighta44dd262013-04-10 21:12:00 -07004197
4198 @Override
4199 public void onFinishedInputEvent(Object token, boolean handled) {
4200 QueuedInputEvent q = (QueuedInputEvent)token;
4201 if (handled) {
4202 finish(q, true);
4203 return;
4204 }
4205 forward(q);
4206 }
Jeff Brownf9e989d2013-04-04 23:04:03 -07004207 }
4208
4209 /**
4210 * Delivers post-ime input events to the view hierarchy.
4211 */
4212 final class ViewPostImeInputStage extends InputStage {
4213 public ViewPostImeInputStage(InputStage next) {
4214 super(next);
4215 }
4216
4217 @Override
4218 protected int onProcess(QueuedInputEvent q) {
Jeff Brown29c0ed22013-01-14 13:50:37 -08004219 if (q.mEvent instanceof KeyEvent) {
Jeff Brownf9e989d2013-04-04 23:04:03 -07004220 return processKeyEvent(q);
Jeff Brown29c0ed22013-01-14 13:50:37 -08004221 } else {
4222 final int source = q.mEvent.getSource();
Jeff Brownf9e989d2013-04-04 23:04:03 -07004223 if ((source & InputDevice.SOURCE_CLASS_POINTER) != 0) {
4224 return processPointerEvent(q);
4225 } else if ((source & InputDevice.SOURCE_CLASS_TRACKBALL) != 0) {
4226 return processTrackballEvent(q);
Jeff Brown29c0ed22013-01-14 13:50:37 -08004227 } else {
Jeff Brownf9e989d2013-04-04 23:04:03 -07004228 return processGenericMotionEvent(q);
Jeff Brown29c0ed22013-01-14 13:50:37 -08004229 }
4230 }
Jeff Brown29c0ed22013-01-14 13:50:37 -08004231 }
Jeff Brown29c0ed22013-01-14 13:50:37 -08004232
Michael Wright9d744c72014-02-18 21:27:42 -08004233 @Override
4234 protected void onDeliverToNext(QueuedInputEvent q) {
4235 if (mUnbufferedInputDispatch
4236 && q.mEvent instanceof MotionEvent
4237 && ((MotionEvent)q.mEvent).isTouchEvent()
4238 && isTerminalInputEvent(q.mEvent)) {
4239 mUnbufferedInputDispatch = false;
4240 scheduleConsumeBatchedInput();
4241 }
4242 super.onDeliverToNext(q);
4243 }
4244
Jeff Brownf9e989d2013-04-04 23:04:03 -07004245 private int processKeyEvent(QueuedInputEvent q) {
4246 final KeyEvent event = (KeyEvent)q.mEvent;
4247
4248 // Deliver the key to the view hierarchy.
4249 if (mView.dispatchKeyEvent(event)) {
4250 return FINISH_HANDLED;
Jeff Brown21bc5c92011-02-28 18:27:14 -08004251 }
Jeff Brownf9e989d2013-04-04 23:04:03 -07004252
Michael Wright17d28ca2013-10-31 17:47:45 -07004253 if (shouldDropInputEvent(q)) {
4254 return FINISH_NOT_HANDLED;
4255 }
4256
Jeff Brownf9e989d2013-04-04 23:04:03 -07004257 // If the Control modifier is held, try to interpret the key as a shortcut.
4258 if (event.getAction() == KeyEvent.ACTION_DOWN
4259 && event.isCtrlPressed()
4260 && event.getRepeatCount() == 0
4261 && !KeyEvent.isModifierKey(event.getKeyCode())) {
4262 if (mView.dispatchKeyShortcutEvent(event)) {
4263 return FINISH_HANDLED;
4264 }
Michael Wright17d28ca2013-10-31 17:47:45 -07004265 if (shouldDropInputEvent(q)) {
4266 return FINISH_NOT_HANDLED;
4267 }
Jeff Brownf9e989d2013-04-04 23:04:03 -07004268 }
4269
4270 // Apply the fallback event policy.
4271 if (mFallbackEventHandler.dispatchKeyEvent(event)) {
4272 return FINISH_HANDLED;
4273 }
Michael Wright17d28ca2013-10-31 17:47:45 -07004274 if (shouldDropInputEvent(q)) {
4275 return FINISH_NOT_HANDLED;
4276 }
Jeff Brownf9e989d2013-04-04 23:04:03 -07004277
4278 // Handle automatic focus changes.
4279 if (event.getAction() == KeyEvent.ACTION_DOWN) {
4280 int direction = 0;
4281 switch (event.getKeyCode()) {
4282 case KeyEvent.KEYCODE_DPAD_LEFT:
4283 if (event.hasNoModifiers()) {
4284 direction = View.FOCUS_LEFT;
4285 }
4286 break;
4287 case KeyEvent.KEYCODE_DPAD_RIGHT:
4288 if (event.hasNoModifiers()) {
4289 direction = View.FOCUS_RIGHT;
4290 }
4291 break;
4292 case KeyEvent.KEYCODE_DPAD_UP:
4293 if (event.hasNoModifiers()) {
4294 direction = View.FOCUS_UP;
4295 }
4296 break;
4297 case KeyEvent.KEYCODE_DPAD_DOWN:
4298 if (event.hasNoModifiers()) {
4299 direction = View.FOCUS_DOWN;
4300 }
4301 break;
4302 case KeyEvent.KEYCODE_TAB:
4303 if (event.hasNoModifiers()) {
4304 direction = View.FOCUS_FORWARD;
4305 } else if (event.hasModifiers(KeyEvent.META_SHIFT_ON)) {
4306 direction = View.FOCUS_BACKWARD;
4307 }
4308 break;
4309 }
4310 if (direction != 0) {
4311 View focused = mView.findFocus();
4312 if (focused != null) {
4313 View v = focused.focusSearch(direction);
4314 if (v != null && v != focused) {
4315 // do the math the get the interesting rect
4316 // of previous focused into the coord system of
4317 // newly focused view
4318 focused.getFocusedRect(mTempRect);
4319 if (mView instanceof ViewGroup) {
4320 ((ViewGroup) mView).offsetDescendantRectToMyCoords(
4321 focused, mTempRect);
4322 ((ViewGroup) mView).offsetRectIntoDescendantCoords(
4323 v, mTempRect);
4324 }
4325 if (v.requestFocus(direction, mTempRect)) {
4326 playSoundEffect(SoundEffectConstants
4327 .getContantForFocusDirection(direction));
4328 return FINISH_HANDLED;
4329 }
4330 }
4331
4332 // Give the focused view a last chance to handle the dpad key.
4333 if (mView.dispatchUnhandledMove(focused, direction)) {
4334 return FINISH_HANDLED;
4335 }
4336 } else {
4337 // find the best view to give focus to in this non-touch-mode with no-focus
4338 View v = focusSearch(null, direction);
4339 if (v != null && v.requestFocus(direction)) {
4340 return FINISH_HANDLED;
4341 }
4342 }
4343 }
4344 }
4345 return FORWARD;
Jeff Brown21bc5c92011-02-28 18:27:14 -08004346 }
4347
Jeff Brownf9e989d2013-04-04 23:04:03 -07004348 private int processPointerEvent(QueuedInputEvent q) {
4349 final MotionEvent event = (MotionEvent)q.mEvent;
4350
Vladislav Kaznacheev527905e2016-02-16 16:20:56 -08004351 mAttachInfo.mUnbufferedDispatchRequested = false;
4352 final View eventTarget =
4353 (event.isFromSource(InputDevice.SOURCE_MOUSE) && mCapturingView != null) ?
4354 mCapturingView : mView;
4355 mAttachInfo.mHandlingPointerEvent = true;
4356 boolean handled = eventTarget.dispatchPointerEvent(event);
4357 maybeUpdatePointerIcon(event);
4358 mAttachInfo.mHandlingPointerEvent = false;
4359 if (mAttachInfo.mUnbufferedDispatchRequested && !mUnbufferedInputDispatch) {
4360 mUnbufferedInputDispatch = true;
4361 if (mConsumeBatchedInputScheduled) {
4362 scheduleConsumeBatchedInputImmediately();
4363 }
4364 }
4365 return handled ? FINISH_HANDLED : FORWARD;
4366 }
4367
4368 private void maybeUpdatePointerIcon(MotionEvent event) {
Jun Mukai1db53972015-09-11 18:08:31 -07004369 if (event.getPointerCount() == 1
4370 && event.isFromSource(InputDevice.SOURCE_MOUSE)) {
4371 if (event.getActionMasked() == MotionEvent.ACTION_HOVER_ENTER
4372 || event.getActionMasked() == MotionEvent.ACTION_HOVER_EXIT) {
4373 // Other apps or the window manager may change the icon shape outside of
4374 // this app, therefore the icon shape has to be reset on enter/exit event.
4375 mPointerIconShape = PointerIcon.STYLE_NOT_SPECIFIED;
4376 }
4377
Vladislav Kaznacheevec6a4472016-01-22 12:21:52 -08004378 if (event.getActionMasked() != MotionEvent.ACTION_HOVER_EXIT) {
4379 if (!updatePointerIcon(event) &&
4380 event.getActionMasked() == MotionEvent.ACTION_HOVER_MOVE) {
4381 mPointerIconShape = PointerIcon.STYLE_NOT_SPECIFIED;
Jun Mukai1db53972015-09-11 18:08:31 -07004382 }
4383 }
4384 }
Jeff Brown3915bb82010-11-05 15:02:16 -07004385 }
4386
Jeff Brownf9e989d2013-04-04 23:04:03 -07004387 private int processTrackballEvent(QueuedInputEvent q) {
4388 final MotionEvent event = (MotionEvent)q.mEvent;
4389
4390 if (mView.dispatchTrackballEvent(event)) {
4391 return FINISH_HANDLED;
4392 }
4393 return FORWARD;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08004394 }
4395
Jeff Brownf9e989d2013-04-04 23:04:03 -07004396 private int processGenericMotionEvent(QueuedInputEvent q) {
4397 final MotionEvent event = (MotionEvent)q.mEvent;
Jeff Brown00fa7bd2010-07-02 15:37:36 -07004398
Jeff Brownf9e989d2013-04-04 23:04:03 -07004399 // Deliver the event to the view.
4400 if (mView.dispatchGenericMotionEvent(event)) {
4401 return FINISH_HANDLED;
4402 }
4403 return FORWARD;
Jeff Brown3915bb82010-11-05 15:02:16 -07004404 }
Jeff Brown00fa7bd2010-07-02 15:37:36 -07004405 }
4406
Vladislav Kaznacheevec6a4472016-01-22 12:21:52 -08004407 private void resetPointerIcon(MotionEvent event) {
4408 mPointerIconShape = PointerIcon.STYLE_NOT_SPECIFIED;
4409 updatePointerIcon(event);
4410 }
4411
4412 private boolean updatePointerIcon(MotionEvent event) {
4413 final float x = event.getX();
4414 final float y = event.getY();
Andrii Kulian33c1bc52016-02-29 10:38:59 -08004415 if (mView == null) {
4416 // E.g. click outside a popup to dismiss it
4417 Slog.d(mTag, "updatePointerIcon called after view was removed");
4418 return false;
4419 }
Vladislav Kaznacheevec6a4472016-01-22 12:21:52 -08004420 if (x < 0 || x >= mView.getWidth() || y < 0 || y >= mView.getHeight()) {
Andrii Kulian33c1bc52016-02-29 10:38:59 -08004421 // E.g. when moving window divider with mouse
4422 Slog.d(mTag, "updatePointerIcon called with position out of bounds");
Vladislav Kaznacheevec6a4472016-01-22 12:21:52 -08004423 return false;
4424 }
4425 final PointerIcon pointerIcon = mView.getPointerIcon(event, x, y);
4426 final int pointerShape = (pointerIcon != null) ?
4427 pointerIcon.getStyle() : PointerIcon.STYLE_DEFAULT;
4428
4429 if (mPointerIconShape != pointerShape) {
4430 mPointerIconShape = pointerShape;
4431 if (mPointerIconShape != PointerIcon.STYLE_CUSTOM) {
4432 mCustomPointerIcon = null;
4433 InputManager.getInstance().setPointerIconShape(pointerShape);
4434 return true;
4435 }
4436 }
4437 if (mPointerIconShape == PointerIcon.STYLE_CUSTOM &&
4438 !pointerIcon.equals(mCustomPointerIcon)) {
4439 mCustomPointerIcon = pointerIcon;
4440 InputManager.getInstance().setCustomPointerIcon(mCustomPointerIcon);
4441 }
4442 return true;
4443 }
4444
Jeff Brownf9e989d2013-04-04 23:04:03 -07004445 /**
Jeff Brown678a1252013-04-09 17:46:25 -07004446 * Performs synthesis of new input events from unhandled input events.
Jeff Brownf9e989d2013-04-04 23:04:03 -07004447 */
4448 final class SyntheticInputStage extends InputStage {
Jeff Brown678a1252013-04-09 17:46:25 -07004449 private final SyntheticTrackballHandler mTrackball = new SyntheticTrackballHandler();
4450 private final SyntheticJoystickHandler mJoystick = new SyntheticJoystickHandler();
4451 private final SyntheticTouchNavigationHandler mTouchNavigation =
4452 new SyntheticTouchNavigationHandler();
Michael Wright899d7052014-04-23 17:23:39 -07004453 private final SyntheticKeyboardHandler mKeyboard = new SyntheticKeyboardHandler();
Jeff Brownf9e989d2013-04-04 23:04:03 -07004454
4455 public SyntheticInputStage() {
4456 super(null);
Jeff Brown21bc5c92011-02-28 18:27:14 -08004457 }
4458
Jeff Brownf9e989d2013-04-04 23:04:03 -07004459 @Override
4460 protected int onProcess(QueuedInputEvent q) {
4461 q.mFlags |= QueuedInputEvent.FLAG_RESYNTHESIZED;
4462 if (q.mEvent instanceof MotionEvent) {
Jeff Brown678a1252013-04-09 17:46:25 -07004463 final MotionEvent event = (MotionEvent)q.mEvent;
4464 final int source = event.getSource();
Jeff Brownf9e989d2013-04-04 23:04:03 -07004465 if ((source & InputDevice.SOURCE_CLASS_TRACKBALL) != 0) {
Jeff Brown678a1252013-04-09 17:46:25 -07004466 mTrackball.process(event);
4467 return FINISH_HANDLED;
Jeff Brownf9e989d2013-04-04 23:04:03 -07004468 } else if ((source & InputDevice.SOURCE_CLASS_JOYSTICK) != 0) {
Jeff Brown678a1252013-04-09 17:46:25 -07004469 mJoystick.process(event);
4470 return FINISH_HANDLED;
Jeff Brownf9e989d2013-04-04 23:04:03 -07004471 } else if ((source & InputDevice.SOURCE_TOUCH_NAVIGATION)
4472 == InputDevice.SOURCE_TOUCH_NAVIGATION) {
Jeff Brown678a1252013-04-09 17:46:25 -07004473 mTouchNavigation.process(event);
4474 return FINISH_HANDLED;
Jeff Brownf9e989d2013-04-04 23:04:03 -07004475 }
Michael Wright899d7052014-04-23 17:23:39 -07004476 } else if ((q.mFlags & QueuedInputEvent.FLAG_UNHANDLED) != 0) {
4477 mKeyboard.process((KeyEvent)q.mEvent);
4478 return FINISH_HANDLED;
Jeff Brownf9e989d2013-04-04 23:04:03 -07004479 }
Michael Wright899d7052014-04-23 17:23:39 -07004480
Jeff Brownf9e989d2013-04-04 23:04:03 -07004481 return FORWARD;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08004482 }
4483
Jeff Brownf9e989d2013-04-04 23:04:03 -07004484 @Override
4485 protected void onDeliverToNext(QueuedInputEvent q) {
4486 if ((q.mFlags & QueuedInputEvent.FLAG_RESYNTHESIZED) == 0) {
4487 // Cancel related synthetic events if any prior stage has handled the event.
4488 if (q.mEvent instanceof MotionEvent) {
Jeff Brown678a1252013-04-09 17:46:25 -07004489 final MotionEvent event = (MotionEvent)q.mEvent;
4490 final int source = event.getSource();
Jeff Brownf9e989d2013-04-04 23:04:03 -07004491 if ((source & InputDevice.SOURCE_CLASS_TRACKBALL) != 0) {
Jeff Brown678a1252013-04-09 17:46:25 -07004492 mTrackball.cancel(event);
Jeff Brownf9e989d2013-04-04 23:04:03 -07004493 } else if ((source & InputDevice.SOURCE_CLASS_JOYSTICK) != 0) {
Jeff Brown678a1252013-04-09 17:46:25 -07004494 mJoystick.cancel(event);
Jeff Brownf9e989d2013-04-04 23:04:03 -07004495 } else if ((source & InputDevice.SOURCE_TOUCH_NAVIGATION)
4496 == InputDevice.SOURCE_TOUCH_NAVIGATION) {
Jeff Brown678a1252013-04-09 17:46:25 -07004497 mTouchNavigation.cancel(event);
Jeff Brownf9e989d2013-04-04 23:04:03 -07004498 }
4499 }
4500 }
4501 super.onDeliverToNext(q);
4502 }
Jeff Brown678a1252013-04-09 17:46:25 -07004503 }
Jeff Brownf9e989d2013-04-04 23:04:03 -07004504
Jeff Brown678a1252013-04-09 17:46:25 -07004505 /**
4506 * Creates dpad events from unhandled trackball movements.
4507 */
4508 final class SyntheticTrackballHandler {
4509 private final TrackballAxis mX = new TrackballAxis();
4510 private final TrackballAxis mY = new TrackballAxis();
4511 private long mLastTime;
Jeff Brownf9e989d2013-04-04 23:04:03 -07004512
Jeff Brown678a1252013-04-09 17:46:25 -07004513 public void process(MotionEvent event) {
Jeff Brownf9e989d2013-04-04 23:04:03 -07004514 // Translate the trackball event into DPAD keys and try to deliver those.
Jeff Brownf9e989d2013-04-04 23:04:03 -07004515 long curTime = SystemClock.uptimeMillis();
Jeff Brown678a1252013-04-09 17:46:25 -07004516 if ((mLastTime + MAX_TRACKBALL_DELAY) < curTime) {
Jeff Brownf9e989d2013-04-04 23:04:03 -07004517 // It has been too long since the last movement,
4518 // so restart at the beginning.
Jeff Brown678a1252013-04-09 17:46:25 -07004519 mX.reset(0);
4520 mY.reset(0);
4521 mLastTime = curTime;
Jeff Brownf9e989d2013-04-04 23:04:03 -07004522 }
4523
4524 final int action = event.getAction();
4525 final int metaState = event.getMetaState();
4526 switch (action) {
4527 case MotionEvent.ACTION_DOWN:
Jeff Brown678a1252013-04-09 17:46:25 -07004528 mX.reset(2);
4529 mY.reset(2);
Jeff Brownf9e989d2013-04-04 23:04:03 -07004530 enqueueInputEvent(new KeyEvent(curTime, curTime,
4531 KeyEvent.ACTION_DOWN, KeyEvent.KEYCODE_DPAD_CENTER, 0, metaState,
4532 KeyCharacterMap.VIRTUAL_KEYBOARD, 0, KeyEvent.FLAG_FALLBACK,
4533 InputDevice.SOURCE_KEYBOARD));
4534 break;
4535 case MotionEvent.ACTION_UP:
Jeff Brown678a1252013-04-09 17:46:25 -07004536 mX.reset(2);
4537 mY.reset(2);
Jeff Brownf9e989d2013-04-04 23:04:03 -07004538 enqueueInputEvent(new KeyEvent(curTime, curTime,
4539 KeyEvent.ACTION_UP, KeyEvent.KEYCODE_DPAD_CENTER, 0, metaState,
4540 KeyCharacterMap.VIRTUAL_KEYBOARD, 0, KeyEvent.FLAG_FALLBACK,
4541 InputDevice.SOURCE_KEYBOARD));
4542 break;
4543 }
4544
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08004545 if (DEBUG_TRACKBALL) Log.v(mTag, "TB X=" + mX.position + " step="
Jeff Brown678a1252013-04-09 17:46:25 -07004546 + mX.step + " dir=" + mX.dir + " acc=" + mX.acceleration
Jeff Brownf9e989d2013-04-04 23:04:03 -07004547 + " move=" + event.getX()
Jeff Brown678a1252013-04-09 17:46:25 -07004548 + " / Y=" + mY.position + " step="
4549 + mY.step + " dir=" + mY.dir + " acc=" + mY.acceleration
Jeff Brownf9e989d2013-04-04 23:04:03 -07004550 + " move=" + event.getY());
Jeff Brown678a1252013-04-09 17:46:25 -07004551 final float xOff = mX.collect(event.getX(), event.getEventTime(), "X");
4552 final float yOff = mY.collect(event.getY(), event.getEventTime(), "Y");
Jeff Brownf9e989d2013-04-04 23:04:03 -07004553
4554 // Generate DPAD events based on the trackball movement.
4555 // We pick the axis that has moved the most as the direction of
4556 // the DPAD. When we generate DPAD events for one axis, then the
4557 // other axis is reset -- we don't want to perform DPAD jumps due
4558 // to slight movements in the trackball when making major movements
4559 // along the other axis.
4560 int keycode = 0;
4561 int movement = 0;
4562 float accel = 1;
4563 if (xOff > yOff) {
Jeff Brown678a1252013-04-09 17:46:25 -07004564 movement = mX.generate();
Jeff Brownf9e989d2013-04-04 23:04:03 -07004565 if (movement != 0) {
4566 keycode = movement > 0 ? KeyEvent.KEYCODE_DPAD_RIGHT
4567 : KeyEvent.KEYCODE_DPAD_LEFT;
Jeff Brown678a1252013-04-09 17:46:25 -07004568 accel = mX.acceleration;
4569 mY.reset(2);
Jeff Brownf9e989d2013-04-04 23:04:03 -07004570 }
4571 } else if (yOff > 0) {
Jeff Brown678a1252013-04-09 17:46:25 -07004572 movement = mY.generate();
Jeff Brownf9e989d2013-04-04 23:04:03 -07004573 if (movement != 0) {
4574 keycode = movement > 0 ? KeyEvent.KEYCODE_DPAD_DOWN
4575 : KeyEvent.KEYCODE_DPAD_UP;
Jeff Brown678a1252013-04-09 17:46:25 -07004576 accel = mY.acceleration;
4577 mX.reset(2);
Jeff Brownf9e989d2013-04-04 23:04:03 -07004578 }
4579 }
4580
4581 if (keycode != 0) {
4582 if (movement < 0) movement = -movement;
4583 int accelMovement = (int)(movement * accel);
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08004584 if (DEBUG_TRACKBALL) Log.v(mTag, "Move: movement=" + movement
Jeff Brownf9e989d2013-04-04 23:04:03 -07004585 + " accelMovement=" + accelMovement
4586 + " accel=" + accel);
4587 if (accelMovement > movement) {
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08004588 if (DEBUG_TRACKBALL) Log.v(mTag, "Delivering fake DPAD: "
Jeff Brownf9e989d2013-04-04 23:04:03 -07004589 + keycode);
4590 movement--;
4591 int repeatCount = accelMovement - movement;
4592 enqueueInputEvent(new KeyEvent(curTime, curTime,
4593 KeyEvent.ACTION_MULTIPLE, keycode, repeatCount, metaState,
4594 KeyCharacterMap.VIRTUAL_KEYBOARD, 0, KeyEvent.FLAG_FALLBACK,
4595 InputDevice.SOURCE_KEYBOARD));
4596 }
4597 while (movement > 0) {
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08004598 if (DEBUG_TRACKBALL) Log.v(mTag, "Delivering fake DPAD: "
Jeff Brownf9e989d2013-04-04 23:04:03 -07004599 + keycode);
4600 movement--;
4601 curTime = SystemClock.uptimeMillis();
4602 enqueueInputEvent(new KeyEvent(curTime, curTime,
4603 KeyEvent.ACTION_DOWN, keycode, 0, metaState,
4604 KeyCharacterMap.VIRTUAL_KEYBOARD, 0, KeyEvent.FLAG_FALLBACK,
4605 InputDevice.SOURCE_KEYBOARD));
4606 enqueueInputEvent(new KeyEvent(curTime, curTime,
4607 KeyEvent.ACTION_UP, keycode, 0, metaState,
4608 KeyCharacterMap.VIRTUAL_KEYBOARD, 0, KeyEvent.FLAG_FALLBACK,
4609 InputDevice.SOURCE_KEYBOARD));
4610 }
Jeff Brown678a1252013-04-09 17:46:25 -07004611 mLastTime = curTime;
Jeff Brownf9e989d2013-04-04 23:04:03 -07004612 }
Jeff Brownf9e989d2013-04-04 23:04:03 -07004613 }
4614
Jeff Brown678a1252013-04-09 17:46:25 -07004615 public void cancel(MotionEvent event) {
4616 mLastTime = Integer.MIN_VALUE;
Jeff Brown3915bb82010-11-05 15:02:16 -07004617
Jeff Brownf9e989d2013-04-04 23:04:03 -07004618 // If we reach this, we consumed a trackball event.
4619 // Because we will not translate the trackball event into a key event,
4620 // touch mode will not exit, so we exit touch mode here.
4621 if (mView != null && mAdded) {
4622 ensureTouchMode(false);
Jeff Brown00fa7bd2010-07-02 15:37:36 -07004623 }
4624 }
Jeff Brown678a1252013-04-09 17:46:25 -07004625 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08004626
Jeff Brown678a1252013-04-09 17:46:25 -07004627 /**
4628 * Maintains state information for a single trackball axis, generating
4629 * discrete (DPAD) movements based on raw trackball motion.
4630 */
4631 static final class TrackballAxis {
4632 /**
4633 * The maximum amount of acceleration we will apply.
4634 */
4635 static final float MAX_ACCELERATION = 20;
4636
4637 /**
4638 * The maximum amount of time (in milliseconds) between events in order
4639 * for us to consider the user to be doing fast trackball movements,
4640 * and thus apply an acceleration.
4641 */
4642 static final long FAST_MOVE_TIME = 150;
4643
4644 /**
4645 * Scaling factor to the time (in milliseconds) between events to how
4646 * much to multiple/divide the current acceleration. When movement
4647 * is < FAST_MOVE_TIME this multiplies the acceleration; when >
4648 * FAST_MOVE_TIME it divides it.
4649 */
4650 static final float ACCEL_MOVE_SCALING_FACTOR = (1.0f/40);
4651
4652 static final float FIRST_MOVEMENT_THRESHOLD = 0.5f;
4653 static final float SECOND_CUMULATIVE_MOVEMENT_THRESHOLD = 2.0f;
4654 static final float SUBSEQUENT_INCREMENTAL_MOVEMENT_THRESHOLD = 1.0f;
4655
4656 float position;
4657 float acceleration = 1;
4658 long lastMoveTime = 0;
4659 int step;
4660 int dir;
4661 int nonAccelMovement;
4662
4663 void reset(int _step) {
4664 position = 0;
4665 acceleration = 1;
4666 lastMoveTime = 0;
4667 step = _step;
4668 dir = 0;
Jeff Brown6f2fba42011-02-19 01:08:02 -08004669 }
4670
Jeff Brown678a1252013-04-09 17:46:25 -07004671 /**
4672 * Add trackball movement into the state. If the direction of movement
4673 * has been reversed, the state is reset before adding the
4674 * movement (so that you don't have to compensate for any previously
4675 * collected movement before see the result of the movement in the
4676 * new direction).
4677 *
4678 * @return Returns the absolute value of the amount of movement
4679 * collected so far.
4680 */
4681 float collect(float off, long time, String axis) {
4682 long normTime;
4683 if (off > 0) {
4684 normTime = (long)(off * FAST_MOVE_TIME);
4685 if (dir < 0) {
4686 if (DEBUG_TRACKBALL) Log.v(TAG, axis + " reversed to positive!");
4687 position = 0;
4688 step = 0;
4689 acceleration = 1;
4690 lastMoveTime = 0;
4691 }
4692 dir = 1;
4693 } else if (off < 0) {
4694 normTime = (long)((-off) * FAST_MOVE_TIME);
4695 if (dir > 0) {
4696 if (DEBUG_TRACKBALL) Log.v(TAG, axis + " reversed to negative!");
4697 position = 0;
4698 step = 0;
4699 acceleration = 1;
4700 lastMoveTime = 0;
4701 }
4702 dir = -1;
4703 } else {
4704 normTime = 0;
4705 }
4706
4707 // The number of milliseconds between each movement that is
4708 // considered "normal" and will not result in any acceleration
4709 // or deceleration, scaled by the offset we have here.
4710 if (normTime > 0) {
4711 long delta = time - lastMoveTime;
4712 lastMoveTime = time;
4713 float acc = acceleration;
4714 if (delta < normTime) {
4715 // The user is scrolling rapidly, so increase acceleration.
4716 float scale = (normTime-delta) * ACCEL_MOVE_SCALING_FACTOR;
4717 if (scale > 1) acc *= scale;
4718 if (DEBUG_TRACKBALL) Log.v(TAG, axis + " accelerate: off="
4719 + off + " normTime=" + normTime + " delta=" + delta
4720 + " scale=" + scale + " acc=" + acc);
4721 acceleration = acc < MAX_ACCELERATION ? acc : MAX_ACCELERATION;
4722 } else {
4723 // The user is scrolling slowly, so decrease acceleration.
4724 float scale = (delta-normTime) * ACCEL_MOVE_SCALING_FACTOR;
4725 if (scale > 1) acc /= scale;
4726 if (DEBUG_TRACKBALL) Log.v(TAG, axis + " deccelerate: off="
4727 + off + " normTime=" + normTime + " delta=" + delta
4728 + " scale=" + scale + " acc=" + acc);
4729 acceleration = acc > 1 ? acc : 1;
4730 }
4731 }
4732 position += off;
4733 return Math.abs(position);
Jeff Brown6f2fba42011-02-19 01:08:02 -08004734 }
Jeff Browncb1404e2011-01-15 18:14:15 -08004735
Jeff Brown678a1252013-04-09 17:46:25 -07004736 /**
4737 * Generate the number of discrete movement events appropriate for
4738 * the currently collected trackball movement.
4739 *
4740 * @return Returns the number of discrete movements, either positive
4741 * or negative, or 0 if there is not enough trackball movement yet
4742 * for a discrete movement.
4743 */
4744 int generate() {
4745 int movement = 0;
4746 nonAccelMovement = 0;
4747 do {
4748 final int dir = position >= 0 ? 1 : -1;
4749 switch (step) {
4750 // If we are going to execute the first step, then we want
4751 // to do this as soon as possible instead of waiting for
4752 // a full movement, in order to make things look responsive.
4753 case 0:
4754 if (Math.abs(position) < FIRST_MOVEMENT_THRESHOLD) {
4755 return movement;
4756 }
4757 movement += dir;
4758 nonAccelMovement += dir;
4759 step = 1;
4760 break;
4761 // If we have generated the first movement, then we need
4762 // to wait for the second complete trackball motion before
4763 // generating the second discrete movement.
4764 case 1:
4765 if (Math.abs(position) < SECOND_CUMULATIVE_MOVEMENT_THRESHOLD) {
4766 return movement;
4767 }
4768 movement += dir;
4769 nonAccelMovement += dir;
4770 position -= SECOND_CUMULATIVE_MOVEMENT_THRESHOLD * dir;
4771 step = 2;
4772 break;
4773 // After the first two, we generate discrete movements
4774 // consistently with the trackball, applying an acceleration
4775 // if the trackball is moving quickly. This is a simple
4776 // acceleration on top of what we already compute based
4777 // on how quickly the wheel is being turned, to apply
4778 // a longer increasing acceleration to continuous movement
4779 // in one direction.
4780 default:
4781 if (Math.abs(position) < SUBSEQUENT_INCREMENTAL_MOVEMENT_THRESHOLD) {
4782 return movement;
4783 }
4784 movement += dir;
4785 position -= dir * SUBSEQUENT_INCREMENTAL_MOVEMENT_THRESHOLD;
4786 float acc = acceleration;
4787 acc *= 1.1f;
4788 acceleration = acc < MAX_ACCELERATION ? acc : acceleration;
4789 break;
4790 }
4791 } while (true);
4792 }
4793 }
4794
4795 /**
4796 * Creates dpad events from unhandled joystick movements.
4797 */
4798 final class SyntheticJoystickHandler extends Handler {
Michael Wright9adca062014-03-19 11:51:26 -07004799 private final static String TAG = "SyntheticJoystickHandler";
Jeff Brown678a1252013-04-09 17:46:25 -07004800 private final static int MSG_ENQUEUE_X_AXIS_KEY_REPEAT = 1;
4801 private final static int MSG_ENQUEUE_Y_AXIS_KEY_REPEAT = 2;
4802
4803 private int mLastXDirection;
4804 private int mLastYDirection;
4805 private int mLastXKeyCode;
4806 private int mLastYKeyCode;
4807
4808 public SyntheticJoystickHandler() {
4809 super(true);
4810 }
4811
4812 @Override
4813 public void handleMessage(Message msg) {
4814 switch (msg.what) {
4815 case MSG_ENQUEUE_X_AXIS_KEY_REPEAT:
4816 case MSG_ENQUEUE_Y_AXIS_KEY_REPEAT: {
4817 KeyEvent oldEvent = (KeyEvent)msg.obj;
4818 KeyEvent e = KeyEvent.changeTimeRepeat(oldEvent,
4819 SystemClock.uptimeMillis(),
4820 oldEvent.getRepeatCount() + 1);
4821 if (mAttachInfo.mHasWindowFocus) {
4822 enqueueInputEvent(e);
4823 Message m = obtainMessage(msg.what, e);
4824 m.setAsynchronous(true);
4825 sendMessageDelayed(m, ViewConfiguration.getKeyRepeatDelay());
4826 }
4827 } break;
4828 }
4829 }
4830
4831 public void process(MotionEvent event) {
Michael Wright9adca062014-03-19 11:51:26 -07004832 switch(event.getActionMasked()) {
4833 case MotionEvent.ACTION_CANCEL:
4834 cancel(event);
4835 break;
4836 case MotionEvent.ACTION_MOVE:
4837 update(event, true);
4838 break;
4839 default:
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08004840 Log.w(mTag, "Unexpected action: " + event.getActionMasked());
Michael Wright9adca062014-03-19 11:51:26 -07004841 }
Jeff Brown678a1252013-04-09 17:46:25 -07004842 }
4843
Michael Wright9adca062014-03-19 11:51:26 -07004844 private void cancel(MotionEvent event) {
4845 removeMessages(MSG_ENQUEUE_X_AXIS_KEY_REPEAT);
4846 removeMessages(MSG_ENQUEUE_Y_AXIS_KEY_REPEAT);
Jeff Brown678a1252013-04-09 17:46:25 -07004847 update(event, false);
4848 }
4849
4850 private void update(MotionEvent event, boolean synthesizeNewKeys) {
Jeff Brownf9e989d2013-04-04 23:04:03 -07004851 final long time = event.getEventTime();
4852 final int metaState = event.getMetaState();
4853 final int deviceId = event.getDeviceId();
4854 final int source = event.getSource();
4855
4856 int xDirection = joystickAxisValueToDirection(
4857 event.getAxisValue(MotionEvent.AXIS_HAT_X));
4858 if (xDirection == 0) {
4859 xDirection = joystickAxisValueToDirection(event.getX());
Jeff Browncb1404e2011-01-15 18:14:15 -08004860 }
4861
Jeff Brownf9e989d2013-04-04 23:04:03 -07004862 int yDirection = joystickAxisValueToDirection(
4863 event.getAxisValue(MotionEvent.AXIS_HAT_Y));
4864 if (yDirection == 0) {
4865 yDirection = joystickAxisValueToDirection(event.getY());
4866 }
Jeff Browncb1404e2011-01-15 18:14:15 -08004867
Jeff Brown678a1252013-04-09 17:46:25 -07004868 if (xDirection != mLastXDirection) {
4869 if (mLastXKeyCode != 0) {
4870 removeMessages(MSG_ENQUEUE_X_AXIS_KEY_REPEAT);
Jeff Brownf9e989d2013-04-04 23:04:03 -07004871 enqueueInputEvent(new KeyEvent(time, time,
Jeff Brown678a1252013-04-09 17:46:25 -07004872 KeyEvent.ACTION_UP, mLastXKeyCode, 0, metaState,
Jeff Brownf9e989d2013-04-04 23:04:03 -07004873 deviceId, 0, KeyEvent.FLAG_FALLBACK, source));
Jeff Brown678a1252013-04-09 17:46:25 -07004874 mLastXKeyCode = 0;
Jeff Brownf9e989d2013-04-04 23:04:03 -07004875 }
4876
Jeff Brown678a1252013-04-09 17:46:25 -07004877 mLastXDirection = xDirection;
Jeff Brownf9e989d2013-04-04 23:04:03 -07004878
4879 if (xDirection != 0 && synthesizeNewKeys) {
Jeff Brown678a1252013-04-09 17:46:25 -07004880 mLastXKeyCode = xDirection > 0
Jeff Brownf9e989d2013-04-04 23:04:03 -07004881 ? KeyEvent.KEYCODE_DPAD_RIGHT : KeyEvent.KEYCODE_DPAD_LEFT;
4882 final KeyEvent e = new KeyEvent(time, time,
Jeff Brown678a1252013-04-09 17:46:25 -07004883 KeyEvent.ACTION_DOWN, mLastXKeyCode, 0, metaState,
Jeff Brownf9e989d2013-04-04 23:04:03 -07004884 deviceId, 0, KeyEvent.FLAG_FALLBACK, source);
4885 enqueueInputEvent(e);
Jeff Brown678a1252013-04-09 17:46:25 -07004886 Message m = obtainMessage(MSG_ENQUEUE_X_AXIS_KEY_REPEAT, e);
Jeff Brownf9e989d2013-04-04 23:04:03 -07004887 m.setAsynchronous(true);
Michael Wrightb9618522013-04-10 15:20:56 -07004888 sendMessageDelayed(m, ViewConfiguration.getKeyRepeatTimeout());
Jeff Brownf9e989d2013-04-04 23:04:03 -07004889 }
4890 }
4891
Jeff Brown678a1252013-04-09 17:46:25 -07004892 if (yDirection != mLastYDirection) {
4893 if (mLastYKeyCode != 0) {
4894 removeMessages(MSG_ENQUEUE_Y_AXIS_KEY_REPEAT);
Jeff Brownf9e989d2013-04-04 23:04:03 -07004895 enqueueInputEvent(new KeyEvent(time, time,
Jeff Brown678a1252013-04-09 17:46:25 -07004896 KeyEvent.ACTION_UP, mLastYKeyCode, 0, metaState,
Jeff Brownf9e989d2013-04-04 23:04:03 -07004897 deviceId, 0, KeyEvent.FLAG_FALLBACK, source));
Jeff Brown678a1252013-04-09 17:46:25 -07004898 mLastYKeyCode = 0;
Jeff Brownf9e989d2013-04-04 23:04:03 -07004899 }
4900
Jeff Brown678a1252013-04-09 17:46:25 -07004901 mLastYDirection = yDirection;
Jeff Brownf9e989d2013-04-04 23:04:03 -07004902
4903 if (yDirection != 0 && synthesizeNewKeys) {
Jeff Brown678a1252013-04-09 17:46:25 -07004904 mLastYKeyCode = yDirection > 0
Jeff Brownf9e989d2013-04-04 23:04:03 -07004905 ? KeyEvent.KEYCODE_DPAD_DOWN : KeyEvent.KEYCODE_DPAD_UP;
4906 final KeyEvent e = new KeyEvent(time, time,
Jeff Brown678a1252013-04-09 17:46:25 -07004907 KeyEvent.ACTION_DOWN, mLastYKeyCode, 0, metaState,
Jeff Brownf9e989d2013-04-04 23:04:03 -07004908 deviceId, 0, KeyEvent.FLAG_FALLBACK, source);
4909 enqueueInputEvent(e);
Jeff Brown678a1252013-04-09 17:46:25 -07004910 Message m = obtainMessage(MSG_ENQUEUE_Y_AXIS_KEY_REPEAT, e);
Jeff Brownf9e989d2013-04-04 23:04:03 -07004911 m.setAsynchronous(true);
Jeff Brown678a1252013-04-09 17:46:25 -07004912 sendMessageDelayed(m, ViewConfiguration.getKeyRepeatTimeout());
Jeff Brownf9e989d2013-04-04 23:04:03 -07004913 }
Jeff Browncb1404e2011-01-15 18:14:15 -08004914 }
4915 }
4916
Jeff Brownf9e989d2013-04-04 23:04:03 -07004917 private int joystickAxisValueToDirection(float value) {
4918 if (value >= 0.5f) {
4919 return 1;
4920 } else if (value <= -0.5f) {
4921 return -1;
4922 } else {
4923 return 0;
Jeff Browncb1404e2011-01-15 18:14:15 -08004924 }
4925 }
Jeff Brown678a1252013-04-09 17:46:25 -07004926 }
Jeff Browncb1404e2011-01-15 18:14:15 -08004927
Jeff Brown678a1252013-04-09 17:46:25 -07004928 /**
4929 * Creates dpad events from unhandled touch navigation movements.
4930 */
4931 final class SyntheticTouchNavigationHandler extends Handler {
Jeff Brown4dac9012013-04-10 01:03:19 -07004932 private static final String LOCAL_TAG = "SyntheticTouchNavigationHandler";
4933 private static final boolean LOCAL_DEBUG = false;
Jeff Brown678a1252013-04-09 17:46:25 -07004934
Jeff Brown4dac9012013-04-10 01:03:19 -07004935 // Assumed nominal width and height in millimeters of a touch navigation pad,
4936 // if no resolution information is available from the input system.
4937 private static final float DEFAULT_WIDTH_MILLIMETERS = 48;
4938 private static final float DEFAULT_HEIGHT_MILLIMETERS = 48;
Jeff Brown678a1252013-04-09 17:46:25 -07004939
Jeff Brown4dac9012013-04-10 01:03:19 -07004940 /* TODO: These constants should eventually be moved to ViewConfiguration. */
Jeff Brown678a1252013-04-09 17:46:25 -07004941
Jeff Brown4dac9012013-04-10 01:03:19 -07004942 // The nominal distance traveled to move by one unit.
4943 private static final int TICK_DISTANCE_MILLIMETERS = 12;
4944
4945 // Minimum and maximum fling velocity in ticks per second.
4946 // The minimum velocity should be set such that we perform enough ticks per
4947 // second that the fling appears to be fluid. For example, if we set the minimum
4948 // to 2 ticks per second, then there may be up to half a second delay between the next
4949 // to last and last ticks which is noticeably discrete and jerky. This value should
4950 // probably not be set to anything less than about 4.
4951 // If fling accuracy is a problem then consider tuning the tick distance instead.
4952 private static final float MIN_FLING_VELOCITY_TICKS_PER_SECOND = 6f;
4953 private static final float MAX_FLING_VELOCITY_TICKS_PER_SECOND = 20f;
4954
4955 // Fling velocity decay factor applied after each new key is emitted.
4956 // This parameter controls the deceleration and overall duration of the fling.
4957 // The fling stops automatically when its velocity drops below the minimum
4958 // fling velocity defined above.
4959 private static final float FLING_TICK_DECAY = 0.8f;
4960
4961 /* The input device that we are tracking. */
4962
4963 private int mCurrentDeviceId = -1;
4964 private int mCurrentSource;
4965 private boolean mCurrentDeviceSupported;
4966
4967 /* Configuration for the current input device. */
4968
Jeff Brown4dac9012013-04-10 01:03:19 -07004969 // The scaled tick distance. A movement of this amount should generally translate
4970 // into a single dpad event in a given direction.
4971 private float mConfigTickDistance;
4972
4973 // The minimum and maximum scaled fling velocity.
4974 private float mConfigMinFlingVelocity;
4975 private float mConfigMaxFlingVelocity;
4976
4977 /* Tracking state. */
4978
4979 // The velocity tracker for detecting flings.
4980 private VelocityTracker mVelocityTracker;
4981
4982 // The active pointer id, or -1 if none.
4983 private int mActivePointerId = -1;
4984
John Reck79d81e62013-11-05 13:26:57 -08004985 // Location where tracking started.
Jeff Brown4dac9012013-04-10 01:03:19 -07004986 private float mStartX;
4987 private float mStartY;
4988
4989 // Most recently observed position.
4990 private float mLastX;
4991 private float mLastY;
4992
4993 // Accumulated movement delta since the last direction key was sent.
Jeff Brown678a1252013-04-09 17:46:25 -07004994 private float mAccumulatedX;
4995 private float mAccumulatedY;
4996
Jeff Brown4dac9012013-04-10 01:03:19 -07004997 // Set to true if any movement was delivered to the app.
4998 // Implies that tap slop was exceeded.
4999 private boolean mConsumedMovement;
Jeff Brown678a1252013-04-09 17:46:25 -07005000
Jeff Brown4dac9012013-04-10 01:03:19 -07005001 // The most recently sent key down event.
5002 // The keycode remains set until the direction changes or a fling ends
5003 // so that repeated key events may be generated as required.
5004 private long mPendingKeyDownTime;
5005 private int mPendingKeyCode = KeyEvent.KEYCODE_UNKNOWN;
5006 private int mPendingKeyRepeatCount;
5007 private int mPendingKeyMetaState;
Jeff Brown678a1252013-04-09 17:46:25 -07005008
Jeff Brown4dac9012013-04-10 01:03:19 -07005009 // The current fling velocity while a fling is in progress.
5010 private boolean mFlinging;
5011 private float mFlingVelocity;
Jeff Brown678a1252013-04-09 17:46:25 -07005012
5013 public SyntheticTouchNavigationHandler() {
5014 super(true);
Jeff Brown678a1252013-04-09 17:46:25 -07005015 }
5016
5017 public void process(MotionEvent event) {
Jeff Brown4dac9012013-04-10 01:03:19 -07005018 // Update the current device information.
5019 final long time = event.getEventTime();
5020 final int deviceId = event.getDeviceId();
5021 final int source = event.getSource();
5022 if (mCurrentDeviceId != deviceId || mCurrentSource != source) {
5023 finishKeys(time);
5024 finishTracking(time);
5025 mCurrentDeviceId = deviceId;
5026 mCurrentSource = source;
5027 mCurrentDeviceSupported = false;
5028 InputDevice device = event.getDevice();
5029 if (device != null) {
5030 // In order to support an input device, we must know certain
5031 // characteristics about it, such as its size and resolution.
5032 InputDevice.MotionRange xRange = device.getMotionRange(MotionEvent.AXIS_X);
5033 InputDevice.MotionRange yRange = device.getMotionRange(MotionEvent.AXIS_Y);
5034 if (xRange != null && yRange != null) {
5035 mCurrentDeviceSupported = true;
Jeff Brown678a1252013-04-09 17:46:25 -07005036
Jeff Brown4dac9012013-04-10 01:03:19 -07005037 // Infer the resolution if it not actually known.
5038 float xRes = xRange.getResolution();
5039 if (xRes <= 0) {
5040 xRes = xRange.getRange() / DEFAULT_WIDTH_MILLIMETERS;
5041 }
5042 float yRes = yRange.getResolution();
5043 if (yRes <= 0) {
5044 yRes = yRange.getRange() / DEFAULT_HEIGHT_MILLIMETERS;
5045 }
5046 float nominalRes = (xRes + yRes) * 0.5f;
Jeff Brown678a1252013-04-09 17:46:25 -07005047
Jeff Brown4dac9012013-04-10 01:03:19 -07005048 // Precompute all of the configuration thresholds we will need.
Jeff Brown4dac9012013-04-10 01:03:19 -07005049 mConfigTickDistance = TICK_DISTANCE_MILLIMETERS * nominalRes;
5050 mConfigMinFlingVelocity =
5051 MIN_FLING_VELOCITY_TICKS_PER_SECOND * mConfigTickDistance;
5052 mConfigMaxFlingVelocity =
5053 MAX_FLING_VELOCITY_TICKS_PER_SECOND * mConfigTickDistance;
5054
5055 if (LOCAL_DEBUG) {
5056 Log.d(LOCAL_TAG, "Configured device " + mCurrentDeviceId
5057 + " (" + Integer.toHexString(mCurrentSource) + "): "
Jeff Brown4dac9012013-04-10 01:03:19 -07005058 + ", mConfigTickDistance=" + mConfigTickDistance
5059 + ", mConfigMinFlingVelocity=" + mConfigMinFlingVelocity
5060 + ", mConfigMaxFlingVelocity=" + mConfigMaxFlingVelocity);
5061 }
5062 }
5063 }
Jeff Brown678a1252013-04-09 17:46:25 -07005064 }
Jeff Brown4dac9012013-04-10 01:03:19 -07005065 if (!mCurrentDeviceSupported) {
Jeff Brown678a1252013-04-09 17:46:25 -07005066 return;
5067 }
5068
Jeff Brown4dac9012013-04-10 01:03:19 -07005069 // Handle the event.
5070 final int action = event.getActionMasked();
5071 switch (action) {
Jeff Brown678a1252013-04-09 17:46:25 -07005072 case MotionEvent.ACTION_DOWN: {
Jeff Brown4dac9012013-04-10 01:03:19 -07005073 boolean caughtFling = mFlinging;
5074 finishKeys(time);
5075 finishTracking(time);
5076 mActivePointerId = event.getPointerId(0);
5077 mVelocityTracker = VelocityTracker.obtain();
5078 mVelocityTracker.addMovement(event);
Jeff Brown4dac9012013-04-10 01:03:19 -07005079 mStartX = event.getX();
5080 mStartY = event.getY();
5081 mLastX = mStartX;
5082 mLastY = mStartY;
Jeff Brown678a1252013-04-09 17:46:25 -07005083 mAccumulatedX = 0;
5084 mAccumulatedY = 0;
Jeff Brown4dac9012013-04-10 01:03:19 -07005085
5086 // If we caught a fling, then pretend that the tap slop has already
5087 // been exceeded to suppress taps whose only purpose is to stop the fling.
5088 mConsumedMovement = caughtFling;
Jeff Brown678a1252013-04-09 17:46:25 -07005089 break;
5090 }
5091
Jeff Brown4dac9012013-04-10 01:03:19 -07005092 case MotionEvent.ACTION_MOVE:
Jeff Brown678a1252013-04-09 17:46:25 -07005093 case MotionEvent.ACTION_UP: {
Jeff Brown4dac9012013-04-10 01:03:19 -07005094 if (mActivePointerId < 0) {
5095 break;
5096 }
5097 final int index = event.findPointerIndex(mActivePointerId);
5098 if (index < 0) {
5099 finishKeys(time);
5100 finishTracking(time);
5101 break;
5102 }
Jeff Brown678a1252013-04-09 17:46:25 -07005103
Jeff Brown4dac9012013-04-10 01:03:19 -07005104 mVelocityTracker.addMovement(event);
5105 final float x = event.getX(index);
5106 final float y = event.getY(index);
5107 mAccumulatedX += x - mLastX;
5108 mAccumulatedY += y - mLastY;
5109 mLastX = x;
5110 mLastY = y;
5111
5112 // Consume any accumulated movement so far.
5113 final int metaState = event.getMetaState();
5114 consumeAccumulatedMovement(time, metaState);
5115
5116 // Detect taps and flings.
5117 if (action == MotionEvent.ACTION_UP) {
Michael Wright88d7f792013-08-27 15:45:42 -07005118 if (mConsumedMovement && mPendingKeyCode != KeyEvent.KEYCODE_UNKNOWN) {
Jeff Brown4dac9012013-04-10 01:03:19 -07005119 // It might be a fling.
5120 mVelocityTracker.computeCurrentVelocity(1000, mConfigMaxFlingVelocity);
5121 final float vx = mVelocityTracker.getXVelocity(mActivePointerId);
5122 final float vy = mVelocityTracker.getYVelocity(mActivePointerId);
5123 if (!startFling(time, vx, vy)) {
5124 finishKeys(time);
Jeff Brown678a1252013-04-09 17:46:25 -07005125 }
5126 }
Jeff Brown4dac9012013-04-10 01:03:19 -07005127 finishTracking(time);
Jeff Brown678a1252013-04-09 17:46:25 -07005128 }
Jeff Brown4dac9012013-04-10 01:03:19 -07005129 break;
5130 }
5131
5132 case MotionEvent.ACTION_CANCEL: {
5133 finishKeys(time);
5134 finishTracking(time);
Jeff Brown678a1252013-04-09 17:46:25 -07005135 break;
5136 }
5137 }
Jeff Browncb1404e2011-01-15 18:14:15 -08005138 }
Jeff Brown4dac9012013-04-10 01:03:19 -07005139
5140 public void cancel(MotionEvent event) {
5141 if (mCurrentDeviceId == event.getDeviceId()
5142 && mCurrentSource == event.getSource()) {
5143 final long time = event.getEventTime();
5144 finishKeys(time);
5145 finishTracking(time);
5146 }
5147 }
5148
5149 private void finishKeys(long time) {
5150 cancelFling();
5151 sendKeyUp(time);
5152 }
5153
5154 private void finishTracking(long time) {
5155 if (mActivePointerId >= 0) {
5156 mActivePointerId = -1;
5157 mVelocityTracker.recycle();
5158 mVelocityTracker = null;
5159 }
5160 }
5161
5162 private void consumeAccumulatedMovement(long time, int metaState) {
5163 final float absX = Math.abs(mAccumulatedX);
5164 final float absY = Math.abs(mAccumulatedY);
5165 if (absX >= absY) {
5166 if (absX >= mConfigTickDistance) {
5167 mAccumulatedX = consumeAccumulatedMovement(time, metaState, mAccumulatedX,
5168 KeyEvent.KEYCODE_DPAD_LEFT, KeyEvent.KEYCODE_DPAD_RIGHT);
5169 mAccumulatedY = 0;
5170 mConsumedMovement = true;
5171 }
5172 } else {
5173 if (absY >= mConfigTickDistance) {
5174 mAccumulatedY = consumeAccumulatedMovement(time, metaState, mAccumulatedY,
5175 KeyEvent.KEYCODE_DPAD_UP, KeyEvent.KEYCODE_DPAD_DOWN);
5176 mAccumulatedX = 0;
5177 mConsumedMovement = true;
5178 }
5179 }
5180 }
5181
5182 private float consumeAccumulatedMovement(long time, int metaState,
5183 float accumulator, int negativeKeyCode, int positiveKeyCode) {
5184 while (accumulator <= -mConfigTickDistance) {
5185 sendKeyDownOrRepeat(time, negativeKeyCode, metaState);
5186 accumulator += mConfigTickDistance;
5187 }
5188 while (accumulator >= mConfigTickDistance) {
5189 sendKeyDownOrRepeat(time, positiveKeyCode, metaState);
5190 accumulator -= mConfigTickDistance;
5191 }
5192 return accumulator;
5193 }
5194
5195 private void sendKeyDownOrRepeat(long time, int keyCode, int metaState) {
5196 if (mPendingKeyCode != keyCode) {
5197 sendKeyUp(time);
5198 mPendingKeyDownTime = time;
5199 mPendingKeyCode = keyCode;
5200 mPendingKeyRepeatCount = 0;
5201 } else {
5202 mPendingKeyRepeatCount += 1;
5203 }
5204 mPendingKeyMetaState = metaState;
5205
5206 // Note: Normally we would pass FLAG_LONG_PRESS when the repeat count is 1
5207 // but it doesn't quite make sense when simulating the events in this way.
5208 if (LOCAL_DEBUG) {
5209 Log.d(LOCAL_TAG, "Sending key down: keyCode=" + mPendingKeyCode
5210 + ", repeatCount=" + mPendingKeyRepeatCount
5211 + ", metaState=" + Integer.toHexString(mPendingKeyMetaState));
5212 }
5213 enqueueInputEvent(new KeyEvent(mPendingKeyDownTime, time,
5214 KeyEvent.ACTION_DOWN, mPendingKeyCode, mPendingKeyRepeatCount,
5215 mPendingKeyMetaState, mCurrentDeviceId,
5216 KeyEvent.FLAG_FALLBACK, mCurrentSource));
5217 }
5218
5219 private void sendKeyUp(long time) {
5220 if (mPendingKeyCode != KeyEvent.KEYCODE_UNKNOWN) {
5221 if (LOCAL_DEBUG) {
5222 Log.d(LOCAL_TAG, "Sending key up: keyCode=" + mPendingKeyCode
5223 + ", metaState=" + Integer.toHexString(mPendingKeyMetaState));
5224 }
5225 enqueueInputEvent(new KeyEvent(mPendingKeyDownTime, time,
5226 KeyEvent.ACTION_UP, mPendingKeyCode, 0, mPendingKeyMetaState,
5227 mCurrentDeviceId, 0, KeyEvent.FLAG_FALLBACK,
5228 mCurrentSource));
5229 mPendingKeyCode = KeyEvent.KEYCODE_UNKNOWN;
5230 }
5231 }
5232
5233 private boolean startFling(long time, float vx, float vy) {
5234 if (LOCAL_DEBUG) {
5235 Log.d(LOCAL_TAG, "Considering fling: vx=" + vx + ", vy=" + vy
5236 + ", min=" + mConfigMinFlingVelocity);
5237 }
5238
5239 // Flings must be oriented in the same direction as the preceding movements.
5240 switch (mPendingKeyCode) {
5241 case KeyEvent.KEYCODE_DPAD_LEFT:
5242 if (-vx >= mConfigMinFlingVelocity
5243 && Math.abs(vy) < mConfigMinFlingVelocity) {
5244 mFlingVelocity = -vx;
5245 break;
5246 }
5247 return false;
5248
5249 case KeyEvent.KEYCODE_DPAD_RIGHT:
5250 if (vx >= mConfigMinFlingVelocity
5251 && Math.abs(vy) < mConfigMinFlingVelocity) {
5252 mFlingVelocity = vx;
5253 break;
5254 }
5255 return false;
5256
5257 case KeyEvent.KEYCODE_DPAD_UP:
5258 if (-vy >= mConfigMinFlingVelocity
5259 && Math.abs(vx) < mConfigMinFlingVelocity) {
5260 mFlingVelocity = -vy;
5261 break;
5262 }
5263 return false;
5264
5265 case KeyEvent.KEYCODE_DPAD_DOWN:
5266 if (vy >= mConfigMinFlingVelocity
5267 && Math.abs(vx) < mConfigMinFlingVelocity) {
5268 mFlingVelocity = vy;
5269 break;
5270 }
5271 return false;
5272 }
5273
5274 // Post the first fling event.
5275 mFlinging = postFling(time);
5276 return mFlinging;
5277 }
5278
5279 private boolean postFling(long time) {
5280 // The idea here is to estimate the time when the pointer would have
5281 // traveled one tick distance unit given the current fling velocity.
5282 // This effect creates continuity of motion.
5283 if (mFlingVelocity >= mConfigMinFlingVelocity) {
5284 long delay = (long)(mConfigTickDistance / mFlingVelocity * 1000);
5285 postAtTime(mFlingRunnable, time + delay);
5286 if (LOCAL_DEBUG) {
5287 Log.d(LOCAL_TAG, "Posted fling: velocity="
5288 + mFlingVelocity + ", delay=" + delay
5289 + ", keyCode=" + mPendingKeyCode);
5290 }
5291 return true;
5292 }
5293 return false;
5294 }
5295
5296 private void cancelFling() {
5297 if (mFlinging) {
5298 removeCallbacks(mFlingRunnable);
5299 mFlinging = false;
5300 }
5301 }
5302
5303 private final Runnable mFlingRunnable = new Runnable() {
5304 @Override
5305 public void run() {
5306 final long time = SystemClock.uptimeMillis();
5307 sendKeyDownOrRepeat(time, mPendingKeyCode, mPendingKeyMetaState);
5308 mFlingVelocity *= FLING_TICK_DECAY;
5309 if (!postFling(time)) {
5310 mFlinging = false;
5311 finishKeys(time);
5312 }
5313 }
5314 };
Jeff Browncb1404e2011-01-15 18:14:15 -08005315 }
5316
Michael Wright899d7052014-04-23 17:23:39 -07005317 final class SyntheticKeyboardHandler {
5318 public void process(KeyEvent event) {
5319 if ((event.getFlags() & KeyEvent.FLAG_FALLBACK) != 0) {
5320 return;
5321 }
5322
5323 final KeyCharacterMap kcm = event.getKeyCharacterMap();
5324 final int keyCode = event.getKeyCode();
5325 final int metaState = event.getMetaState();
5326
5327 // Check for fallback actions specified by the key character map.
5328 KeyCharacterMap.FallbackAction fallbackAction =
5329 kcm.getFallbackAction(keyCode, metaState);
5330 if (fallbackAction != null) {
5331 final int flags = event.getFlags() | KeyEvent.FLAG_FALLBACK;
5332 KeyEvent fallbackEvent = KeyEvent.obtain(
5333 event.getDownTime(), event.getEventTime(),
5334 event.getAction(), fallbackAction.keyCode,
5335 event.getRepeatCount(), fallbackAction.metaState,
5336 event.getDeviceId(), event.getScanCode(),
5337 flags, event.getSource(), null);
5338 fallbackAction.recycle();
5339 enqueueInputEvent(fallbackEvent);
5340 }
5341 }
5342 }
5343
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08005344 /**
Jeff Brown4e6319b2010-12-13 10:36:51 -08005345 * Returns true if the key is used for keyboard navigation.
5346 * @param keyEvent The key event.
5347 * @return True if the key is used for keyboard navigation.
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08005348 */
Jeff Brown4e6319b2010-12-13 10:36:51 -08005349 private static boolean isNavigationKey(KeyEvent keyEvent) {
5350 switch (keyEvent.getKeyCode()) {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08005351 case KeyEvent.KEYCODE_DPAD_LEFT:
5352 case KeyEvent.KEYCODE_DPAD_RIGHT:
5353 case KeyEvent.KEYCODE_DPAD_UP:
5354 case KeyEvent.KEYCODE_DPAD_DOWN:
Jeff Brown4e6319b2010-12-13 10:36:51 -08005355 case KeyEvent.KEYCODE_DPAD_CENTER:
5356 case KeyEvent.KEYCODE_PAGE_UP:
5357 case KeyEvent.KEYCODE_PAGE_DOWN:
5358 case KeyEvent.KEYCODE_MOVE_HOME:
5359 case KeyEvent.KEYCODE_MOVE_END:
5360 case KeyEvent.KEYCODE_TAB:
5361 case KeyEvent.KEYCODE_SPACE:
5362 case KeyEvent.KEYCODE_ENTER:
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08005363 return true;
5364 }
5365 return false;
5366 }
5367
5368 /**
Jeff Brown4e6319b2010-12-13 10:36:51 -08005369 * Returns true if the key is used for typing.
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08005370 * @param keyEvent The key event.
Jeff Brown4e6319b2010-12-13 10:36:51 -08005371 * @return True if the key is used for typing.
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08005372 */
Jeff Brown4e6319b2010-12-13 10:36:51 -08005373 private static boolean isTypingKey(KeyEvent keyEvent) {
5374 return keyEvent.getUnicodeChar() > 0;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08005375 }
5376
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08005377 /**
Jeff Brown4e6319b2010-12-13 10:36:51 -08005378 * See if the key event means we should leave touch mode (and leave touch mode if so).
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08005379 * @param event The key event.
5380 * @return Whether this key event should be consumed (meaning the act of
5381 * leaving touch mode alone is considered the event).
5382 */
5383 private boolean checkForLeavingTouchModeAndConsume(KeyEvent event) {
Jeff Brown4e6319b2010-12-13 10:36:51 -08005384 // Only relevant in touch mode.
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08005385 if (!mAttachInfo.mInTouchMode) {
5386 return false;
5387 }
5388
Jeff Brown4e6319b2010-12-13 10:36:51 -08005389 // Only consider leaving touch mode on DOWN or MULTIPLE actions, never on UP.
5390 final int action = event.getAction();
5391 if (action != KeyEvent.ACTION_DOWN && action != KeyEvent.ACTION_MULTIPLE) {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08005392 return false;
5393 }
5394
Jeff Brown4e6319b2010-12-13 10:36:51 -08005395 // Don't leave touch mode if the IME told us not to.
5396 if ((event.getFlags() & KeyEvent.FLAG_KEEP_TOUCH_MODE) != 0) {
5397 return false;
5398 }
5399
5400 // If the key can be used for keyboard navigation then leave touch mode
5401 // and select a focused view if needed (in ensureTouchMode).
5402 // When a new focused view is selected, we consume the navigation key because
5403 // navigation doesn't make much sense unless a view already has focus so
5404 // the key's purpose is to set focus.
5405 if (isNavigationKey(event)) {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08005406 return ensureTouchMode(false);
5407 }
Jeff Brown4e6319b2010-12-13 10:36:51 -08005408
5409 // If the key can be used for typing then leave touch mode
5410 // and select a focused view if needed (in ensureTouchMode).
5411 // Always allow the view to process the typing key.
5412 if (isTypingKey(event)) {
5413 ensureTouchMode(false);
5414 return false;
5415 }
5416
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08005417 return false;
5418 }
5419
Christopher Tatea53146c2010-09-07 11:57:52 -07005420 /* drag/drop */
Christopher Tate407b4e92010-11-30 17:14:08 -08005421 void setLocalDragState(Object obj) {
5422 mLocalDragState = obj;
5423 }
5424
Christopher Tatea53146c2010-09-07 11:57:52 -07005425 private void handleDragEvent(DragEvent event) {
5426 // From the root, only drag start/end/location are dispatched. entered/exited
5427 // are determined and dispatched by the viewgroup hierarchy, who then report
5428 // that back here for ultimate reporting back to the framework.
5429 if (mView != null && mAdded) {
5430 final int what = event.mAction;
5431
5432 if (what == DragEvent.ACTION_DRAG_EXITED) {
5433 // A direct EXITED event means that the window manager knows we've just crossed
5434 // a window boundary, so the current drag target within this one must have
5435 // just been exited. Send it the usual notifications and then we're done
5436 // for now.
Chris Tate9d1ab882010-11-02 15:55:39 -07005437 mView.dispatchDragEvent(event);
Christopher Tatea53146c2010-09-07 11:57:52 -07005438 } else {
5439 // Cache the drag description when the operation starts, then fill it in
5440 // on subsequent calls as a convenience
5441 if (what == DragEvent.ACTION_DRAG_STARTED) {
Chris Tate9d1ab882010-11-02 15:55:39 -07005442 mCurrentDragView = null; // Start the current-recipient tracking
Christopher Tatea53146c2010-09-07 11:57:52 -07005443 mDragDescription = event.mClipDescription;
5444 } else {
5445 event.mClipDescription = mDragDescription;
5446 }
5447
5448 // For events with a [screen] location, translate into window coordinates
5449 if ((what == DragEvent.ACTION_DRAG_LOCATION) || (what == DragEvent.ACTION_DROP)) {
5450 mDragPoint.set(event.mX, event.mY);
5451 if (mTranslator != null) {
5452 mTranslator.translatePointInScreenToAppWindow(mDragPoint);
5453 }
5454
5455 if (mCurScrollY != 0) {
5456 mDragPoint.offset(0, mCurScrollY);
5457 }
5458
5459 event.mX = mDragPoint.x;
5460 event.mY = mDragPoint.y;
5461 }
5462
5463 // Remember who the current drag target is pre-dispatch
5464 final View prevDragView = mCurrentDragView;
5465
5466 // Now dispatch the drag/drop event
Chris Tated4533f12010-10-19 15:15:08 -07005467 boolean result = mView.dispatchDragEvent(event);
Christopher Tatea53146c2010-09-07 11:57:52 -07005468
5469 // If we changed apparent drag target, tell the OS about it
5470 if (prevDragView != mCurrentDragView) {
5471 try {
5472 if (prevDragView != null) {
Jeff Brown98365d72012-08-19 20:30:52 -07005473 mWindowSession.dragRecipientExited(mWindow);
Christopher Tatea53146c2010-09-07 11:57:52 -07005474 }
5475 if (mCurrentDragView != null) {
Jeff Brown98365d72012-08-19 20:30:52 -07005476 mWindowSession.dragRecipientEntered(mWindow);
Christopher Tatea53146c2010-09-07 11:57:52 -07005477 }
5478 } catch (RemoteException e) {
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08005479 Slog.e(mTag, "Unable to note drag target change");
Christopher Tatea53146c2010-09-07 11:57:52 -07005480 }
Christopher Tatea53146c2010-09-07 11:57:52 -07005481 }
Chris Tated4533f12010-10-19 15:15:08 -07005482
Christopher Tate407b4e92010-11-30 17:14:08 -08005483 // Report the drop result when we're done
Chris Tated4533f12010-10-19 15:15:08 -07005484 if (what == DragEvent.ACTION_DROP) {
Christopher Tate1fc014f2011-01-19 12:56:26 -08005485 mDragDescription = null;
Chris Tated4533f12010-10-19 15:15:08 -07005486 try {
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08005487 Log.i(mTag, "Reporting drop result: " + result);
Jeff Brown98365d72012-08-19 20:30:52 -07005488 mWindowSession.reportDropResult(mWindow, result);
Chris Tated4533f12010-10-19 15:15:08 -07005489 } catch (RemoteException e) {
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08005490 Log.e(mTag, "Unable to report drop result");
Chris Tated4533f12010-10-19 15:15:08 -07005491 }
5492 }
Christopher Tate407b4e92010-11-30 17:14:08 -08005493
Vladislav Kaznacheev82063912015-11-20 14:20:13 -08005494 // When the drag operation ends, reset drag-related state
Christopher Tate407b4e92010-11-30 17:14:08 -08005495 if (what == DragEvent.ACTION_DRAG_ENDED) {
5496 setLocalDragState(null);
Vladislav Kaznacheev82063912015-11-20 14:20:13 -08005497 mAttachInfo.mDragToken = null;
Vladislav Kaznacheev4f639742015-11-18 13:21:35 -08005498 if (mAttachInfo.mDragSurface != null) {
5499 mAttachInfo.mDragSurface.release();
5500 mAttachInfo.mDragSurface = null;
5501 }
Christopher Tate407b4e92010-11-30 17:14:08 -08005502 }
Christopher Tatea53146c2010-09-07 11:57:52 -07005503 }
5504 }
5505 event.recycle();
5506 }
5507
Dianne Hackborn9a230e02011-10-06 11:51:27 -07005508 public void handleDispatchSystemUiVisibilityChanged(SystemUiVisibilityInfo args) {
5509 if (mSeq != args.seq) {
5510 // The sequence has changed, so we need to update our value and make
5511 // sure to do a traversal afterward so the window manager is given our
5512 // most recent data.
5513 mSeq = args.seq;
5514 mAttachInfo.mForceReportNewAttributes = true;
Igor Murashkina86ab6402013-08-30 12:58:36 -07005515 scheduleTraversals();
Joe Onorato14782f72011-01-25 19:53:17 -08005516 }
Dianne Hackborn9a230e02011-10-06 11:51:27 -07005517 if (mView == null) return;
5518 if (args.localChanges != 0) {
Dianne Hackborn9a230e02011-10-06 11:51:27 -07005519 mView.updateLocalSystemUiVisibility(args.localValue, args.localChanges);
Dianne Hackborn9a230e02011-10-06 11:51:27 -07005520 }
Chris Craikd36a81f2014-07-17 10:16:51 -07005521
5522 int visibility = args.globalVisibility&View.SYSTEM_UI_CLEARABLE_FLAGS;
5523 if (visibility != mAttachInfo.mGlobalSystemUiVisibility) {
5524 mAttachInfo.mGlobalSystemUiVisibility = visibility;
5525 mView.dispatchSystemUiVisibilityChanged(visibility);
Dianne Hackborncf675782012-05-10 15:07:24 -07005526 }
Joe Onorato664644d2011-01-23 17:53:23 -08005527 }
5528
Craig Mautner9c795042014-10-28 19:59:59 -07005529 public void handleDispatchWindowShown() {
5530 mAttachInfo.mTreeObserver.dispatchOnWindowShown();
5531 }
5532
Clara Bayarrifcd7e802016-03-10 12:58:18 +00005533 public void handleRequestKeyboardShortcuts(IResultReceiver receiver, int deviceId) {
Clara Bayarri75e09792015-07-29 16:20:40 +01005534 Bundle data = new Bundle();
5535 ArrayList<KeyboardShortcutGroup> list = new ArrayList<>();
5536 if (mView != null) {
Clara Bayarrifcd7e802016-03-10 12:58:18 +00005537 mView.requestKeyboardShortcuts(list, deviceId);
Clara Bayarri75e09792015-07-29 16:20:40 +01005538 }
5539 data.putParcelableArrayList(WindowManager.PARCEL_KEY_SHORTCUTS_ARRAY, list);
5540 try {
5541 receiver.send(0, data);
5542 } catch (RemoteException e) {
5543 }
5544 }
5545
Christopher Tate2c095f32010-10-04 14:13:40 -07005546 public void getLastTouchPoint(Point outLocation) {
5547 outLocation.x = (int) mLastTouchPoint.x;
5548 outLocation.y = (int) mLastTouchPoint.y;
5549 }
5550
Vladislav Kaznacheevba761122016-01-22 12:09:45 -08005551 public int getLastTouchSource() {
5552 return mLastTouchSource;
5553 }
5554
Chris Tate9d1ab882010-11-02 15:55:39 -07005555 public void setDragFocus(View newDragTarget) {
Christopher Tatea53146c2010-09-07 11:57:52 -07005556 if (mCurrentDragView != newDragTarget) {
Chris Tate048691c2010-10-12 17:39:18 -07005557 mCurrentDragView = newDragTarget;
Christopher Tatea53146c2010-09-07 11:57:52 -07005558 }
Christopher Tatea53146c2010-09-07 11:57:52 -07005559 }
5560
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08005561 private AudioManager getAudioManager() {
5562 if (mView == null) {
5563 throw new IllegalStateException("getAudioManager called when there is no mView");
5564 }
5565 if (mAudioManager == null) {
5566 mAudioManager = (AudioManager) mView.getContext().getSystemService(Context.AUDIO_SERVICE);
5567 }
5568 return mAudioManager;
5569 }
5570
Svetoslav Ganov8643aa02011-04-20 12:12:33 -07005571 public AccessibilityInteractionController getAccessibilityInteractionController() {
5572 if (mView == null) {
5573 throw new IllegalStateException("getAccessibilityInteractionController"
5574 + " called when there is no mView");
5575 }
Gilles Debunne5ac84422011-10-19 09:35:58 -07005576 if (mAccessibilityInteractionController == null) {
Svetoslav Ganov42138042012-03-20 11:51:39 -07005577 mAccessibilityInteractionController = new AccessibilityInteractionController(this);
Svetoslav Ganov8643aa02011-04-20 12:12:33 -07005578 }
Gilles Debunne5ac84422011-10-19 09:35:58 -07005579 return mAccessibilityInteractionController;
Svetoslav Ganov8643aa02011-04-20 12:12:33 -07005580 }
5581
Mitsuru Oshima8169dae2009-04-28 18:12:09 -07005582 private int relayoutWindow(WindowManager.LayoutParams params, int viewVisibility,
5583 boolean insetsPending) throws RemoteException {
Mitsuru Oshima64f59342009-06-21 00:03:11 -07005584
5585 float appScale = mAttachInfo.mApplicationScale;
Mitsuru Oshima3d914922009-05-13 22:29:15 -07005586 boolean restore = false;
Mitsuru Oshima64f59342009-06-21 00:03:11 -07005587 if (params != null && mTranslator != null) {
Mitsuru Oshimae5fb3282009-06-09 21:16:08 -07005588 restore = true;
5589 params.backup();
Mitsuru Oshima64f59342009-06-21 00:03:11 -07005590 mTranslator.translateWindowLayout(params);
Mitsuru Oshima9189cab2009-06-03 11:19:12 -07005591 }
Mitsuru Oshima64f59342009-06-21 00:03:11 -07005592 if (params != null) {
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08005593 if (DBG) Log.d(mTag, "WindowLayout in layoutWindow:" + params);
Mitsuru Oshima3d914922009-05-13 22:29:15 -07005594 }
Dianne Hackborn694f79b2010-03-17 19:44:59 -07005595 mPendingConfiguration.seq = 0;
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08005596 //Log.d(mTag, ">>>>>> CALLING relayout");
Dianne Hackborn180c4842011-09-13 12:39:25 -07005597 if (params != null && mOrigWindowType != params.type) {
5598 // For compatibility with old apps, don't crash here.
Michael Wright5bd69e62015-05-14 14:48:08 +01005599 if (mTargetSdkVersion < Build.VERSION_CODES.ICE_CREAM_SANDWICH) {
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08005600 Slog.w(mTag, "Window type can not be changed after "
Dianne Hackborn180c4842011-09-13 12:39:25 -07005601 + "the window is added; ignoring change of " + mView);
5602 params.type = mOrigWindowType;
5603 }
5604 }
Jeff Brown98365d72012-08-19 20:30:52 -07005605 int relayoutResult = mWindowSession.relayout(
Dianne Hackborn9a230e02011-10-06 11:51:27 -07005606 mWindow, mSeq, params,
Dianne Hackborn189ee182010-12-02 21:48:53 -08005607 (int) (mView.getMeasuredWidth() * appScale + 0.5f),
5608 (int) (mView.getMeasuredHeight() * appScale + 0.5f),
Jeff Brown98365d72012-08-19 20:30:52 -07005609 viewVisibility, insetsPending ? WindowManagerGlobal.RELAYOUT_INSETS_PENDING : 0,
Dianne Hackbornc4aad012013-02-22 15:05:25 -08005610 mWinFrame, mPendingOverscanInsets, mPendingContentInsets, mPendingVisibleInsets,
Jorim Jaggi2e95a482016-01-14 17:36:55 -08005611 mPendingStableInsets, mPendingOutsets, mPendingBackDropFrame, mPendingConfiguration,
5612 mSurface);
Jorim Jaggi0ffd49c2016-02-12 15:04:21 -08005613
5614 mPendingAlwaysConsumeNavBar =
5615 (relayoutResult & WindowManagerGlobal.RELAYOUT_RES_CONSUME_ALWAYS_NAV_BAR) != 0;
5616
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08005617 //Log.d(mTag, "<<<<<< BACK FROM relayout");
Mitsuru Oshima3d914922009-05-13 22:29:15 -07005618 if (restore) {
Mitsuru Oshimae5fb3282009-06-09 21:16:08 -07005619 params.restore();
Mitsuru Oshima3d914922009-05-13 22:29:15 -07005620 }
Igor Murashkina86ab6402013-08-30 12:58:36 -07005621
Mitsuru Oshima64f59342009-06-21 00:03:11 -07005622 if (mTranslator != null) {
5623 mTranslator.translateRectInScreenToAppWinFrame(mWinFrame);
Dianne Hackbornc4aad012013-02-22 15:05:25 -08005624 mTranslator.translateRectInScreenToAppWindow(mPendingOverscanInsets);
Mitsuru Oshima64f59342009-06-21 00:03:11 -07005625 mTranslator.translateRectInScreenToAppWindow(mPendingContentInsets);
5626 mTranslator.translateRectInScreenToAppWindow(mPendingVisibleInsets);
Adrian Roosfa104232014-06-20 16:10:14 -07005627 mTranslator.translateRectInScreenToAppWindow(mPendingStableInsets);
Mitsuru Oshima9189cab2009-06-03 11:19:12 -07005628 }
Mitsuru Oshima8169dae2009-04-28 18:12:09 -07005629 return relayoutResult;
5630 }
Romain Guy8506ab42009-06-11 17:35:47 -07005631
Mitsuru Oshima9189cab2009-06-03 11:19:12 -07005632 /**
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08005633 * {@inheritDoc}
5634 */
Igor Murashkina86ab6402013-08-30 12:58:36 -07005635 @Override
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08005636 public void playSoundEffect(int effectId) {
5637 checkThread();
5638
Jean-Michel Trivi13b18fd2010-05-05 09:18:15 -07005639 try {
5640 final AudioManager audioManager = getAudioManager();
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08005641
Jean-Michel Trivi13b18fd2010-05-05 09:18:15 -07005642 switch (effectId) {
5643 case SoundEffectConstants.CLICK:
5644 audioManager.playSoundEffect(AudioManager.FX_KEY_CLICK);
5645 return;
5646 case SoundEffectConstants.NAVIGATION_DOWN:
5647 audioManager.playSoundEffect(AudioManager.FX_FOCUS_NAVIGATION_DOWN);
5648 return;
5649 case SoundEffectConstants.NAVIGATION_LEFT:
5650 audioManager.playSoundEffect(AudioManager.FX_FOCUS_NAVIGATION_LEFT);
5651 return;
5652 case SoundEffectConstants.NAVIGATION_RIGHT:
5653 audioManager.playSoundEffect(AudioManager.FX_FOCUS_NAVIGATION_RIGHT);
5654 return;
5655 case SoundEffectConstants.NAVIGATION_UP:
5656 audioManager.playSoundEffect(AudioManager.FX_FOCUS_NAVIGATION_UP);
5657 return;
5658 default:
5659 throw new IllegalArgumentException("unknown effect id " + effectId +
5660 " not defined in " + SoundEffectConstants.class.getCanonicalName());
5661 }
5662 } catch (IllegalStateException e) {
5663 // Exception thrown by getAudioManager() when mView is null
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08005664 Log.e(mTag, "FATAL EXCEPTION when attempting to play sound effect: " + e);
Jean-Michel Trivi13b18fd2010-05-05 09:18:15 -07005665 e.printStackTrace();
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08005666 }
5667 }
5668
5669 /**
5670 * {@inheritDoc}
5671 */
Igor Murashkina86ab6402013-08-30 12:58:36 -07005672 @Override
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08005673 public boolean performHapticFeedback(int effectId, boolean always) {
5674 try {
Jeff Brown98365d72012-08-19 20:30:52 -07005675 return mWindowSession.performHapticFeedback(mWindow, effectId, always);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08005676 } catch (RemoteException e) {
5677 return false;
5678 }
5679 }
5680
5681 /**
5682 * {@inheritDoc}
5683 */
Igor Murashkina86ab6402013-08-30 12:58:36 -07005684 @Override
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08005685 public View focusSearch(View focused, int direction) {
5686 checkThread();
5687 if (!(mView instanceof ViewGroup)) {
5688 return null;
5689 }
5690 return FocusFinder.getInstance().findNextFocus((ViewGroup) mView, focused, direction);
5691 }
5692
5693 public void debug() {
5694 mView.debug();
5695 }
Igor Murashkina86ab6402013-08-30 12:58:36 -07005696
Jeff Brown5182c782013-10-15 20:31:52 -07005697 public void dump(String prefix, FileDescriptor fd, PrintWriter writer, String[] args) {
5698 String innerPrefix = prefix + " ";
5699 writer.print(prefix); writer.println("ViewRoot:");
5700 writer.print(innerPrefix); writer.print("mAdded="); writer.print(mAdded);
5701 writer.print(" mRemoved="); writer.println(mRemoved);
5702 writer.print(innerPrefix); writer.print("mConsumeBatchedInputScheduled=");
5703 writer.println(mConsumeBatchedInputScheduled);
Michael Wright9d744c72014-02-18 21:27:42 -08005704 writer.print(innerPrefix); writer.print("mConsumeBatchedInputImmediatelyScheduled=");
5705 writer.println(mConsumeBatchedInputImmediatelyScheduled);
Jeff Brown5182c782013-10-15 20:31:52 -07005706 writer.print(innerPrefix); writer.print("mPendingInputEventCount=");
5707 writer.println(mPendingInputEventCount);
5708 writer.print(innerPrefix); writer.print("mProcessInputEventsScheduled=");
5709 writer.println(mProcessInputEventsScheduled);
5710 writer.print(innerPrefix); writer.print("mTraversalScheduled=");
5711 writer.print(mTraversalScheduled);
Daniel Koulomzin087ae472015-12-16 17:52:25 -05005712 writer.print(innerPrefix); writer.print("mIsAmbientMode=");
5713 writer.print(mIsAmbientMode);
Jeff Brown5182c782013-10-15 20:31:52 -07005714 if (mTraversalScheduled) {
5715 writer.print(" (barrier="); writer.print(mTraversalBarrier); writer.println(")");
5716 } else {
5717 writer.println();
5718 }
5719 mFirstInputStage.dump(innerPrefix, writer);
5720
5721 mChoreographer.dump(prefix, writer);
5722
5723 writer.print(prefix); writer.println("View Hierarchy:");
5724 dumpViewHierarchy(innerPrefix, writer, mView);
5725 }
5726
5727 private void dumpViewHierarchy(String prefix, PrintWriter writer, View view) {
5728 writer.print(prefix);
5729 if (view == null) {
5730 writer.println("null");
5731 return;
5732 }
5733 writer.println(view.toString());
5734 if (!(view instanceof ViewGroup)) {
5735 return;
5736 }
5737 ViewGroup grp = (ViewGroup)view;
5738 final int N = grp.getChildCount();
5739 if (N <= 0) {
5740 return;
5741 }
5742 prefix = prefix + " ";
5743 for (int i=0; i<N; i++) {
5744 dumpViewHierarchy(prefix, writer, grp.getChildAt(i));
5745 }
5746 }
5747
Romain Guy211370f2012-02-01 16:10:55 -08005748 public void dumpGfxInfo(int[] info) {
Romain Guy54ab3472012-06-14 12:52:53 -07005749 info[0] = info[1] = 0;
Romain Guy65b345f2011-07-27 18:51:50 -07005750 if (mView != null) {
5751 getGfxInfo(mView, info);
Romain Guy65b345f2011-07-27 18:51:50 -07005752 }
5753 }
5754
Romain Guya998dff2012-03-23 18:58:36 -07005755 private static void getGfxInfo(View view, int[] info) {
Chris Craik64a12e12014-03-28 18:12:12 -07005756 RenderNode renderNode = view.mRenderNode;
Romain Guy65b345f2011-07-27 18:51:50 -07005757 info[0]++;
Chris Craik64a12e12014-03-28 18:12:12 -07005758 if (renderNode != null) {
John Reckfe5e7b72014-05-23 17:42:28 -07005759 info[1] += renderNode.getDebugSize();
Romain Guy65b345f2011-07-27 18:51:50 -07005760 }
5761
5762 if (view instanceof ViewGroup) {
5763 ViewGroup group = (ViewGroup) view;
5764
5765 int count = group.getChildCount();
5766 for (int i = 0; i < count; i++) {
5767 getGfxInfo(group.getChildAt(i), info);
5768 }
5769 }
5770 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08005771
Craig Mautner8f303ad2013-06-14 11:32:22 -07005772 /**
5773 * @param immediate True, do now if not in traversal. False, put on queue and do later.
5774 * @return True, request has been queued. False, request has been completed.
5775 */
5776 boolean die(boolean immediate) {
Craig Mautnerdf2390a2012-08-29 20:59:22 -07005777 // Make sure we do execute immediately if we are in the middle of a traversal or the damage
5778 // done by dispatchDetachedFromWindow will cause havoc on return.
5779 if (immediate && !mIsInTraversal) {
Dianne Hackborn94d69142009-09-28 22:14:42 -07005780 doDie();
Craig Mautner8f303ad2013-06-14 11:32:22 -07005781 return false;
Dianne Hackborn94d69142009-09-28 22:14:42 -07005782 }
Craig Mautner8f303ad2013-06-14 11:32:22 -07005783
5784 if (!mIsDrawing) {
5785 destroyHardwareRenderer();
5786 } else {
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08005787 Log.e(mTag, "Attempting to destroy the window while drawing!\n" +
Craig Mautner8f303ad2013-06-14 11:32:22 -07005788 " window=" + this + ", title=" + mWindowAttributes.getTitle());
5789 }
5790 mHandler.sendEmptyMessage(MSG_DIE);
5791 return true;
Dianne Hackborn94d69142009-09-28 22:14:42 -07005792 }
5793
5794 void doDie() {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08005795 checkThread();
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08005796 if (LOCAL_LOGV) Log.v(mTag, "DIE in " + this + " of " + mSurface);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08005797 synchronized (this) {
Craig Mautner8f303ad2013-06-14 11:32:22 -07005798 if (mRemoved) {
5799 return;
5800 }
5801 mRemoved = true;
Romain Guy16260e72011-09-01 14:26:11 -07005802 if (mAdded) {
Romain Guy16260e72011-09-01 14:26:11 -07005803 dispatchDetachedFromWindow();
5804 }
5805
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08005806 if (mAdded && !mFirst) {
Romain Guy29d89972010-09-22 16:10:57 -07005807 destroyHardwareRenderer();
5808
Romain Guyedbca122012-04-04 18:25:53 -07005809 if (mView != null) {
5810 int viewVisibility = mView.getVisibility();
5811 boolean viewVisibilityChanged = mViewVisibility != viewVisibility;
5812 if (mWindowAttributesChanged || viewVisibilityChanged) {
5813 // If layout params have been changed, first give them
5814 // to the window manager to make sure it has the correct
5815 // animation info.
5816 try {
5817 if ((relayoutWindow(mWindowAttributes, viewVisibility, false)
Jeff Brown98365d72012-08-19 20:30:52 -07005818 & WindowManagerGlobal.RELAYOUT_RES_FIRST_TIME) != 0) {
5819 mWindowSession.finishDrawing(mWindow);
Romain Guyedbca122012-04-04 18:25:53 -07005820 }
5821 } catch (RemoteException e) {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08005822 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08005823 }
Craig Mautner8f303ad2013-06-14 11:32:22 -07005824
Romain Guyedbca122012-04-04 18:25:53 -07005825 mSurface.release();
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08005826 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08005827 }
Romain Guyedbca122012-04-04 18:25:53 -07005828
5829 mAdded = false;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08005830 }
Craig Mautner05eb7302013-06-03 17:24:21 -07005831 WindowManagerGlobal.getInstance().doRemoveView(this);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08005832 }
5833
Dianne Hackborn5fd21692011-06-07 14:09:47 -07005834 public void requestUpdateConfiguration(Configuration config) {
Jeff Browna175a5b2012-02-15 19:18:31 -08005835 Message msg = mHandler.obtainMessage(MSG_UPDATE_CONFIGURATION, config);
5836 mHandler.sendMessage(msg);
Dianne Hackborn5fd21692011-06-07 14:09:47 -07005837 }
5838
Dianne Hackborna53de062012-05-08 18:53:51 -07005839 public void loadSystemProperties() {
Romain Guy5bb3c732012-11-29 17:52:58 -08005840 mHandler.post(new Runnable() {
5841 @Override
5842 public void run() {
5843 // Profiling
5844 mProfileRendering = SystemProperties.getBoolean(PROPERTY_PROFILE_RENDERING, false);
5845 profileRendering(mAttachInfo.mHasWindowFocus);
5846
5847 // Hardware rendering
5848 if (mAttachInfo.mHardwareRenderer != null) {
John Reckcec24ae2013-11-05 13:27:50 -08005849 if (mAttachInfo.mHardwareRenderer.loadSystemProperties()) {
Romain Guy5bb3c732012-11-29 17:52:58 -08005850 invalidate();
5851 }
5852 }
5853
5854 // Layout debugging
5855 boolean layout = SystemProperties.getBoolean(View.DEBUG_LAYOUT_PROPERTY, false);
5856 if (layout != mAttachInfo.mDebugLayout) {
5857 mAttachInfo.mDebugLayout = layout;
5858 if (!mHandler.hasMessages(MSG_INVALIDATE_WORLD)) {
5859 mHandler.sendEmptyMessageDelayed(MSG_INVALIDATE_WORLD, 200);
5860 }
5861 }
Dianne Hackborna53de062012-05-08 18:53:51 -07005862 }
Romain Guy5bb3c732012-11-29 17:52:58 -08005863 });
Dianne Hackborna53de062012-05-08 18:53:51 -07005864 }
5865
Romain Guy29d89972010-09-22 16:10:57 -07005866 private void destroyHardwareRenderer() {
John Reck51aaf902015-12-02 15:08:07 -08005867 ThreadedRenderer hardwareRenderer = mAttachInfo.mHardwareRenderer;
Romain Guya998dff2012-03-23 18:58:36 -07005868
5869 if (hardwareRenderer != null) {
5870 if (mView != null) {
5871 hardwareRenderer.destroyHardwareResources(mView);
5872 }
John Reckf47a5942014-06-30 16:20:04 -07005873 hardwareRenderer.destroy();
Romain Guya998dff2012-03-23 18:58:36 -07005874 hardwareRenderer.setRequested(false);
5875
Chris Craikd36a81f2014-07-17 10:16:51 -07005876 mAttachInfo.mHardwareRenderer = null;
5877 mAttachInfo.mHardwareAccelerated = false;
Romain Guy29d89972010-09-22 16:10:57 -07005878 }
5879 }
5880
Yohei Yukawa1fa5f592016-03-24 10:24:23 -07005881 public void dispatchFinishInputConnection(InputConnection connection) {
5882 Message msg = mHandler.obtainMessage(MSG_FINISH_INPUT_CONNECTION, connection);
5883 mHandler.sendMessage(msg);
5884 }
5885
Dianne Hackbornc4aad012013-02-22 15:05:25 -08005886 public void dispatchResized(Rect frame, Rect overscanInsets, Rect contentInsets,
Filip Gruszczynski2217f612015-05-26 11:32:08 -07005887 Rect visibleInsets, Rect stableInsets, Rect outsets, boolean reportDraw,
Jorim Jaggi0ffd49c2016-02-12 15:04:21 -08005888 Configuration newConfig, Rect backDropFrame, boolean forceLayout,
5889 boolean alwaysConsumeNavBar) {
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08005890 if (DEBUG_LAYOUT) Log.v(mTag, "Resizing " + this + ": frame=" + frame.toShortString()
Svetoslav Ganov758143e2012-08-06 16:40:27 -07005891 + " contentInsets=" + contentInsets.toShortString()
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08005892 + " visibleInsets=" + visibleInsets.toShortString()
Chong Zhangd153c4f2015-11-06 20:26:40 -08005893 + " reportDraw=" + reportDraw
5894 + " backDropFrame=" + backDropFrame);
Chong Zhangdcee1de2015-10-06 10:26:00 -07005895
5896 // Tell all listeners that we are resizing the window so that the chrome can get
5897 // updated as fast as possible on a separate thread,
5898 if (mDragResizing) {
Jorim Jaggi9511b0f2016-01-29 19:12:44 -08005899 boolean fullscreen = frame.equals(backDropFrame);
Chong Zhangdcee1de2015-10-06 10:26:00 -07005900 synchronized (mWindowCallbacks) {
5901 for (int i = mWindowCallbacks.size() - 1; i >= 0; i--) {
Jorim Jaggi9511b0f2016-01-29 19:12:44 -08005902 mWindowCallbacks.get(i).onWindowSizeIsChanging(backDropFrame, fullscreen,
5903 visibleInsets, stableInsets);
Chong Zhangdcee1de2015-10-06 10:26:00 -07005904 }
5905 }
5906 }
5907
Svetoslav Ganov758143e2012-08-06 16:40:27 -07005908 Message msg = mHandler.obtainMessage(reportDraw ? MSG_RESIZED_REPORT : MSG_RESIZED);
Mitsuru Oshima64f59342009-06-21 00:03:11 -07005909 if (mTranslator != null) {
Svetoslav Ganov758143e2012-08-06 16:40:27 -07005910 mTranslator.translateRectInScreenToAppWindow(frame);
Dianne Hackbornc4aad012013-02-22 15:05:25 -08005911 mTranslator.translateRectInScreenToAppWindow(overscanInsets);
Dianne Hackborn3556c9a2012-05-04 16:31:25 -07005912 mTranslator.translateRectInScreenToAppWindow(contentInsets);
Mitsuru Oshima64f59342009-06-21 00:03:11 -07005913 mTranslator.translateRectInScreenToAppWindow(visibleInsets);
Mitsuru Oshima9189cab2009-06-03 11:19:12 -07005914 }
Svetoslav Ganov758143e2012-08-06 16:40:27 -07005915 SomeArgs args = SomeArgs.obtain();
5916 final boolean sameProcessCall = (Binder.getCallingPid() == android.os.Process.myPid());
5917 args.arg1 = sameProcessCall ? new Rect(frame) : frame;
5918 args.arg2 = sameProcessCall ? new Rect(contentInsets) : contentInsets;
5919 args.arg3 = sameProcessCall ? new Rect(visibleInsets) : visibleInsets;
5920 args.arg4 = sameProcessCall && newConfig != null ? new Configuration(newConfig) : newConfig;
Dianne Hackbornc4aad012013-02-22 15:05:25 -08005921 args.arg5 = sameProcessCall ? new Rect(overscanInsets) : overscanInsets;
Adrian Roosfa104232014-06-20 16:10:14 -07005922 args.arg6 = sameProcessCall ? new Rect(stableInsets) : stableInsets;
Filip Gruszczynski2217f612015-05-26 11:32:08 -07005923 args.arg7 = sameProcessCall ? new Rect(outsets) : outsets;
Jorim Jaggia7262a82015-11-03 15:15:40 +01005924 args.arg8 = sameProcessCall ? new Rect(backDropFrame) : backDropFrame;
Jorim Jaggia4a58ef2016-01-27 02:10:08 -08005925 args.argi1 = forceLayout ? 1 : 0;
Jorim Jaggi0ffd49c2016-02-12 15:04:21 -08005926 args.argi2 = alwaysConsumeNavBar ? 1 : 0;
Svetoslav Ganov758143e2012-08-06 16:40:27 -07005927 msg.obj = args;
Jeff Browna175a5b2012-02-15 19:18:31 -08005928 mHandler.sendMessage(msg);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08005929 }
Chet Haase1f4786b2011-11-02 10:51:52 -07005930
Craig Mautner5702d4d2012-06-30 14:10:16 -07005931 public void dispatchMoved(int newX, int newY) {
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08005932 if (DEBUG_LAYOUT) Log.v(mTag, "Window moved " + this + ": newX=" + newX + " newY=" + newY);
Craig Mautner5702d4d2012-06-30 14:10:16 -07005933 if (mTranslator != null) {
5934 PointF point = new PointF(newX, newY);
5935 mTranslator.translatePointInScreenToAppWindow(point);
5936 newX = (int) (point.x + 0.5);
5937 newY = (int) (point.y + 0.5);
5938 }
5939 Message msg = mHandler.obtainMessage(MSG_WINDOW_MOVED, newX, newY);
5940 mHandler.sendMessage(msg);
5941 }
5942
Jeff Brown4952dfd2011-11-30 19:23:22 -08005943 /**
5944 * Represents a pending input event that is waiting in a queue.
5945 *
5946 * Input events are processed in serial order by the timestamp specified by
Romain Guy211370f2012-02-01 16:10:55 -08005947 * {@link InputEvent#getEventTimeNano()}. In general, the input dispatcher delivers
Jeff Brown4952dfd2011-11-30 19:23:22 -08005948 * one input event to the application at a time and waits for the application
5949 * to finish handling it before delivering the next one.
5950 *
5951 * However, because the application or IME can synthesize and inject multiple
5952 * key events at a time without going through the input dispatcher, we end up
5953 * needing a queue on the application's side.
5954 */
5955 private static final class QueuedInputEvent {
Jeff Brownf9e989d2013-04-04 23:04:03 -07005956 public static final int FLAG_DELIVER_POST_IME = 1 << 0;
5957 public static final int FLAG_DEFERRED = 1 << 1;
5958 public static final int FLAG_FINISHED = 1 << 2;
5959 public static final int FLAG_FINISHED_HANDLED = 1 << 3;
5960 public static final int FLAG_RESYNTHESIZED = 1 << 4;
Michael Wright899d7052014-04-23 17:23:39 -07005961 public static final int FLAG_UNHANDLED = 1 << 5;
Jeff Brown4952dfd2011-11-30 19:23:22 -08005962
5963 public QueuedInputEvent mNext;
5964
5965 public InputEvent mEvent;
Jeff Brown32cbc38552011-12-01 14:01:49 -08005966 public InputEventReceiver mReceiver;
Jeff Brown4952dfd2011-11-30 19:23:22 -08005967 public int mFlags;
Jeff Brownf9e989d2013-04-04 23:04:03 -07005968
5969 public boolean shouldSkipIme() {
5970 if ((mFlags & FLAG_DELIVER_POST_IME) != 0) {
5971 return true;
5972 }
5973 return mEvent instanceof MotionEvent
5974 && mEvent.isFromSource(InputDevice.SOURCE_CLASS_POINTER);
5975 }
Michael Wright899d7052014-04-23 17:23:39 -07005976
5977 public boolean shouldSendToSynthesizer() {
5978 if ((mFlags & FLAG_UNHANDLED) != 0) {
5979 return true;
5980 }
5981
5982 return false;
5983 }
Michael Wright06a79252014-05-05 17:45:29 -07005984
5985 @Override
5986 public String toString() {
5987 StringBuilder sb = new StringBuilder("QueuedInputEvent{flags=");
5988 boolean hasPrevious = false;
5989 hasPrevious = flagToString("DELIVER_POST_IME", FLAG_DELIVER_POST_IME, hasPrevious, sb);
5990 hasPrevious = flagToString("DEFERRED", FLAG_DEFERRED, hasPrevious, sb);
5991 hasPrevious = flagToString("FINISHED", FLAG_FINISHED, hasPrevious, sb);
5992 hasPrevious = flagToString("FINISHED_HANDLED", FLAG_FINISHED_HANDLED, hasPrevious, sb);
5993 hasPrevious = flagToString("RESYNTHESIZED", FLAG_RESYNTHESIZED, hasPrevious, sb);
5994 hasPrevious = flagToString("UNHANDLED", FLAG_UNHANDLED, hasPrevious, sb);
5995 if (!hasPrevious) {
5996 sb.append("0");
5997 }
5998 sb.append(", hasNextQueuedEvent=" + (mEvent != null ? "true" : "false"));
5999 sb.append(", hasInputEventReceiver=" + (mReceiver != null ? "true" : "false"));
6000 sb.append(", mEvent=" + mEvent + "}");
6001 return sb.toString();
6002 }
6003
6004 private boolean flagToString(String name, int flag,
6005 boolean hasPrevious, StringBuilder sb) {
6006 if ((mFlags & flag) != 0) {
6007 if (hasPrevious) {
6008 sb.append("|");
6009 }
6010 sb.append(name);
6011 return true;
6012 }
6013 return hasPrevious;
6014 }
Jeff Brown4952dfd2011-11-30 19:23:22 -08006015 }
6016
6017 private QueuedInputEvent obtainQueuedInputEvent(InputEvent event,
Jeff Brown32cbc38552011-12-01 14:01:49 -08006018 InputEventReceiver receiver, int flags) {
Jeff Brown4952dfd2011-11-30 19:23:22 -08006019 QueuedInputEvent q = mQueuedInputEventPool;
6020 if (q != null) {
6021 mQueuedInputEventPoolSize -= 1;
6022 mQueuedInputEventPool = q.mNext;
6023 q.mNext = null;
6024 } else {
6025 q = new QueuedInputEvent();
Jeff Brown46b9ac02010-04-22 18:58:52 -07006026 }
6027
Jeff Brown4952dfd2011-11-30 19:23:22 -08006028 q.mEvent = event;
Jeff Brown32cbc38552011-12-01 14:01:49 -08006029 q.mReceiver = receiver;
Jeff Brown4952dfd2011-11-30 19:23:22 -08006030 q.mFlags = flags;
Jeff Brown4952dfd2011-11-30 19:23:22 -08006031 return q;
6032 }
6033
6034 private void recycleQueuedInputEvent(QueuedInputEvent q) {
6035 q.mEvent = null;
Jeff Brown32cbc38552011-12-01 14:01:49 -08006036 q.mReceiver = null;
Jeff Brown4952dfd2011-11-30 19:23:22 -08006037
6038 if (mQueuedInputEventPoolSize < MAX_QUEUED_INPUT_EVENT_POOL_SIZE) {
6039 mQueuedInputEventPoolSize += 1;
6040 q.mNext = mQueuedInputEventPool;
6041 mQueuedInputEventPool = q;
6042 }
6043 }
6044
Jeff Brownf9261d22012-02-03 13:49:15 -08006045 void enqueueInputEvent(InputEvent event) {
6046 enqueueInputEvent(event, null, 0, false);
6047 }
6048
Jeff Brown4952dfd2011-11-30 19:23:22 -08006049 void enqueueInputEvent(InputEvent event,
Jeff Brownf9261d22012-02-03 13:49:15 -08006050 InputEventReceiver receiver, int flags, boolean processImmediately) {
Michael Wright5bd69e62015-05-14 14:48:08 +01006051 adjustInputEventForCompatibility(event);
Jeff Brown32cbc38552011-12-01 14:01:49 -08006052 QueuedInputEvent q = obtainQueuedInputEvent(event, receiver, flags);
Jeff Brown4952dfd2011-11-30 19:23:22 -08006053
Jeff Brown4952dfd2011-11-30 19:23:22 -08006054 // Always enqueue the input event in order, regardless of its time stamp.
6055 // We do this because the application or the IME may inject key events
6056 // in response to touch events and we want to ensure that the injected keys
6057 // are processed in the order they were received and we cannot trust that
6058 // the time stamp of injected events are monotonic.
Michael Wrightc8a7e542013-03-20 17:58:33 -07006059 QueuedInputEvent last = mPendingInputEventTail;
Jeff Brown4952dfd2011-11-30 19:23:22 -08006060 if (last == null) {
Michael Wrightc8a7e542013-03-20 17:58:33 -07006061 mPendingInputEventHead = q;
6062 mPendingInputEventTail = q;
Jeff Brown4952dfd2011-11-30 19:23:22 -08006063 } else {
Jeff Brown4952dfd2011-11-30 19:23:22 -08006064 last.mNext = q;
Michael Wrightc8a7e542013-03-20 17:58:33 -07006065 mPendingInputEventTail = q;
Jeff Brown4952dfd2011-11-30 19:23:22 -08006066 }
Michael Wright95ae9422013-03-14 10:58:50 -07006067 mPendingInputEventCount += 1;
6068 Trace.traceCounter(Trace.TRACE_TAG_INPUT, mPendingInputEventQueueLengthCounterName,
6069 mPendingInputEventCount);
Jeff Brown4952dfd2011-11-30 19:23:22 -08006070
Jeff Brownf9261d22012-02-03 13:49:15 -08006071 if (processImmediately) {
6072 doProcessInputEvents();
6073 } else {
6074 scheduleProcessInputEvents();
6075 }
Jeff Brown4952dfd2011-11-30 19:23:22 -08006076 }
6077
6078 private void scheduleProcessInputEvents() {
Jeff Brown96e942d2011-11-30 19:55:01 -08006079 if (!mProcessInputEventsScheduled) {
6080 mProcessInputEventsScheduled = true;
Jeff Brownebb2d8d2012-03-23 17:14:34 -07006081 Message msg = mHandler.obtainMessage(MSG_PROCESS_INPUT_EVENTS);
6082 msg.setAsynchronous(true);
6083 mHandler.sendMessage(msg);
Jeff Brown4952dfd2011-11-30 19:23:22 -08006084 }
6085 }
6086
Jeff Brownebb2d8d2012-03-23 17:14:34 -07006087 void doProcessInputEvents() {
Jeff Brownf9e989d2013-04-04 23:04:03 -07006088 // Deliver all pending input events in the queue.
Michael Wrightc8a7e542013-03-20 17:58:33 -07006089 while (mPendingInputEventHead != null) {
6090 QueuedInputEvent q = mPendingInputEventHead;
6091 mPendingInputEventHead = q.mNext;
6092 if (mPendingInputEventHead == null) {
6093 mPendingInputEventTail = null;
6094 }
Jeff Brown4952dfd2011-11-30 19:23:22 -08006095 q.mNext = null;
Jeff Brown29c0ed22013-01-14 13:50:37 -08006096
Michael Wright95ae9422013-03-14 10:58:50 -07006097 mPendingInputEventCount -= 1;
6098 Trace.traceCounter(Trace.TRACE_TAG_INPUT, mPendingInputEventQueueLengthCounterName,
6099 mPendingInputEventCount);
6100
John Reckba6adf62015-02-19 14:36:50 -08006101 long eventTime = q.mEvent.getEventTimeNano();
6102 long oldestEventTime = eventTime;
6103 if (q.mEvent instanceof MotionEvent) {
6104 MotionEvent me = (MotionEvent)q.mEvent;
6105 if (me.getHistorySize() > 0) {
6106 oldestEventTime = me.getHistoricalEventTimeNano(0);
6107 }
6108 }
6109 mChoreographer.mFrameInfo.updateInputEventTime(eventTime, oldestEventTime);
6110
Jeff Brownf9e989d2013-04-04 23:04:03 -07006111 deliverInputEvent(q);
Jeff Brown4952dfd2011-11-30 19:23:22 -08006112 }
6113
6114 // We are done processing all input events that we can process right now
6115 // so we can clear the pending flag immediately.
Jeff Brown96e942d2011-11-30 19:55:01 -08006116 if (mProcessInputEventsScheduled) {
6117 mProcessInputEventsScheduled = false;
Jeff Browna175a5b2012-02-15 19:18:31 -08006118 mHandler.removeMessages(MSG_PROCESS_INPUT_EVENTS);
Jeff Brown4952dfd2011-11-30 19:23:22 -08006119 }
6120 }
6121
Jeff Brownf9e989d2013-04-04 23:04:03 -07006122 private void deliverInputEvent(QueuedInputEvent q) {
Michael Wrightd2c3adc2014-02-18 22:50:50 -08006123 Trace.asyncTraceBegin(Trace.TRACE_TAG_VIEW, "deliverInputEvent",
6124 q.mEvent.getSequenceNumber());
6125 if (mInputEventConsistencyVerifier != null) {
6126 mInputEventConsistencyVerifier.onInputEvent(q.mEvent, 0);
6127 }
Michael Wrightc8a7e542013-03-20 17:58:33 -07006128
Michael Wright899d7052014-04-23 17:23:39 -07006129 InputStage stage;
6130 if (q.shouldSendToSynthesizer()) {
6131 stage = mSyntheticInputStage;
6132 } else {
6133 stage = q.shouldSkipIme() ? mFirstPostImeInputStage : mFirstInputStage;
6134 }
6135
Michael Wrightd2c3adc2014-02-18 22:50:50 -08006136 if (stage != null) {
6137 stage.deliver(q);
6138 } else {
6139 finishInputEvent(q);
Michael Wrightbf020962013-03-28 17:27:50 -07006140 }
Michael Wrightbf020962013-03-28 17:27:50 -07006141 }
6142
Jeff Brownf9e989d2013-04-04 23:04:03 -07006143 private void finishInputEvent(QueuedInputEvent q) {
Michael Wrightd2c3adc2014-02-18 22:50:50 -08006144 Trace.asyncTraceEnd(Trace.TRACE_TAG_VIEW, "deliverInputEvent",
6145 q.mEvent.getSequenceNumber());
Michael Wright9d744c72014-02-18 21:27:42 -08006146
Jeff Brown32cbc38552011-12-01 14:01:49 -08006147 if (q.mReceiver != null) {
Jeff Brownf9e989d2013-04-04 23:04:03 -07006148 boolean handled = (q.mFlags & QueuedInputEvent.FLAG_FINISHED_HANDLED) != 0;
Jeff Brown32cbc38552011-12-01 14:01:49 -08006149 q.mReceiver.finishInputEvent(q.mEvent, handled);
Jeff Brown92cc2d82011-12-02 01:19:47 -08006150 } else {
6151 q.mEvent.recycleIfNeededAfterDispatch();
Jeff Brown4952dfd2011-11-30 19:23:22 -08006152 }
6153
6154 recycleQueuedInputEvent(q);
Jeff Brown29c0ed22013-01-14 13:50:37 -08006155 }
Jeff Brown4952dfd2011-11-30 19:23:22 -08006156
Michael Wright5bd69e62015-05-14 14:48:08 +01006157 private void adjustInputEventForCompatibility(InputEvent e) {
Dianne Hackborn0e3de6c2015-07-29 15:20:21 -07006158 if (mTargetSdkVersion < Build.VERSION_CODES.M && e instanceof MotionEvent) {
Michael Wright5bd69e62015-05-14 14:48:08 +01006159 MotionEvent motion = (MotionEvent) e;
6160 final int mask =
6161 MotionEvent.BUTTON_STYLUS_PRIMARY | MotionEvent.BUTTON_STYLUS_SECONDARY;
6162 final int buttonState = motion.getButtonState();
6163 final int compatButtonState = (buttonState & mask) >> 4;
6164 if (compatButtonState != 0) {
6165 motion.setButtonState(buttonState | compatButtonState);
6166 }
6167 }
6168 }
6169
Jeff Brownf9e989d2013-04-04 23:04:03 -07006170 static boolean isTerminalInputEvent(InputEvent event) {
Jeff Brown29c0ed22013-01-14 13:50:37 -08006171 if (event instanceof KeyEvent) {
6172 final KeyEvent keyEvent = (KeyEvent)event;
6173 return keyEvent.getAction() == KeyEvent.ACTION_UP;
6174 } else {
6175 final MotionEvent motionEvent = (MotionEvent)event;
6176 final int action = motionEvent.getAction();
6177 return action == MotionEvent.ACTION_UP
6178 || action == MotionEvent.ACTION_CANCEL
6179 || action == MotionEvent.ACTION_HOVER_EXIT;
Jeff Brown4952dfd2011-11-30 19:23:22 -08006180 }
6181 }
6182
Jeff Brownebb2d8d2012-03-23 17:14:34 -07006183 void scheduleConsumeBatchedInput() {
6184 if (!mConsumeBatchedInputScheduled) {
6185 mConsumeBatchedInputScheduled = true;
6186 mChoreographer.postCallback(Choreographer.CALLBACK_INPUT,
6187 mConsumedBatchedInputRunnable, null);
Jeff Brown4a06c802012-02-15 15:06:01 -08006188 }
6189 }
Jeff Brownebb2d8d2012-03-23 17:14:34 -07006190
6191 void unscheduleConsumeBatchedInput() {
6192 if (mConsumeBatchedInputScheduled) {
6193 mConsumeBatchedInputScheduled = false;
6194 mChoreographer.removeCallbacks(Choreographer.CALLBACK_INPUT,
6195 mConsumedBatchedInputRunnable, null);
6196 }
6197 }
6198
Michael Wright9d744c72014-02-18 21:27:42 -08006199 void scheduleConsumeBatchedInputImmediately() {
6200 if (!mConsumeBatchedInputImmediatelyScheduled) {
6201 unscheduleConsumeBatchedInput();
6202 mConsumeBatchedInputImmediatelyScheduled = true;
6203 mHandler.post(mConsumeBatchedInputImmediatelyRunnable);
6204 }
6205 }
6206
Jeff Brown771526c2012-04-27 15:13:25 -07006207 void doConsumeBatchedInput(long frameTimeNanos) {
Jeff Brownebb2d8d2012-03-23 17:14:34 -07006208 if (mConsumeBatchedInputScheduled) {
6209 mConsumeBatchedInputScheduled = false;
Jeff Brown330314c2012-04-27 02:20:22 -07006210 if (mInputEventReceiver != null) {
Michael Wright9d744c72014-02-18 21:27:42 -08006211 if (mInputEventReceiver.consumeBatchedInputEvents(frameTimeNanos)
6212 && frameTimeNanos != -1) {
Michael Wright62ce65d2013-10-25 14:50:36 -07006213 // If we consumed a batch here, we want to go ahead and schedule the
6214 // consumption of batched input events on the next frame. Otherwise, we would
6215 // wait until we have more input events pending and might get starved by other
Michael Wright9d744c72014-02-18 21:27:42 -08006216 // things occurring in the process. If the frame time is -1, however, then
6217 // we're in a non-batching mode, so there's no need to schedule this.
Michael Wright62ce65d2013-10-25 14:50:36 -07006218 scheduleConsumeBatchedInput();
6219 }
Jeff Brownebb2d8d2012-03-23 17:14:34 -07006220 }
Jeff Brown330314c2012-04-27 02:20:22 -07006221 doProcessInputEvents();
Jeff Brownebb2d8d2012-03-23 17:14:34 -07006222 }
6223 }
6224
6225 final class TraversalRunnable implements Runnable {
6226 @Override
6227 public void run() {
6228 doTraversal();
6229 }
6230 }
6231 final TraversalRunnable mTraversalRunnable = new TraversalRunnable();
Jeff Brown4a06c802012-02-15 15:06:01 -08006232
Jeff Brown32cbc38552011-12-01 14:01:49 -08006233 final class WindowInputEventReceiver extends InputEventReceiver {
6234 public WindowInputEventReceiver(InputChannel inputChannel, Looper looper) {
6235 super(inputChannel, looper);
Jeff Brown46b9ac02010-04-22 18:58:52 -07006236 }
Jeff Brown32cbc38552011-12-01 14:01:49 -08006237
6238 @Override
6239 public void onInputEvent(InputEvent event) {
Jeff Brownf9261d22012-02-03 13:49:15 -08006240 enqueueInputEvent(event, this, 0, true);
Jeff Brown32cbc38552011-12-01 14:01:49 -08006241 }
Jeff Brown072ec962012-02-07 14:46:57 -08006242
6243 @Override
6244 public void onBatchedInputEventPending() {
Michael Wright9d744c72014-02-18 21:27:42 -08006245 if (mUnbufferedInputDispatch) {
6246 super.onBatchedInputEventPending();
6247 } else {
6248 scheduleConsumeBatchedInput();
6249 }
Jeff Brownebb2d8d2012-03-23 17:14:34 -07006250 }
6251
6252 @Override
6253 public void dispose() {
6254 unscheduleConsumeBatchedInput();
6255 super.dispose();
Jeff Brown072ec962012-02-07 14:46:57 -08006256 }
Jeff Brown32cbc38552011-12-01 14:01:49 -08006257 }
6258 WindowInputEventReceiver mInputEventReceiver;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08006259
Jeff Brownebb2d8d2012-03-23 17:14:34 -07006260 final class ConsumeBatchedInputRunnable implements Runnable {
6261 @Override
6262 public void run() {
Jeff Brown771526c2012-04-27 15:13:25 -07006263 doConsumeBatchedInput(mChoreographer.getFrameTimeNanos());
Jeff Brownebb2d8d2012-03-23 17:14:34 -07006264 }
6265 }
6266 final ConsumeBatchedInputRunnable mConsumedBatchedInputRunnable =
6267 new ConsumeBatchedInputRunnable();
6268 boolean mConsumeBatchedInputScheduled;
6269
Michael Wright9d744c72014-02-18 21:27:42 -08006270 final class ConsumeBatchedInputImmediatelyRunnable implements Runnable {
6271 @Override
6272 public void run() {
6273 doConsumeBatchedInput(-1);
6274 }
6275 }
6276 final ConsumeBatchedInputImmediatelyRunnable mConsumeBatchedInputImmediatelyRunnable =
6277 new ConsumeBatchedInputImmediatelyRunnable();
6278 boolean mConsumeBatchedInputImmediatelyScheduled;
6279
Jeff Brown6cb7b462012-03-05 13:21:17 -08006280 final class InvalidateOnAnimationRunnable implements Runnable {
6281 private boolean mPosted;
Igor Murashkina86ab6402013-08-30 12:58:36 -07006282 private final ArrayList<View> mViews = new ArrayList<View>();
6283 private final ArrayList<AttachInfo.InvalidateInfo> mViewRects =
Jeff Brown6cb7b462012-03-05 13:21:17 -08006284 new ArrayList<AttachInfo.InvalidateInfo>();
6285 private View[] mTempViews;
6286 private AttachInfo.InvalidateInfo[] mTempViewRects;
6287
6288 public void addView(View view) {
6289 synchronized (this) {
6290 mViews.add(view);
6291 postIfNeededLocked();
6292 }
6293 }
6294
6295 public void addViewRect(AttachInfo.InvalidateInfo info) {
6296 synchronized (this) {
6297 mViewRects.add(info);
6298 postIfNeededLocked();
6299 }
6300 }
6301
6302 public void removeView(View view) {
6303 synchronized (this) {
6304 mViews.remove(view);
6305
6306 for (int i = mViewRects.size(); i-- > 0; ) {
6307 AttachInfo.InvalidateInfo info = mViewRects.get(i);
6308 if (info.target == view) {
6309 mViewRects.remove(i);
Svetoslav Ganovabae2a12012-11-27 16:59:37 -08006310 info.recycle();
Jeff Brown6cb7b462012-03-05 13:21:17 -08006311 }
6312 }
6313
6314 if (mPosted && mViews.isEmpty() && mViewRects.isEmpty()) {
Jeff Brownebb2d8d2012-03-23 17:14:34 -07006315 mChoreographer.removeCallbacks(Choreographer.CALLBACK_ANIMATION, this, null);
Jeff Brown6cb7b462012-03-05 13:21:17 -08006316 mPosted = false;
6317 }
6318 }
6319 }
6320
6321 @Override
6322 public void run() {
6323 final int viewCount;
6324 final int viewRectCount;
6325 synchronized (this) {
6326 mPosted = false;
6327
6328 viewCount = mViews.size();
6329 if (viewCount != 0) {
6330 mTempViews = mViews.toArray(mTempViews != null
6331 ? mTempViews : new View[viewCount]);
6332 mViews.clear();
6333 }
6334
6335 viewRectCount = mViewRects.size();
6336 if (viewRectCount != 0) {
6337 mTempViewRects = mViewRects.toArray(mTempViewRects != null
6338 ? mTempViewRects : new AttachInfo.InvalidateInfo[viewRectCount]);
6339 mViewRects.clear();
6340 }
6341 }
6342
6343 for (int i = 0; i < viewCount; i++) {
6344 mTempViews[i].invalidate();
Adam Powell3e3c4ee2012-05-16 17:34:21 -07006345 mTempViews[i] = null;
Jeff Brown6cb7b462012-03-05 13:21:17 -08006346 }
6347
6348 for (int i = 0; i < viewRectCount; i++) {
6349 final View.AttachInfo.InvalidateInfo info = mTempViewRects[i];
6350 info.target.invalidate(info.left, info.top, info.right, info.bottom);
Svetoslav Ganovabae2a12012-11-27 16:59:37 -08006351 info.recycle();
Jeff Brown6cb7b462012-03-05 13:21:17 -08006352 }
6353 }
6354
6355 private void postIfNeededLocked() {
6356 if (!mPosted) {
Jeff Brownebb2d8d2012-03-23 17:14:34 -07006357 mChoreographer.postCallback(Choreographer.CALLBACK_ANIMATION, this, null);
Jeff Brown6cb7b462012-03-05 13:21:17 -08006358 mPosted = true;
6359 }
6360 }
6361 }
6362 final InvalidateOnAnimationRunnable mInvalidateOnAnimationRunnable =
6363 new InvalidateOnAnimationRunnable();
6364
Jeff Browna175a5b2012-02-15 19:18:31 -08006365 public void dispatchInvalidateDelayed(View view, long delayMilliseconds) {
6366 Message msg = mHandler.obtainMessage(MSG_INVALIDATE, view);
6367 mHandler.sendMessageDelayed(msg, delayMilliseconds);
6368 }
6369
Jeff Browna175a5b2012-02-15 19:18:31 -08006370 public void dispatchInvalidateRectDelayed(AttachInfo.InvalidateInfo info,
6371 long delayMilliseconds) {
6372 final Message msg = mHandler.obtainMessage(MSG_INVALIDATE_RECT, info);
6373 mHandler.sendMessageDelayed(msg, delayMilliseconds);
6374 }
6375
Jeff Brown6cb7b462012-03-05 13:21:17 -08006376 public void dispatchInvalidateOnAnimation(View view) {
6377 mInvalidateOnAnimationRunnable.addView(view);
6378 }
6379
6380 public void dispatchInvalidateRectOnAnimation(AttachInfo.InvalidateInfo info) {
6381 mInvalidateOnAnimationRunnable.addViewRect(info);
6382 }
6383
6384 public void cancelInvalidate(View view) {
6385 mHandler.removeMessages(MSG_INVALIDATE, view);
6386 // fixme: might leak the AttachInfo.InvalidateInfo objects instead of returning
6387 // them to the pool
6388 mHandler.removeMessages(MSG_INVALIDATE_RECT, view);
6389 mInvalidateOnAnimationRunnable.removeView(view);
6390 }
6391
keunyoung30f420f2013-08-02 14:23:10 -07006392 public void dispatchInputEvent(InputEvent event) {
Jae Seo6a6059a2014-04-17 21:35:29 -07006393 dispatchInputEvent(event, null);
6394 }
6395
6396 public void dispatchInputEvent(InputEvent event, InputEventReceiver receiver) {
6397 SomeArgs args = SomeArgs.obtain();
6398 args.arg1 = event;
6399 args.arg2 = receiver;
6400 Message msg = mHandler.obtainMessage(MSG_DISPATCH_INPUT_EVENT, args);
Jeff Browne0dbd002012-02-15 19:34:58 -08006401 msg.setAsynchronous(true);
Jeff Browna175a5b2012-02-15 19:18:31 -08006402 mHandler.sendMessage(msg);
6403 }
6404
Michael Wright899d7052014-04-23 17:23:39 -07006405 public void synthesizeInputEvent(InputEvent event) {
6406 Message msg = mHandler.obtainMessage(MSG_SYNTHESIZE_INPUT_EVENT, event);
6407 msg.setAsynchronous(true);
6408 mHandler.sendMessage(msg);
6409 }
6410
Jeff Browna175a5b2012-02-15 19:18:31 -08006411 public void dispatchKeyFromIme(KeyEvent event) {
6412 Message msg = mHandler.obtainMessage(MSG_DISPATCH_KEY_FROM_IME, event);
Jeff Browne0dbd002012-02-15 19:34:58 -08006413 msg.setAsynchronous(true);
Jeff Browna175a5b2012-02-15 19:18:31 -08006414 mHandler.sendMessage(msg);
Jeff Browncb1404e2011-01-15 18:14:15 -08006415 }
6416
Michael Wright899d7052014-04-23 17:23:39 -07006417 /**
6418 * Reinject unhandled {@link InputEvent}s in order to synthesize fallbacks events.
6419 *
6420 * Note that it is the responsibility of the caller of this API to recycle the InputEvent it
6421 * passes in.
6422 */
Michael Wright3da28342014-04-22 17:00:11 -07006423 public void dispatchUnhandledInputEvent(InputEvent event) {
Michael Wright899d7052014-04-23 17:23:39 -07006424 if (event instanceof MotionEvent) {
6425 event = MotionEvent.obtain((MotionEvent) event);
Michael Wright3da28342014-04-22 17:00:11 -07006426 }
Michael Wright899d7052014-04-23 17:23:39 -07006427 synthesizeInputEvent(event);
Michael Wright3da28342014-04-22 17:00:11 -07006428 }
6429
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08006430 public void dispatchAppVisibility(boolean visible) {
Jeff Browna175a5b2012-02-15 19:18:31 -08006431 Message msg = mHandler.obtainMessage(MSG_DISPATCH_APP_VISIBILITY);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08006432 msg.arg1 = visible ? 1 : 0;
Jeff Browna175a5b2012-02-15 19:18:31 -08006433 mHandler.sendMessage(msg);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08006434 }
6435
6436 public void dispatchGetNewSurface() {
Jeff Browna175a5b2012-02-15 19:18:31 -08006437 Message msg = mHandler.obtainMessage(MSG_DISPATCH_GET_NEW_SURFACE);
6438 mHandler.sendMessage(msg);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08006439 }
6440
6441 public void windowFocusChanged(boolean hasFocus, boolean inTouchMode) {
6442 Message msg = Message.obtain();
Jeff Browna175a5b2012-02-15 19:18:31 -08006443 msg.what = MSG_WINDOW_FOCUS_CHANGED;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08006444 msg.arg1 = hasFocus ? 1 : 0;
6445 msg.arg2 = inTouchMode ? 1 : 0;
Jeff Browna175a5b2012-02-15 19:18:31 -08006446 mHandler.sendMessage(msg);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08006447 }
6448
Craig Mautner9c795042014-10-28 19:59:59 -07006449 public void dispatchWindowShown() {
6450 mHandler.sendEmptyMessage(MSG_DISPATCH_WINDOW_SHOWN);
6451 }
6452
Dianne Hackbornffa42482009-09-23 22:20:11 -07006453 public void dispatchCloseSystemDialogs(String reason) {
6454 Message msg = Message.obtain();
Jeff Browna175a5b2012-02-15 19:18:31 -08006455 msg.what = MSG_CLOSE_SYSTEM_DIALOGS;
Dianne Hackbornffa42482009-09-23 22:20:11 -07006456 msg.obj = reason;
Jeff Browna175a5b2012-02-15 19:18:31 -08006457 mHandler.sendMessage(msg);
Dianne Hackbornffa42482009-09-23 22:20:11 -07006458 }
Christopher Tatea53146c2010-09-07 11:57:52 -07006459
6460 public void dispatchDragEvent(DragEvent event) {
Chris Tate91e9bb32010-10-12 12:58:43 -07006461 final int what;
6462 if (event.getAction() == DragEvent.ACTION_DRAG_LOCATION) {
Jeff Browna175a5b2012-02-15 19:18:31 -08006463 what = MSG_DISPATCH_DRAG_LOCATION_EVENT;
6464 mHandler.removeMessages(what);
Chris Tate91e9bb32010-10-12 12:58:43 -07006465 } else {
Jeff Browna175a5b2012-02-15 19:18:31 -08006466 what = MSG_DISPATCH_DRAG_EVENT;
Chris Tate91e9bb32010-10-12 12:58:43 -07006467 }
Jeff Browna175a5b2012-02-15 19:18:31 -08006468 Message msg = mHandler.obtainMessage(what, event);
6469 mHandler.sendMessage(msg);
Christopher Tatea53146c2010-09-07 11:57:52 -07006470 }
6471
Vladislav Kaznacheevec6a4472016-01-22 12:21:52 -08006472 public void updatePointerIcon(float x, float y) {
6473 final int what = MSG_UPDATE_POINTER_ICON;
6474 mHandler.removeMessages(what);
6475 final long now = SystemClock.uptimeMillis();
6476 final MotionEvent event = MotionEvent.obtain(
6477 0, now, MotionEvent.ACTION_HOVER_MOVE, x, y, 0);
6478 Message msg = mHandler.obtainMessage(what, event);
6479 mHandler.sendMessage(msg);
6480 }
6481
Dianne Hackborn9a230e02011-10-06 11:51:27 -07006482 public void dispatchSystemUiVisibilityChanged(int seq, int globalVisibility,
6483 int localValue, int localChanges) {
6484 SystemUiVisibilityInfo args = new SystemUiVisibilityInfo();
6485 args.seq = seq;
6486 args.globalVisibility = globalVisibility;
6487 args.localValue = localValue;
6488 args.localChanges = localChanges;
Jeff Browna175a5b2012-02-15 19:18:31 -08006489 mHandler.sendMessage(mHandler.obtainMessage(MSG_DISPATCH_SYSTEM_UI_VISIBILITY, args));
6490 }
6491
6492 public void dispatchCheckFocus() {
6493 if (!mHandler.hasMessages(MSG_CHECK_FOCUS)) {
6494 // This will result in a call to checkFocus() below.
6495 mHandler.sendEmptyMessage(MSG_CHECK_FOCUS);
6496 }
Joe Onorato664644d2011-01-23 17:53:23 -08006497 }
6498
Clara Bayarrifcd7e802016-03-10 12:58:18 +00006499 public void dispatchRequestKeyboardShortcuts(IResultReceiver receiver, int deviceId) {
6500 mHandler.obtainMessage(
6501 MSG_REQUEST_KEYBOARD_SHORTCUTS, deviceId, 0, receiver).sendToTarget();
Clara Bayarri75e09792015-07-29 16:20:40 +01006502 }
6503
svetoslavganov75986cf2009-05-14 22:28:01 -07006504 /**
Svetoslav Ganoveeee4d22011-06-10 20:51:30 -07006505 * Post a callback to send a
6506 * {@link AccessibilityEvent#TYPE_WINDOW_CONTENT_CHANGED} event.
Svetoslav Ganova0156172011-06-26 17:55:44 -07006507 * This event is send at most once every
6508 * {@link ViewConfiguration#getSendRecurringAccessibilityEventsInterval()}.
Svetoslav Ganoveeee4d22011-06-10 20:51:30 -07006509 */
Alan Viverette77e9a282013-09-12 17:16:09 -07006510 private void postSendWindowContentChangedCallback(View source, int changeType) {
Svetoslav Ganova0156172011-06-26 17:55:44 -07006511 if (mSendWindowContentChangedAccessibilityEvent == null) {
6512 mSendWindowContentChangedAccessibilityEvent =
6513 new SendWindowContentChangedAccessibilityEvent();
Svetoslav Ganoveeee4d22011-06-10 20:51:30 -07006514 }
Alan Viverette77e9a282013-09-12 17:16:09 -07006515 mSendWindowContentChangedAccessibilityEvent.runOrPost(source, changeType);
Svetoslav Ganoveeee4d22011-06-10 20:51:30 -07006516 }
6517
6518 /**
6519 * Remove a posted callback to send a
6520 * {@link AccessibilityEvent#TYPE_WINDOW_CONTENT_CHANGED} event.
6521 */
6522 private void removeSendWindowContentChangedCallback() {
Svetoslav Ganova0156172011-06-26 17:55:44 -07006523 if (mSendWindowContentChangedAccessibilityEvent != null) {
Jeff Browna175a5b2012-02-15 19:18:31 -08006524 mHandler.removeCallbacks(mSendWindowContentChangedAccessibilityEvent);
Svetoslav Ganoveeee4d22011-06-10 20:51:30 -07006525 }
6526 }
6527
Igor Murashkina86ab6402013-08-30 12:58:36 -07006528 @Override
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08006529 public boolean showContextMenuForChild(View originalView) {
6530 return false;
6531 }
6532
Igor Murashkina86ab6402013-08-30 12:58:36 -07006533 @Override
Oren Blasberged391262015-09-01 12:12:51 -07006534 public boolean showContextMenuForChild(View originalView, float x, float y) {
6535 return false;
6536 }
6537
6538 @Override
Adam Powell6e346362010-07-23 10:18:23 -07006539 public ActionMode startActionModeForChild(View originalView, ActionMode.Callback callback) {
6540 return null;
6541 }
6542
Igor Murashkina86ab6402013-08-30 12:58:36 -07006543 @Override
Clara Bayarri4423d912015-03-02 19:42:48 +00006544 public ActionMode startActionModeForChild(
6545 View originalView, ActionMode.Callback callback, int type) {
6546 return null;
6547 }
6548
6549 @Override
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08006550 public void createContextMenu(ContextMenu menu) {
6551 }
6552
Igor Murashkina86ab6402013-08-30 12:58:36 -07006553 @Override
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08006554 public void childDrawableStateChanged(View child) {
6555 }
6556
Igor Murashkina86ab6402013-08-30 12:58:36 -07006557 @Override
Svetoslav Ganov736c2752011-04-22 18:30:36 -07006558 public boolean requestSendAccessibilityEvent(View child, AccessibilityEvent event) {
George Mount41725de2015-04-09 08:23:05 -07006559 if (mView == null || mStopped || mPausedForTransition) {
Svetoslav Ganov736c2752011-04-22 18:30:36 -07006560 return false;
6561 }
Svetoslav Ganov45a02e02012-06-17 15:07:29 -07006562 // Intercept accessibility focus events fired by virtual nodes to keep
6563 // track of accessibility focus position in such nodes.
Svetoslav Ganov791fd312012-05-14 15:12:30 -07006564 final int eventType = event.getEventType();
6565 switch (eventType) {
6566 case AccessibilityEvent.TYPE_VIEW_ACCESSIBILITY_FOCUSED: {
Svetoslav Ganov45a02e02012-06-17 15:07:29 -07006567 final long sourceNodeId = event.getSourceNodeId();
6568 final int accessibilityViewId = AccessibilityNodeInfo.getAccessibilityViewId(
6569 sourceNodeId);
6570 View source = mView.findViewByAccessibilityId(accessibilityViewId);
6571 if (source != null) {
6572 AccessibilityNodeProvider provider = source.getAccessibilityNodeProvider();
6573 if (provider != null) {
Svetoslavb3ba1d42014-09-26 15:20:40 -07006574 final int virtualNodeId = AccessibilityNodeInfo.getVirtualDescendantId(
6575 sourceNodeId);
6576 final AccessibilityNodeInfo node;
6577 if (virtualNodeId == AccessibilityNodeInfo.UNDEFINED_ITEM_ID) {
6578 node = provider.createAccessibilityNodeInfo(
6579 AccessibilityNodeProvider.HOST_VIEW_ID);
6580 } else {
6581 node = provider.createAccessibilityNodeInfo(virtualNodeId);
6582 }
Svetoslav Ganov45a02e02012-06-17 15:07:29 -07006583 setAccessibilityFocus(source, node);
6584 }
Svetoslav Ganov791fd312012-05-14 15:12:30 -07006585 }
Svetoslav Ganov791fd312012-05-14 15:12:30 -07006586 } break;
6587 case AccessibilityEvent.TYPE_VIEW_ACCESSIBILITY_FOCUS_CLEARED: {
Svetoslav Ganov45a02e02012-06-17 15:07:29 -07006588 final long sourceNodeId = event.getSourceNodeId();
6589 final int accessibilityViewId = AccessibilityNodeInfo.getAccessibilityViewId(
6590 sourceNodeId);
6591 View source = mView.findViewByAccessibilityId(accessibilityViewId);
6592 if (source != null) {
6593 AccessibilityNodeProvider provider = source.getAccessibilityNodeProvider();
6594 if (provider != null) {
6595 setAccessibilityFocus(null, null);
6596 }
Svetoslav Ganov791fd312012-05-14 15:12:30 -07006597 }
Svetoslav Ganov791fd312012-05-14 15:12:30 -07006598 } break;
Svetoslavf0c758b2014-09-03 17:47:37 -07006599
6600
6601 case AccessibilityEvent.TYPE_WINDOW_CONTENT_CHANGED: {
Alan Viverette34457f52015-03-25 13:09:20 -07006602 handleWindowContentChangedEvent(event);
Svetoslavf0c758b2014-09-03 17:47:37 -07006603 } break;
Svetoslav Ganov791fd312012-05-14 15:12:30 -07006604 }
Svetoslav Ganov8643aa02011-04-20 12:12:33 -07006605 mAccessibilityManager.sendAccessibilityEvent(event);
Svetoslav Ganov736c2752011-04-22 18:30:36 -07006606 return true;
6607 }
6608
Alan Viverette34457f52015-03-25 13:09:20 -07006609 /**
6610 * Updates the focused virtual view, when necessary, in response to a
6611 * content changed event.
6612 * <p>
6613 * This is necessary to get updated bounds after a position change.
6614 *
6615 * @param event an accessibility event of type
6616 * {@link AccessibilityEvent#TYPE_WINDOW_CONTENT_CHANGED}
6617 */
6618 private void handleWindowContentChangedEvent(AccessibilityEvent event) {
Alan Viverette25acc7e2015-05-19 11:32:08 -07006619 final View focusedHost = mAccessibilityFocusedHost;
6620 if (focusedHost == null || mAccessibilityFocusedVirtualView == null) {
6621 // No virtual view focused, nothing to do here.
Alan Viverette34457f52015-03-25 13:09:20 -07006622 return;
6623 }
6624
Alan Viverette25acc7e2015-05-19 11:32:08 -07006625 final AccessibilityNodeProvider provider = focusedHost.getAccessibilityNodeProvider();
Alan Viverette34457f52015-03-25 13:09:20 -07006626 if (provider == null) {
Alan Viverette25acc7e2015-05-19 11:32:08 -07006627 // Error state: virtual view with no provider. Clear focus.
6628 mAccessibilityFocusedHost = null;
6629 mAccessibilityFocusedVirtualView = null;
6630 focusedHost.clearAccessibilityFocusNoCallbacks();
Alan Viverette34457f52015-03-25 13:09:20 -07006631 return;
6632 }
6633
6634 // We only care about change types that may affect the bounds of the
6635 // focused virtual view.
6636 final int changes = event.getContentChangeTypes();
6637 if ((changes & AccessibilityEvent.CONTENT_CHANGE_TYPE_SUBTREE) == 0
6638 && changes != AccessibilityEvent.CONTENT_CHANGE_TYPE_UNDEFINED) {
6639 return;
6640 }
6641
6642 final long eventSourceNodeId = event.getSourceNodeId();
6643 final int changedViewId = AccessibilityNodeInfo.getAccessibilityViewId(eventSourceNodeId);
6644
6645 // Search up the tree for subtree containment.
6646 boolean hostInSubtree = false;
6647 View root = mAccessibilityFocusedHost;
6648 while (root != null && !hostInSubtree) {
6649 if (changedViewId == root.getAccessibilityViewId()) {
6650 hostInSubtree = true;
6651 } else {
6652 final ViewParent parent = root.getParent();
6653 if (parent instanceof View) {
6654 root = (View) parent;
6655 } else {
6656 root = null;
6657 }
6658 }
6659 }
6660
6661 // We care only about changes in subtrees containing the host view.
6662 if (!hostInSubtree) {
6663 return;
6664 }
6665
6666 final long focusedSourceNodeId = mAccessibilityFocusedVirtualView.getSourceNodeId();
6667 int focusedChildId = AccessibilityNodeInfo.getVirtualDescendantId(focusedSourceNodeId);
6668 if (focusedChildId == AccessibilityNodeInfo.UNDEFINED_ITEM_ID) {
6669 // TODO: Should we clear the focused virtual view?
6670 focusedChildId = AccessibilityNodeProvider.HOST_VIEW_ID;
6671 }
6672
6673 // Refresh the node for the focused virtual view.
Alan Viverettea7ea65e2015-05-15 11:30:21 -07006674 final Rect oldBounds = mTempRect;
6675 mAccessibilityFocusedVirtualView.getBoundsInScreen(oldBounds);
Alan Viverette34457f52015-03-25 13:09:20 -07006676 mAccessibilityFocusedVirtualView = provider.createAccessibilityNodeInfo(focusedChildId);
Alan Viverette25acc7e2015-05-19 11:32:08 -07006677 if (mAccessibilityFocusedVirtualView == null) {
6678 // Error state: The node no longer exists. Clear focus.
6679 mAccessibilityFocusedHost = null;
6680 focusedHost.clearAccessibilityFocusNoCallbacks();
6681
6682 // This will probably fail, but try to keep the provider's internal
6683 // state consistent by clearing focus.
6684 provider.performAction(focusedChildId,
6685 AccessibilityAction.ACTION_CLEAR_ACCESSIBILITY_FOCUS.getId(), null);
Alan Viverettea7ea65e2015-05-15 11:30:21 -07006686 invalidateRectOnScreen(oldBounds);
Alan Viverette25acc7e2015-05-19 11:32:08 -07006687 } else {
6688 // The node was refreshed, invalidate bounds if necessary.
6689 final Rect newBounds = mAccessibilityFocusedVirtualView.getBoundsInScreen();
6690 if (!oldBounds.equals(newBounds)) {
6691 oldBounds.union(newBounds);
6692 invalidateRectOnScreen(oldBounds);
6693 }
Alan Viverettea7ea65e2015-05-15 11:30:21 -07006694 }
Alan Viverette34457f52015-03-25 13:09:20 -07006695 }
6696
Svetoslav Ganov42138042012-03-20 11:51:39 -07006697 @Override
Alan Viverette77e9a282013-09-12 17:16:09 -07006698 public void notifySubtreeAccessibilityStateChanged(View child, View source, int changeType) {
6699 postSendWindowContentChangedCallback(source, changeType);
Svetoslav Ganov42138042012-03-20 11:51:39 -07006700 }
6701
Fabrice Di Meglio9dd4c5c2013-02-08 18:15:07 -08006702 @Override
6703 public boolean canResolveLayoutDirection() {
6704 return true;
6705 }
6706
6707 @Override
6708 public boolean isLayoutDirectionResolved() {
6709 return true;
6710 }
6711
6712 @Override
6713 public int getLayoutDirection() {
6714 return View.LAYOUT_DIRECTION_RESOLVED_DEFAULT;
6715 }
6716
6717 @Override
6718 public boolean canResolveTextDirection() {
6719 return true;
6720 }
6721
6722 @Override
6723 public boolean isTextDirectionResolved() {
6724 return true;
6725 }
6726
6727 @Override
6728 public int getTextDirection() {
6729 return View.TEXT_DIRECTION_RESOLVED_DEFAULT;
6730 }
6731
6732 @Override
6733 public boolean canResolveTextAlignment() {
6734 return true;
6735 }
6736
6737 @Override
6738 public boolean isTextAlignmentResolved() {
6739 return true;
6740 }
6741
6742 @Override
6743 public int getTextAlignment() {
6744 return View.TEXT_ALIGNMENT_RESOLVED_DEFAULT;
6745 }
6746
Svetoslav Ganov42138042012-03-20 11:51:39 -07006747 private View getCommonPredecessor(View first, View second) {
Chris Craikd36a81f2014-07-17 10:16:51 -07006748 if (mTempHashSet == null) {
6749 mTempHashSet = new HashSet<View>();
Svetoslav Ganov42138042012-03-20 11:51:39 -07006750 }
Chris Craikd36a81f2014-07-17 10:16:51 -07006751 HashSet<View> seen = mTempHashSet;
6752 seen.clear();
6753 View firstCurrent = first;
6754 while (firstCurrent != null) {
6755 seen.add(firstCurrent);
6756 ViewParent firstCurrentParent = firstCurrent.mParent;
6757 if (firstCurrentParent instanceof View) {
6758 firstCurrent = (View) firstCurrentParent;
6759 } else {
6760 firstCurrent = null;
6761 }
6762 }
6763 View secondCurrent = second;
6764 while (secondCurrent != null) {
6765 if (seen.contains(secondCurrent)) {
6766 seen.clear();
6767 return secondCurrent;
6768 }
6769 ViewParent secondCurrentParent = secondCurrent.mParent;
6770 if (secondCurrentParent instanceof View) {
6771 secondCurrent = (View) secondCurrentParent;
6772 } else {
6773 secondCurrent = null;
6774 }
6775 }
6776 seen.clear();
Svetoslav Ganov42138042012-03-20 11:51:39 -07006777 return null;
6778 }
6779
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08006780 void checkThread() {
6781 if (mThread != Thread.currentThread()) {
6782 throw new CalledFromWrongThreadException(
6783 "Only the original thread that created a view hierarchy can touch its views.");
6784 }
6785 }
6786
Igor Murashkina86ab6402013-08-30 12:58:36 -07006787 @Override
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08006788 public void requestDisallowInterceptTouchEvent(boolean disallowIntercept) {
Joe Onoratoc6cc0f82011-04-12 11:53:13 -07006789 // ViewAncestor never intercepts touch event, so this can be a no-op
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08006790 }
6791
Igor Murashkina86ab6402013-08-30 12:58:36 -07006792 @Override
Svetoslav Ganov1cf70bb2012-08-06 10:53:34 -07006793 public boolean requestChildRectangleOnScreen(View child, Rect rectangle, boolean immediate) {
Yigit Boyard62d5e92016-01-19 18:56:20 -08006794 if (rectangle == null) {
6795 return scrollToRectOrFocus(null, immediate);
6796 }
6797 rectangle.offset(child.getLeft() - child.getScrollX(),
6798 child.getTop() - child.getScrollY());
Svetoslav Ganov1cf70bb2012-08-06 10:53:34 -07006799 final boolean scrolled = scrollToRectOrFocus(rectangle, immediate);
Yigit Boyard62d5e92016-01-19 18:56:20 -08006800 mTempRect.set(rectangle);
6801 mTempRect.offset(0, -mCurScrollY);
6802 mTempRect.offset(mAttachInfo.mWindowLeft, mAttachInfo.mWindowTop);
6803 try {
6804 mWindowSession.onRectangleOnScreenRequested(mWindow, mTempRect);
6805 } catch (RemoteException re) {
6806 /* ignore */
Svetoslav Ganov1cf70bb2012-08-06 10:53:34 -07006807 }
6808 return scrolled;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08006809 }
Romain Guy8506ab42009-06-11 17:35:47 -07006810
Igor Murashkina86ab6402013-08-30 12:58:36 -07006811 @Override
Adam Powell539ee872012-02-03 19:00:49 -08006812 public void childHasTransientStateChanged(View child, boolean hasTransientState) {
6813 // Do nothing.
6814 }
6815
Adam Powell10ba2772014-04-15 09:46:51 -07006816 @Override
6817 public boolean onStartNestedScroll(View child, View target, int nestedScrollAxes) {
6818 return false;
6819 }
6820
6821 @Override
6822 public void onStopNestedScroll(View target) {
6823 }
6824
6825 @Override
6826 public void onNestedScrollAccepted(View child, View target, int nestedScrollAxes) {
6827 }
6828
6829 @Override
6830 public void onNestedScroll(View target, int dxConsumed, int dyConsumed,
6831 int dxUnconsumed, int dyUnconsumed) {
6832 }
6833
6834 @Override
6835 public void onNestedPreScroll(View target, int dx, int dy, int[] consumed) {
6836 }
6837
6838 @Override
Adam Powellb36e4f92014-05-01 10:23:33 -07006839 public boolean onNestedFling(View target, float velocityX, float velocityY, boolean consumed) {
Adam Powell10ba2772014-04-15 09:46:51 -07006840 return false;
6841 }
6842
Adam Powellb72be592014-07-16 21:41:31 -07006843 @Override
6844 public boolean onNestedPreFling(View target, float velocityX, float velocityY) {
6845 return false;
6846 }
6847
Adam Powellb6ab0982015-01-07 17:00:12 -08006848 @Override
6849 public boolean onNestedPrePerformAccessibilityAction(View target, int action, Bundle args) {
6850 return false;
6851 }
6852
Jorim Jaggib774e552015-08-24 14:52:45 -07006853 /**
6854 * Force the window to report its next draw.
6855 * <p>
6856 * This method is only supposed to be used to speed up the interaction from SystemUI and window
6857 * manager when waiting for the first frame to be drawn when turning on the screen. DO NOT USE
6858 * unless you fully understand this interaction.
6859 * @hide
6860 */
6861 public void setReportNextDraw() {
6862 mReportNextDraw = true;
6863 invalidate();
6864 }
6865
Craig Mautnerbc57cd12013-08-19 15:47:42 -07006866 void changeCanvasOpacity(boolean opaque) {
Filip Gruszczynski34dab0b2015-12-22 08:29:07 -08006867 Log.d(mTag, "changeCanvasOpacity: opaque=" + opaque);
John Reck63a06672014-05-07 13:45:54 -07006868 if (mAttachInfo.mHardwareRenderer != null) {
6869 mAttachInfo.mHardwareRenderer.setOpaque(opaque);
6870 }
Craig Mautnerbc57cd12013-08-19 15:47:42 -07006871 }
6872
Dianne Hackborndc8a7f62010-05-10 11:29:34 -07006873 class TakenSurfaceHolder extends BaseSurfaceHolder {
6874 @Override
6875 public boolean onAllowLockCanvas() {
6876 return mDrawingAllowed;
6877 }
6878
6879 @Override
6880 public void onRelayoutContainer() {
6881 // Not currently interesting -- from changing between fixed and layout size.
6882 }
6883
Igor Murashkina86ab6402013-08-30 12:58:36 -07006884 @Override
Dianne Hackborndc8a7f62010-05-10 11:29:34 -07006885 public void setFormat(int format) {
6886 ((RootViewSurfaceTaker)mView).setSurfaceFormat(format);
6887 }
6888
Igor Murashkina86ab6402013-08-30 12:58:36 -07006889 @Override
Dianne Hackborndc8a7f62010-05-10 11:29:34 -07006890 public void setType(int type) {
6891 ((RootViewSurfaceTaker)mView).setSurfaceType(type);
6892 }
Igor Murashkina86ab6402013-08-30 12:58:36 -07006893
Dianne Hackborndc8a7f62010-05-10 11:29:34 -07006894 @Override
6895 public void onUpdateSurface() {
6896 // We take care of format and type changes on our own.
6897 throw new IllegalStateException("Shouldn't be here");
6898 }
6899
Igor Murashkina86ab6402013-08-30 12:58:36 -07006900 @Override
Dianne Hackborndc8a7f62010-05-10 11:29:34 -07006901 public boolean isCreating() {
6902 return mIsCreating;
6903 }
6904
6905 @Override
6906 public void setFixedSize(int width, int height) {
6907 throw new UnsupportedOperationException(
6908 "Currently only support sizing from layout");
6909 }
Igor Murashkina86ab6402013-08-30 12:58:36 -07006910
6911 @Override
Dianne Hackborndc8a7f62010-05-10 11:29:34 -07006912 public void setKeepScreenOn(boolean screenOn) {
6913 ((RootViewSurfaceTaker)mView).setSurfaceKeepScreenOn(screenOn);
6914 }
6915 }
Romain Guy8506ab42009-06-11 17:35:47 -07006916
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08006917 static class W extends IWindow.Stub {
Dianne Hackborn6dd005b2011-07-18 13:22:50 -07006918 private final WeakReference<ViewRootImpl> mViewAncestor;
Jeff Brown98365d72012-08-19 20:30:52 -07006919 private final IWindowSession mWindowSession;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08006920
Dianne Hackborn6dd005b2011-07-18 13:22:50 -07006921 W(ViewRootImpl viewAncestor) {
6922 mViewAncestor = new WeakReference<ViewRootImpl>(viewAncestor);
Jeff Brown98365d72012-08-19 20:30:52 -07006923 mWindowSession = viewAncestor.mWindowSession;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08006924 }
6925
Igor Murashkina86ab6402013-08-30 12:58:36 -07006926 @Override
Dianne Hackbornc4aad012013-02-22 15:05:25 -08006927 public void resized(Rect frame, Rect overscanInsets, Rect contentInsets,
Filip Gruszczynski2217f612015-05-26 11:32:08 -07006928 Rect visibleInsets, Rect stableInsets, Rect outsets, boolean reportDraw,
Jorim Jaggi0ffd49c2016-02-12 15:04:21 -08006929 Configuration newConfig, Rect backDropFrame, boolean forceLayout,
6930 boolean alwaysConsumeNavBar) {
Dianne Hackborn6dd005b2011-07-18 13:22:50 -07006931 final ViewRootImpl viewAncestor = mViewAncestor.get();
Svetoslav Ganov8643aa02011-04-20 12:12:33 -07006932 if (viewAncestor != null) {
Dianne Hackbornc4aad012013-02-22 15:05:25 -08006933 viewAncestor.dispatchResized(frame, overscanInsets, contentInsets,
Jorim Jaggia4a58ef2016-01-27 02:10:08 -08006934 visibleInsets, stableInsets, outsets, reportDraw, newConfig, backDropFrame,
Jorim Jaggi0ffd49c2016-02-12 15:04:21 -08006935 forceLayout, alwaysConsumeNavBar);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08006936 }
6937 }
6938
Craig Mautner5702d4d2012-06-30 14:10:16 -07006939 @Override
6940 public void moved(int newX, int newY) {
6941 final ViewRootImpl viewAncestor = mViewAncestor.get();
6942 if (viewAncestor != null) {
6943 viewAncestor.dispatchMoved(newX, newY);
6944 }
6945 }
6946
Igor Murashkina86ab6402013-08-30 12:58:36 -07006947 @Override
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08006948 public void dispatchAppVisibility(boolean visible) {
Dianne Hackborn6dd005b2011-07-18 13:22:50 -07006949 final ViewRootImpl viewAncestor = mViewAncestor.get();
Svetoslav Ganov8643aa02011-04-20 12:12:33 -07006950 if (viewAncestor != null) {
6951 viewAncestor.dispatchAppVisibility(visible);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08006952 }
6953 }
6954
Igor Murashkina86ab6402013-08-30 12:58:36 -07006955 @Override
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08006956 public void dispatchGetNewSurface() {
Dianne Hackborn6dd005b2011-07-18 13:22:50 -07006957 final ViewRootImpl viewAncestor = mViewAncestor.get();
Svetoslav Ganov8643aa02011-04-20 12:12:33 -07006958 if (viewAncestor != null) {
6959 viewAncestor.dispatchGetNewSurface();
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08006960 }
6961 }
6962
Igor Murashkina86ab6402013-08-30 12:58:36 -07006963 @Override
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08006964 public void windowFocusChanged(boolean hasFocus, boolean inTouchMode) {
Dianne Hackborn6dd005b2011-07-18 13:22:50 -07006965 final ViewRootImpl viewAncestor = mViewAncestor.get();
Svetoslav Ganov8643aa02011-04-20 12:12:33 -07006966 if (viewAncestor != null) {
6967 viewAncestor.windowFocusChanged(hasFocus, inTouchMode);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08006968 }
6969 }
6970
6971 private static int checkCallingPermission(String permission) {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08006972 try {
6973 return ActivityManagerNative.getDefault().checkPermission(
6974 permission, Binder.getCallingPid(), Binder.getCallingUid());
6975 } catch (RemoteException e) {
6976 return PackageManager.PERMISSION_DENIED;
6977 }
6978 }
6979
Igor Murashkina86ab6402013-08-30 12:58:36 -07006980 @Override
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08006981 public void executeCommand(String command, String parameters, ParcelFileDescriptor out) {
Dianne Hackborn6dd005b2011-07-18 13:22:50 -07006982 final ViewRootImpl viewAncestor = mViewAncestor.get();
Svetoslav Ganov8643aa02011-04-20 12:12:33 -07006983 if (viewAncestor != null) {
6984 final View view = viewAncestor.mView;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08006985 if (view != null) {
6986 if (checkCallingPermission(Manifest.permission.DUMP) !=
6987 PackageManager.PERMISSION_GRANTED) {
6988 throw new SecurityException("Insufficient permissions to invoke"
6989 + " executeCommand() from pid=" + Binder.getCallingPid()
6990 + ", uid=" + Binder.getCallingUid());
6991 }
6992
6993 OutputStream clientStream = null;
6994 try {
6995 clientStream = new ParcelFileDescriptor.AutoCloseOutputStream(out);
6996 ViewDebug.dispatchCommand(view, command, parameters, clientStream);
6997 } catch (IOException e) {
6998 e.printStackTrace();
6999 } finally {
7000 if (clientStream != null) {
7001 try {
7002 clientStream.close();
7003 } catch (IOException e) {
7004 e.printStackTrace();
7005 }
7006 }
7007 }
7008 }
7009 }
7010 }
Igor Murashkina86ab6402013-08-30 12:58:36 -07007011
7012 @Override
Dianne Hackbornffa42482009-09-23 22:20:11 -07007013 public void closeSystemDialogs(String reason) {
Dianne Hackborn6dd005b2011-07-18 13:22:50 -07007014 final ViewRootImpl viewAncestor = mViewAncestor.get();
Svetoslav Ganov8643aa02011-04-20 12:12:33 -07007015 if (viewAncestor != null) {
7016 viewAncestor.dispatchCloseSystemDialogs(reason);
Dianne Hackbornffa42482009-09-23 22:20:11 -07007017 }
7018 }
Igor Murashkina86ab6402013-08-30 12:58:36 -07007019
7020 @Override
Marco Nelissenbf6956b2009-11-09 15:21:13 -08007021 public void dispatchWallpaperOffsets(float x, float y, float xStep, float yStep,
7022 boolean sync) {
Dianne Hackborn19382ac2009-09-11 21:13:37 -07007023 if (sync) {
7024 try {
Jeff Brown98365d72012-08-19 20:30:52 -07007025 mWindowSession.wallpaperOffsetsComplete(asBinder());
Dianne Hackborn19382ac2009-09-11 21:13:37 -07007026 } catch (RemoteException e) {
7027 }
7028 }
Dianne Hackborn72c82ab2009-08-11 21:13:54 -07007029 }
Svetoslav Ganov8643aa02011-04-20 12:12:33 -07007030
Igor Murashkina86ab6402013-08-30 12:58:36 -07007031 @Override
Dianne Hackborn75804932009-10-20 20:15:20 -07007032 public void dispatchWallpaperCommand(String action, int x, int y,
7033 int z, Bundle extras, boolean sync) {
7034 if (sync) {
7035 try {
Jeff Brown98365d72012-08-19 20:30:52 -07007036 mWindowSession.wallpaperCommandComplete(asBinder(), null);
Dianne Hackborn75804932009-10-20 20:15:20 -07007037 } catch (RemoteException e) {
7038 }
7039 }
7040 }
Christopher Tatea53146c2010-09-07 11:57:52 -07007041
7042 /* Drag/drop */
Igor Murashkina86ab6402013-08-30 12:58:36 -07007043 @Override
Christopher Tatea53146c2010-09-07 11:57:52 -07007044 public void dispatchDragEvent(DragEvent event) {
Dianne Hackborn6dd005b2011-07-18 13:22:50 -07007045 final ViewRootImpl viewAncestor = mViewAncestor.get();
Svetoslav Ganov8643aa02011-04-20 12:12:33 -07007046 if (viewAncestor != null) {
7047 viewAncestor.dispatchDragEvent(event);
Christopher Tatea53146c2010-09-07 11:57:52 -07007048 }
7049 }
Joe Onorato664644d2011-01-23 17:53:23 -08007050
Igor Murashkina86ab6402013-08-30 12:58:36 -07007051 @Override
Vladislav Kaznacheevec6a4472016-01-22 12:21:52 -08007052 public void updatePointerIcon(float x, float y) {
7053 final ViewRootImpl viewAncestor = mViewAncestor.get();
7054 if (viewAncestor != null) {
7055 viewAncestor.updatePointerIcon(x, y);
7056 }
7057 }
7058
7059 @Override
Dianne Hackborn9a230e02011-10-06 11:51:27 -07007060 public void dispatchSystemUiVisibilityChanged(int seq, int globalVisibility,
7061 int localValue, int localChanges) {
Dianne Hackborn6dd005b2011-07-18 13:22:50 -07007062 final ViewRootImpl viewAncestor = mViewAncestor.get();
Svetoslav Ganov8643aa02011-04-20 12:12:33 -07007063 if (viewAncestor != null) {
Dianne Hackborn9a230e02011-10-06 11:51:27 -07007064 viewAncestor.dispatchSystemUiVisibilityChanged(seq, globalVisibility,
7065 localValue, localChanges);
Joe Onorato664644d2011-01-23 17:53:23 -08007066 }
7067 }
Dianne Hackborn12d3a942012-04-27 14:16:30 -07007068
Igor Murashkina86ab6402013-08-30 12:58:36 -07007069 @Override
Craig Mautner9c795042014-10-28 19:59:59 -07007070 public void dispatchWindowShown() {
7071 final ViewRootImpl viewAncestor = mViewAncestor.get();
7072 if (viewAncestor != null) {
7073 viewAncestor.dispatchWindowShown();
7074 }
7075 }
Clara Bayarri75e09792015-07-29 16:20:40 +01007076
7077 @Override
Clara Bayarrifcd7e802016-03-10 12:58:18 +00007078 public void requestAppKeyboardShortcuts(IResultReceiver receiver, int deviceId) {
7079 ViewRootImpl viewAncestor = mViewAncestor.get();
7080 if (viewAncestor != null) {
7081 viewAncestor.dispatchRequestKeyboardShortcuts(receiver, deviceId);
7082 }
Clara Bayarri75e09792015-07-29 16:20:40 +01007083 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08007084 }
7085
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08007086 public static final class CalledFromWrongThreadException extends AndroidRuntimeException {
7087 public CalledFromWrongThreadException(String msg) {
7088 super(msg);
7089 }
7090 }
7091
Alan Viverettebea0c7da2015-09-01 16:00:20 -04007092 static HandlerActionQueue getRunQueue() {
7093 HandlerActionQueue rq = sRunQueues.get();
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08007094 if (rq != null) {
7095 return rq;
7096 }
Alan Viverettebea0c7da2015-09-01 16:00:20 -04007097 rq = new HandlerActionQueue();
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08007098 sRunQueues.set(rq);
7099 return rq;
7100 }
Romain Guy8506ab42009-06-11 17:35:47 -07007101
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08007102 /**
Skuhneb8160872015-09-22 09:51:39 -07007103 * Start a drag resizing which will inform all listeners that a window resize is taking place.
7104 */
Jorim Jaggi9511b0f2016-01-29 19:12:44 -08007105 private void startDragResizing(Rect initialBounds, boolean fullscreen, Rect systemInsets,
Jorim Jaggic39c7b02016-03-24 10:47:07 -07007106 Rect stableInsets, int resizeMode) {
Skuhneb8160872015-09-22 09:51:39 -07007107 if (!mDragResizing) {
7108 mDragResizing = true;
Chong Zhangdcee1de2015-10-06 10:26:00 -07007109 synchronized (mWindowCallbacks) {
7110 for (int i = mWindowCallbacks.size() - 1; i >= 0; i--) {
Jorim Jaggi9511b0f2016-01-29 19:12:44 -08007111 mWindowCallbacks.get(i).onWindowDragResizeStart(initialBounds, fullscreen,
Jorim Jaggic39c7b02016-03-24 10:47:07 -07007112 systemInsets, stableInsets, resizeMode);
Skuhneb8160872015-09-22 09:51:39 -07007113 }
7114 }
7115 mFullRedrawNeeded = true;
7116 }
7117 }
7118
7119 /**
7120 * End a drag resize which will inform all listeners that a window resize has ended.
7121 */
7122 private void endDragResizing() {
7123 if (mDragResizing) {
7124 mDragResizing = false;
Chong Zhangdcee1de2015-10-06 10:26:00 -07007125 synchronized (mWindowCallbacks) {
7126 for (int i = mWindowCallbacks.size() - 1; i >= 0; i--) {
7127 mWindowCallbacks.get(i).onWindowDragResizeEnd();
Skuhneb8160872015-09-22 09:51:39 -07007128 }
7129 }
7130 mFullRedrawNeeded = true;
7131 }
7132 }
7133
Chong Zhang8dbd9ad2015-10-09 10:06:11 -07007134 private boolean updateContentDrawBounds() {
7135 boolean updated = false;
7136 synchronized (mWindowCallbacks) {
7137 for (int i = mWindowCallbacks.size() - 1; i >= 0; i--) {
7138 updated |= mWindowCallbacks.get(i).onContentDrawn(
7139 mWindowAttributes.surfaceInsets.left,
7140 mWindowAttributes.surfaceInsets.top,
7141 mWidth, mHeight);
7142 }
7143 }
7144 return updated | (mDragResizing && mReportNextDraw);
7145 }
7146
7147 private void requestDrawWindow() {
7148 if (mReportNextDraw) {
7149 mWindowDrawCountDown = new CountDownLatch(mWindowCallbacks.size());
7150 }
7151 synchronized (mWindowCallbacks) {
7152 for (int i = mWindowCallbacks.size() - 1; i >= 0; i--) {
7153 mWindowCallbacks.get(i).onRequestDraw(mReportNextDraw);
7154 }
7155 }
7156 }
7157
Skuhneb8160872015-09-22 09:51:39 -07007158 /**
Jorim Jaggi4846ee32016-01-07 17:39:12 +01007159 * Tells this instance that its corresponding activity has just relaunched. In this case, we
7160 * need to force a relayout of the window to make sure we get the correct bounds from window
7161 * manager.
7162 */
7163 public void reportActivityRelaunched() {
7164 mActivityRelaunched = true;
7165 }
7166
7167 /**
Svetoslav Ganov8643aa02011-04-20 12:12:33 -07007168 * Class for managing the accessibility interaction connection
7169 * based on the global accessibility state.
7170 */
7171 final class AccessibilityInteractionConnectionManager
7172 implements AccessibilityStateChangeListener {
Igor Murashkina86ab6402013-08-30 12:58:36 -07007173 @Override
Svetoslav Ganov8643aa02011-04-20 12:12:33 -07007174 public void onAccessibilityStateChanged(boolean enabled) {
7175 if (enabled) {
7176 ensureConnection();
Chris Craikcce47eb2014-07-16 15:12:15 -07007177 if (mAttachInfo.mHasWindowFocus) {
Svetoslav Ganov0d04e242012-02-21 13:46:36 -08007178 mView.sendAccessibilityEvent(AccessibilityEvent.TYPE_WINDOW_STATE_CHANGED);
7179 View focusedView = mView.findFocus();
7180 if (focusedView != null && focusedView != mView) {
7181 focusedView.sendAccessibilityEvent(AccessibilityEvent.TYPE_VIEW_FOCUSED);
7182 }
7183 }
Svetoslav Ganov8643aa02011-04-20 12:12:33 -07007184 } else {
7185 ensureNoConnection();
Svetoslav Ganov005b83b2012-04-16 18:17:17 -07007186 mHandler.obtainMessage(MSG_CLEAR_ACCESSIBILITY_FOCUS_HOST).sendToTarget();
Svetoslav Ganov8643aa02011-04-20 12:12:33 -07007187 }
7188 }
7189
7190 public void ensureConnection() {
Chris Craikcce47eb2014-07-16 15:12:15 -07007191 final boolean registered =
Svetoslav8e3feb12014-02-24 13:46:47 -08007192 mAttachInfo.mAccessibilityWindowId != AccessibilityNodeInfo.UNDEFINED_ITEM_ID;
Chris Craikcce47eb2014-07-16 15:12:15 -07007193 if (!registered) {
7194 mAttachInfo.mAccessibilityWindowId =
Svetoslav Ganov0d04e242012-02-21 13:46:36 -08007195 mAccessibilityManager.addAccessibilityInteractionConnection(mWindow,
7196 new AccessibilityInteractionConnection(ViewRootImpl.this));
Svetoslav Ganov8643aa02011-04-20 12:12:33 -07007197 }
7198 }
7199
7200 public void ensureNoConnection() {
Svetoslav Ganov0d04e242012-02-21 13:46:36 -08007201 final boolean registered =
Svetoslav8e3feb12014-02-24 13:46:47 -08007202 mAttachInfo.mAccessibilityWindowId != AccessibilityNodeInfo.UNDEFINED_ITEM_ID;
Svetoslav Ganov8643aa02011-04-20 12:12:33 -07007203 if (registered) {
Svetoslav8e3feb12014-02-24 13:46:47 -08007204 mAttachInfo.mAccessibilityWindowId = AccessibilityNodeInfo.UNDEFINED_ITEM_ID;
Svetoslav Ganov8643aa02011-04-20 12:12:33 -07007205 mAccessibilityManager.removeAccessibilityInteractionConnection(mWindow);
7206 }
7207 }
7208 }
7209
Chris Craikcce47eb2014-07-16 15:12:15 -07007210 final class HighContrastTextManager implements HighTextContrastChangeListener {
7211 HighContrastTextManager() {
7212 mAttachInfo.mHighContrastText = mAccessibilityManager.isHighTextContrastEnabled();
7213 }
7214 @Override
7215 public void onHighTextContrastStateChanged(boolean enabled) {
7216 mAttachInfo.mHighContrastText = enabled;
7217
7218 // Destroy Displaylists so they can be recreated with high contrast recordings
7219 destroyHardwareResources();
Chris Craikd36a81f2014-07-17 10:16:51 -07007220
7221 // Schedule redraw, which will rerecord + redraw all text
7222 invalidate();
Chris Craikcce47eb2014-07-16 15:12:15 -07007223 }
7224 }
7225
Svetoslav Ganov8643aa02011-04-20 12:12:33 -07007226 /**
7227 * This class is an interface this ViewAncestor provides to the
7228 * AccessibilityManagerService to the latter can interact with
7229 * the view hierarchy in this ViewAncestor.
7230 */
Svetoslav Ganovaf5b4f42011-10-28 12:41:38 -07007231 static final class AccessibilityInteractionConnection
Svetoslav Ganov8643aa02011-04-20 12:12:33 -07007232 extends IAccessibilityInteractionConnection.Stub {
Svetoslav Ganov02107852011-10-03 17:06:56 -07007233 private final WeakReference<ViewRootImpl> mViewRootImpl;
Svetoslav Ganov8643aa02011-04-20 12:12:33 -07007234
Svetoslav Ganovf79f4762011-10-28 15:40:32 -07007235 AccessibilityInteractionConnection(ViewRootImpl viewRootImpl) {
7236 mViewRootImpl = new WeakReference<ViewRootImpl>(viewRootImpl);
Svetoslav Ganov8643aa02011-04-20 12:12:33 -07007237 }
7238
Svetoslav Ganov42138042012-03-20 11:51:39 -07007239 @Override
Svetoslav Ganov02107852011-10-03 17:06:56 -07007240 public void findAccessibilityNodeInfoByAccessibilityId(long accessibilityNodeId,
Svetoslav9ae9ed22014-09-02 16:36:35 -07007241 Region interactiveRegion, int interactionId,
7242 IAccessibilityInteractionConnectionCallback callback, int flags,
Svetoslav Ganov152e9bb2012-10-12 20:15:29 -07007243 int interrogatingPid, long interrogatingTid, MagnificationSpec spec) {
Svetoslav Ganov02107852011-10-03 17:06:56 -07007244 ViewRootImpl viewRootImpl = mViewRootImpl.get();
7245 if (viewRootImpl != null && viewRootImpl.mView != null) {
Svetoslav Ganovaf5b4f42011-10-28 12:41:38 -07007246 viewRootImpl.getAccessibilityInteractionController()
Svetoslav Ganov02107852011-10-03 17:06:56 -07007247 .findAccessibilityNodeInfoByAccessibilityIdClientThread(accessibilityNodeId,
Svetoslav9ae9ed22014-09-02 16:36:35 -07007248 interactiveRegion, interactionId, callback, flags, interrogatingPid,
7249 interrogatingTid, spec);
Svetoslav Ganov79311c42012-01-17 20:24:26 -08007250 } else {
7251 // We cannot make the call and notify the caller so it does not wait.
7252 try {
7253 callback.setFindAccessibilityNodeInfosResult(null, interactionId);
7254 } catch (RemoteException re) {
7255 /* best effort - ignore */
7256 }
Svetoslav Ganov601ad802011-06-09 14:47:38 -07007257 }
Svetoslav Ganov8643aa02011-04-20 12:12:33 -07007258 }
7259
Svetoslav Ganov42138042012-03-20 11:51:39 -07007260 @Override
Svetoslav Ganov02107852011-10-03 17:06:56 -07007261 public void performAccessibilityAction(long accessibilityNodeId, int action,
Svetoslav Ganovaa780c12012-04-19 23:01:39 -07007262 Bundle arguments, int interactionId,
7263 IAccessibilityInteractionConnectionCallback callback, int flags,
Svet Ganov7498efd2014-09-03 21:33:00 -07007264 int interrogatingPid, long interrogatingTid) {
Svetoslav Ganov02107852011-10-03 17:06:56 -07007265 ViewRootImpl viewRootImpl = mViewRootImpl.get();
7266 if (viewRootImpl != null && viewRootImpl.mView != null) {
Svetoslav Ganovaf5b4f42011-10-28 12:41:38 -07007267 viewRootImpl.getAccessibilityInteractionController()
Svetoslav Ganovaa780c12012-04-19 23:01:39 -07007268 .performAccessibilityActionClientThread(accessibilityNodeId, action, arguments,
Svet Ganov7498efd2014-09-03 21:33:00 -07007269 interactionId, callback, flags, interrogatingPid, interrogatingTid);
Svetoslav Ganov79311c42012-01-17 20:24:26 -08007270 } else {
Svetoslav Ganov42138042012-03-20 11:51:39 -07007271 // We cannot make the call and notify the caller so it does not wait.
Svetoslav Ganov79311c42012-01-17 20:24:26 -08007272 try {
7273 callback.setPerformAccessibilityActionResult(false, interactionId);
7274 } catch (RemoteException re) {
7275 /* best effort - ignore */
7276 }
Svetoslav Ganov8643aa02011-04-20 12:12:33 -07007277 }
Svetoslav Ganov8643aa02011-04-20 12:12:33 -07007278 }
7279
Svetoslav Ganov42138042012-03-20 11:51:39 -07007280 @Override
Svetoslav Ganov80943d82013-01-02 10:25:37 -08007281 public void findAccessibilityNodeInfosByViewId(long accessibilityNodeId,
Svetoslav9ae9ed22014-09-02 16:36:35 -07007282 String viewId, Region interactiveRegion, int interactionId,
Svetoslav Ganov80943d82013-01-02 10:25:37 -08007283 IAccessibilityInteractionConnectionCallback callback, int flags,
Svetoslav Ganov152e9bb2012-10-12 20:15:29 -07007284 int interrogatingPid, long interrogatingTid, MagnificationSpec spec) {
Svetoslav Ganov02107852011-10-03 17:06:56 -07007285 ViewRootImpl viewRootImpl = mViewRootImpl.get();
7286 if (viewRootImpl != null && viewRootImpl.mView != null) {
Svetoslav Ganovaf5b4f42011-10-28 12:41:38 -07007287 viewRootImpl.getAccessibilityInteractionController()
Svetoslav Ganov80943d82013-01-02 10:25:37 -08007288 .findAccessibilityNodeInfosByViewIdClientThread(accessibilityNodeId,
Svetoslav9ae9ed22014-09-02 16:36:35 -07007289 viewId, interactiveRegion, interactionId, callback, flags,
7290 interrogatingPid, interrogatingTid, spec);
Svetoslav Ganov79311c42012-01-17 20:24:26 -08007291 } else {
Svetoslav Ganov42138042012-03-20 11:51:39 -07007292 // We cannot make the call and notify the caller so it does not wait.
Svetoslav Ganov79311c42012-01-17 20:24:26 -08007293 try {
7294 callback.setFindAccessibilityNodeInfoResult(null, interactionId);
7295 } catch (RemoteException re) {
7296 /* best effort - ignore */
7297 }
7298 }
7299 }
7300
Svetoslav Ganov42138042012-03-20 11:51:39 -07007301 @Override
Svetoslav Ganov79311c42012-01-17 20:24:26 -08007302 public void findAccessibilityNodeInfosByText(long accessibilityNodeId, String text,
Svetoslav9ae9ed22014-09-02 16:36:35 -07007303 Region interactiveRegion, int interactionId,
7304 IAccessibilityInteractionConnectionCallback callback, int flags,
Svetoslav Ganov152e9bb2012-10-12 20:15:29 -07007305 int interrogatingPid, long interrogatingTid, MagnificationSpec spec) {
Svetoslav Ganov79311c42012-01-17 20:24:26 -08007306 ViewRootImpl viewRootImpl = mViewRootImpl.get();
7307 if (viewRootImpl != null && viewRootImpl.mView != null) {
7308 viewRootImpl.getAccessibilityInteractionController()
7309 .findAccessibilityNodeInfosByTextClientThread(accessibilityNodeId, text,
Svetoslav9ae9ed22014-09-02 16:36:35 -07007310 interactiveRegion, interactionId, callback, flags, interrogatingPid,
7311 interrogatingTid, spec);
Svetoslav Ganov79311c42012-01-17 20:24:26 -08007312 } else {
Svetoslav Ganov42138042012-03-20 11:51:39 -07007313 // We cannot make the call and notify the caller so it does not wait.
Svetoslav Ganov79311c42012-01-17 20:24:26 -08007314 try {
7315 callback.setFindAccessibilityNodeInfosResult(null, interactionId);
7316 } catch (RemoteException re) {
7317 /* best effort - ignore */
7318 }
Svetoslav Ganov8643aa02011-04-20 12:12:33 -07007319 }
Svetoslav Ganov8643aa02011-04-20 12:12:33 -07007320 }
Svetoslav Ganov8643aa02011-04-20 12:12:33 -07007321
Svetoslav Ganov42138042012-03-20 11:51:39 -07007322 @Override
Svetoslav9ae9ed22014-09-02 16:36:35 -07007323 public void findFocus(long accessibilityNodeId, int focusType, Region interactiveRegion,
7324 int interactionId, IAccessibilityInteractionConnectionCallback callback, int flags,
Svetoslav Ganov152e9bb2012-10-12 20:15:29 -07007325 int interrogatingPid, long interrogatingTid, MagnificationSpec spec) {
Svetoslav Ganov42138042012-03-20 11:51:39 -07007326 ViewRootImpl viewRootImpl = mViewRootImpl.get();
7327 if (viewRootImpl != null && viewRootImpl.mView != null) {
7328 viewRootImpl.getAccessibilityInteractionController()
Svetoslav9ae9ed22014-09-02 16:36:35 -07007329 .findFocusClientThread(accessibilityNodeId, focusType, interactiveRegion,
7330 interactionId, callback, flags, interrogatingPid, interrogatingTid,
7331 spec);
Svetoslav Ganov8bd69612011-08-23 13:40:30 -07007332 } else {
Svetoslav Ganov42138042012-03-20 11:51:39 -07007333 // We cannot make the call and notify the caller so it does not wait.
Svetoslav Ganov8643aa02011-04-20 12:12:33 -07007334 try {
Svetoslav Ganov42138042012-03-20 11:51:39 -07007335 callback.setFindAccessibilityNodeInfoResult(null, interactionId);
Svetoslav Ganov8643aa02011-04-20 12:12:33 -07007336 } catch (RemoteException re) {
Svetoslav Ganov42138042012-03-20 11:51:39 -07007337 /* best effort - ignore */
Svetoslav Ganov8643aa02011-04-20 12:12:33 -07007338 }
7339 }
7340 }
7341
Svetoslav Ganov42138042012-03-20 11:51:39 -07007342 @Override
Svetoslav9ae9ed22014-09-02 16:36:35 -07007343 public void focusSearch(long accessibilityNodeId, int direction, Region interactiveRegion,
7344 int interactionId, IAccessibilityInteractionConnectionCallback callback, int flags,
Svetoslav Ganov152e9bb2012-10-12 20:15:29 -07007345 int interrogatingPid, long interrogatingTid, MagnificationSpec spec) {
Svetoslav Ganov42138042012-03-20 11:51:39 -07007346 ViewRootImpl viewRootImpl = mViewRootImpl.get();
7347 if (viewRootImpl != null && viewRootImpl.mView != null) {
7348 viewRootImpl.getAccessibilityInteractionController()
Svetoslav9ae9ed22014-09-02 16:36:35 -07007349 .focusSearchClientThread(accessibilityNodeId, direction, interactiveRegion,
7350 interactionId, callback, flags, interrogatingPid, interrogatingTid,
7351 spec);
Svetoslav Ganov8bd69612011-08-23 13:40:30 -07007352 } else {
Svetoslav Ganov42138042012-03-20 11:51:39 -07007353 // We cannot make the call and notify the caller so it does not wait.
Svetoslav Ganov8643aa02011-04-20 12:12:33 -07007354 try {
Svetoslav Ganov42138042012-03-20 11:51:39 -07007355 callback.setFindAccessibilityNodeInfoResult(null, interactionId);
Svetoslav Ganov8643aa02011-04-20 12:12:33 -07007356 } catch (RemoteException re) {
Svetoslav Ganov42138042012-03-20 11:51:39 -07007357 /* best effort - ignore */
Svetoslav Ganov8643aa02011-04-20 12:12:33 -07007358 }
7359 }
7360 }
Svetoslav Ganov8643aa02011-04-20 12:12:33 -07007361 }
Svetoslav Ganoveeee4d22011-06-10 20:51:30 -07007362
Svetoslav Ganova0156172011-06-26 17:55:44 -07007363 private class SendWindowContentChangedAccessibilityEvent implements Runnable {
Alan Viverette77e9a282013-09-12 17:16:09 -07007364 private int mChangeTypes = 0;
7365
Svetoslav Ganov42138042012-03-20 11:51:39 -07007366 public View mSource;
Svetoslav6254f482013-06-04 17:22:14 -07007367 public long mLastEventTimeMillis;
Svetoslav Ganova0156172011-06-26 17:55:44 -07007368
Igor Murashkina86ab6402013-08-30 12:58:36 -07007369 @Override
Svetoslav Ganoveeee4d22011-06-10 20:51:30 -07007370 public void run() {
Svetoslav8d820ecf2013-07-15 17:12:41 -07007371 // The accessibility may be turned off while we were waiting so check again.
7372 if (AccessibilityManager.getInstance(mContext).isEnabled()) {
7373 mLastEventTimeMillis = SystemClock.uptimeMillis();
7374 AccessibilityEvent event = AccessibilityEvent.obtain();
7375 event.setEventType(AccessibilityEvent.TYPE_WINDOW_CONTENT_CHANGED);
Alan Viverette77e9a282013-09-12 17:16:09 -07007376 event.setContentChangeTypes(mChangeTypes);
Svetoslav8d820ecf2013-07-15 17:12:41 -07007377 mSource.sendAccessibilityEventUnchecked(event);
7378 } else {
7379 mLastEventTimeMillis = 0;
7380 }
7381 // In any case reset to initial state.
Svetoslav6254f482013-06-04 17:22:14 -07007382 mSource.resetSubtreeAccessibilityStateChanged();
7383 mSource = null;
Alan Viverette77e9a282013-09-12 17:16:09 -07007384 mChangeTypes = 0;
Svetoslav6254f482013-06-04 17:22:14 -07007385 }
7386
Alan Viverette77e9a282013-09-12 17:16:09 -07007387 public void runOrPost(View source, int changeType) {
Svetoslav Ganov42138042012-03-20 11:51:39 -07007388 if (mSource != null) {
Svetoslav9dafebb2013-06-18 16:29:25 -07007389 // If there is no common predecessor, then mSource points to
7390 // a removed view, hence in this case always prefer the source.
7391 View predecessor = getCommonPredecessor(mSource, source);
7392 mSource = (predecessor != null) ? predecessor : source;
Alan Viverette77e9a282013-09-12 17:16:09 -07007393 mChangeTypes |= changeType;
Svetoslav6254f482013-06-04 17:22:14 -07007394 return;
7395 }
7396 mSource = source;
Alan Viverette77e9a282013-09-12 17:16:09 -07007397 mChangeTypes = changeType;
Svetoslav6254f482013-06-04 17:22:14 -07007398 final long timeSinceLastMillis = SystemClock.uptimeMillis() - mLastEventTimeMillis;
7399 final long minEventIntevalMillis =
7400 ViewConfiguration.getSendRecurringAccessibilityEventsInterval();
7401 if (timeSinceLastMillis >= minEventIntevalMillis) {
Svetoslav9dafebb2013-06-18 16:29:25 -07007402 mSource.removeCallbacks(this);
Svetoslav6254f482013-06-04 17:22:14 -07007403 run();
7404 } else {
7405 mSource.postDelayed(this, minEventIntevalMillis - timeSinceLastMillis);
Svetoslav Ganov79311c42012-01-17 20:24:26 -08007406 }
7407 }
7408 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08007409}