blob: c0f245c10af6edfa02f0117c459837a20a839af7 [file] [log] [blame]
Jason Monk49fa0162017-01-11 09:21:56 -05001/*
2 * Copyright (C) 2017 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file
5 * except in compliance with the License. You may obtain a copy of the License at
6 *
7 * http://www.apache.org/licenses/LICENSE-2.0
8 *
9 * Unless required by applicable law or agreed to in writing, software distributed under the
10 * License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
11 * KIND, either express or implied. See the License for the specific language governing
12 * permissions and limitations under the License.
13 */
14
15package com.android.systemui.statusbar.phone;
16
17import static android.app.StatusBarManager.NAVIGATION_HINT_BACK_ALT;
18import static android.app.StatusBarManager.NAVIGATION_HINT_IME_SHOWN;
19import static android.app.StatusBarManager.WINDOW_STATE_SHOWING;
20import static android.app.StatusBarManager.windowStateToString;
21
22import static com.android.systemui.statusbar.phone.BarTransitions.MODE_SEMI_TRANSPARENT;
23import static com.android.systemui.statusbar.phone.PhoneStatusBar.DEBUG_WINDOW_STATE;
24import static com.android.systemui.statusbar.phone.PhoneStatusBar.dumpBarTransitions;
25
26import android.annotation.Nullable;
27import android.app.ActivityManager;
28import android.app.ActivityManagerNative;
29import android.app.Fragment;
30import android.app.IActivityManager;
31import android.app.StatusBarManager;
32import android.content.BroadcastReceiver;
33import android.content.Context;
34import android.content.Intent;
35import android.content.IntentFilter;
36import android.content.res.Configuration;
37import android.graphics.PixelFormat;
38import android.graphics.Rect;
39import android.inputmethodservice.InputMethodService;
40import android.os.Binder;
41import android.os.Bundle;
42import android.os.Handler;
43import android.os.IBinder;
44import android.os.Message;
45import android.os.PowerManager;
46import android.os.RemoteException;
47import android.os.UserHandle;
48import android.telecom.TelecomManager;
49import android.text.TextUtils;
50import android.util.Log;
51import android.view.IRotationWatcher.Stub;
52import android.view.KeyEvent;
53import android.view.LayoutInflater;
54import android.view.MotionEvent;
55import android.view.View;
56import android.view.ViewGroup;
57import android.view.WindowManager;
58import android.view.WindowManager.LayoutParams;
59import android.view.WindowManagerGlobal;
60import android.view.accessibility.AccessibilityEvent;
61import android.view.accessibility.AccessibilityManager;
62
63import com.android.internal.logging.MetricsLogger;
64import com.android.internal.logging.nano.MetricsProto.MetricsEvent;
65import com.android.keyguard.LatencyTracker;
66import com.android.systemui.R;
67import com.android.systemui.SystemUIApplication;
68import com.android.systemui.assist.AssistManager;
69import com.android.systemui.fragments.FragmentHostManager;
70import com.android.systemui.fragments.FragmentHostManager.FragmentListener;
71import com.android.systemui.recents.Recents;
72import com.android.systemui.stackdivider.Divider;
73import com.android.systemui.statusbar.CommandQueue;
74import com.android.systemui.statusbar.CommandQueue.Callbacks;
75import com.android.systemui.statusbar.policy.KeyButtonView;
76import com.android.systemui.statusbar.stack.StackStateAnimator;
77
78import java.io.FileDescriptor;
79import java.io.PrintWriter;
80import java.util.Locale;
81
82/**
83 * Fragment containing the NavigationBarFragment. Contains logic for what happens
84 * on clicks and view states of the nav bar.
85 */
86public class NavigationBarFragment extends Fragment implements Callbacks {
87
88 private static final String TAG = "NavigationBar";
89 private static final boolean DEBUG = false;
90 private static final String EXTRA_DISABLE_STATE = "disabled_state";
91
92 /** Allow some time inbetween the long press for back and recents. */
93 private static final int LOCK_TO_APP_GESTURE_TOLERENCE = 200;
94
95 protected NavigationBarView mNavigationBarView = null;
96 protected AssistManager mAssistManager;
97
98 private int mNavigationBarWindowState = WINDOW_STATE_SHOWING;
99
100 private int mNavigationIconHints = 0;
101 private int mNavigationBarMode;
102 protected AccessibilityManager mAccessibilityManager;
103
104 private int mDisabledFlags1;
105 private PhoneStatusBar mPhoneStatusBar;
106 private Recents mRecents;
107 private Divider mDivider;
108 private WindowManager mWindowManager;
109 private CommandQueue mCommandQueue;
110 private long mLastLockToAppLongPress;
111
112 private Locale mLocale;
113 private int mLayoutDirection;
114
115 private int mSystemUiVisibility;
116 private LightBarController mLightBarController;
117 private boolean mKeyguardGoingAway;
118
119 public boolean mHomeBlockedThisTouch;
120
121 // ----- Fragment Lifecycle Callbacks -----
122
123 @Override
124 public void onCreate(@Nullable Bundle savedInstanceState) {
125 super.onCreate(savedInstanceState);
126 mCommandQueue = SystemUIApplication.getComponent(getContext(), CommandQueue.class);
127 mCommandQueue.addCallbacks(this);
128 mPhoneStatusBar = SystemUIApplication.getComponent(getContext(), PhoneStatusBar.class);
129 mRecents = SystemUIApplication.getComponent(getContext(), Recents.class);
130 mDivider = SystemUIApplication.getComponent(getContext(), Divider.class);
131 mWindowManager = getContext().getSystemService(WindowManager.class);
132 mAccessibilityManager = getContext().getSystemService(AccessibilityManager.class);
133 if (savedInstanceState != null) {
134 mDisabledFlags1 = savedInstanceState.getInt(EXTRA_DISABLE_STATE, 0);
135 }
136
137 try {
138 WindowManagerGlobal.getWindowManagerService()
139 .watchRotation(mRotationWatcher);
140 } catch (RemoteException e) {
141 throw e.rethrowFromSystemServer();
142 }
143 }
144
145 @Override
146 public void onDestroy() {
147 super.onDestroy();
148 mCommandQueue.removeCallbacks(this);
149 try {
150 WindowManagerGlobal.getWindowManagerService()
151 .removeRotationWatcher(mRotationWatcher);
152 } catch (RemoteException e) {
153 throw e.rethrowFromSystemServer();
154 }
155 }
156
157 @Override
158 public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container,
159 Bundle savedInstanceState) {
160 return inflater.inflate(R.layout.navigation_bar, container, false);
161 }
162
163 @Override
164 public void onViewCreated(View view, @Nullable Bundle savedInstanceState) {
165 super.onViewCreated(view, savedInstanceState);
166 mNavigationBarView = (NavigationBarView) view;
167
168 mNavigationBarView.setDisabledFlags(mDisabledFlags1);
169 mNavigationBarView.setComponents(mRecents, mDivider);
170 mNavigationBarView.setOnVerticalChangedListener(this::onVerticalChanged);
171 mNavigationBarView.setOnTouchListener(this::onNavigationTouch);
172 if (savedInstanceState != null) {
173 mNavigationBarView.getLightTransitionsController().restoreState(savedInstanceState);
174 }
175
176 prepareNavigationBarView();
177 checkNavBarModes();
178
179 IntentFilter filter = new IntentFilter(Intent.ACTION_SCREEN_OFF);
180 filter.addAction(Intent.ACTION_SCREEN_ON);
181 getContext().registerReceiverAsUser(mBroadcastReceiver, UserHandle.ALL, filter, null, null);
182 PowerManager pm = getContext().getSystemService(PowerManager.class);
183 notifyNavigationBarScreenOn(pm.isScreenOn());
184 }
185
186 @Override
187 public void onDestroyView() {
188 super.onDestroyView();
189 getContext().unregisterReceiver(mBroadcastReceiver);
190 }
191
192 @Override
193 public void onSaveInstanceState(Bundle outState) {
194 super.onSaveInstanceState(outState);
195 outState.putInt(EXTRA_DISABLE_STATE, mDisabledFlags1);
196 if (mNavigationBarView != null) {
197 mNavigationBarView.getLightTransitionsController().saveState(outState);
198 }
199 }
200
201 @Override
202 public void onConfigurationChanged(Configuration newConfig) {
203 super.onConfigurationChanged(newConfig);
204 final Locale locale = getContext().getResources().getConfiguration().locale;
205 final int ld = TextUtils.getLayoutDirectionFromLocale(locale);
206 if (!locale.equals(mLocale) || ld != mLayoutDirection) {
207 if (DEBUG) {
208 Log.v(TAG, String.format(
209 "config changed locale/LD: %s (%d) -> %s (%d)", mLocale, mLayoutDirection,
210 locale, ld));
211 }
212 mLocale = locale;
213 mLayoutDirection = ld;
214 refreshLayout(ld);
215 }
216 repositionNavigationBar();
217 }
218
219 @Override
220 public void dump(String prefix, FileDescriptor fd, PrintWriter pw, String[] args) {
221 if (mNavigationBarView != null) {
222 pw.print(" mNavigationBarWindowState=");
223 pw.println(windowStateToString(mNavigationBarWindowState));
224 pw.print(" mNavigationBarMode=");
225 pw.println(BarTransitions.modeToString(mNavigationBarMode));
226 dumpBarTransitions(pw, "mNavigationBarView", mNavigationBarView.getBarTransitions());
227 }
228
229 pw.print(" mNavigationBarView=");
230 if (mNavigationBarView == null) {
231 pw.println("null");
232 } else {
233 mNavigationBarView.dump(fd, pw, args);
234 }
235 }
236
237 // ----- CommandQueue Callbacks -----
238
239 @Override
240 public void setImeWindowStatus(IBinder token, int vis, int backDisposition,
241 boolean showImeSwitcher) {
242 boolean imeShown = (vis & InputMethodService.IME_VISIBLE) != 0;
243 int hints = mNavigationIconHints;
244 if ((backDisposition == InputMethodService.BACK_DISPOSITION_WILL_DISMISS) || imeShown) {
245 hints |= NAVIGATION_HINT_BACK_ALT;
246 } else {
247 hints &= ~NAVIGATION_HINT_BACK_ALT;
248 }
249 if (showImeSwitcher) {
250 hints |= NAVIGATION_HINT_IME_SHOWN;
251 } else {
252 hints &= ~NAVIGATION_HINT_IME_SHOWN;
253 }
254 if (hints == mNavigationIconHints) return;
255
256 mNavigationIconHints = hints;
257
258 if (mNavigationBarView != null) {
259 mNavigationBarView.setNavigationIconHints(hints);
260 }
261 mPhoneStatusBar.checkBarModes();
262 }
263
264 @Override
265 public void topAppWindowChanged(boolean showMenu) {
266 if (mNavigationBarView != null) {
267 mNavigationBarView.setMenuVisibility(showMenu);
268 }
269 }
270
271 @Override
272 public void setWindowState(int window, int state) {
273 if (mNavigationBarView != null
274 && window == StatusBarManager.WINDOW_NAVIGATION_BAR
275 && mNavigationBarWindowState != state) {
276 mNavigationBarWindowState = state;
277 if (DEBUG_WINDOW_STATE) Log.d(TAG, "Navigation bar " + windowStateToString(state));
278 }
279 }
280
281 @Override
282 public void appTransitionPending() {
283 mNavigationBarView.getLightTransitionsController().appTransitionPending();
284 }
285
286 @Override
287 public void appTransitionCancelled() {
288 mNavigationBarView.getLightTransitionsController().appTransitionCancelled();
289 }
290
291 @Override
292 public void appTransitionStarting(long startTime, long duration) {
293 if (mKeyguardGoingAway) return;
294 doAppTransitionStarting(startTime, duration);
295 }
296
297 /**
298 * Calls appTransitionStarting for the nav bar regardless of whether keyguard is going away.
299 * public so PhoneStatusBar can force this when needed.
300 */
301 public void doAppTransitionStarting(long startTime, long duration) {
302 mNavigationBarView.getLightTransitionsController().appTransitionStarting(startTime,
303 duration);
304 }
305
306 // Injected from PhoneStatusBar at creation.
307 public void setCurrentSysuiVisibility(int systemUiVisibility) {
308 mSystemUiVisibility = systemUiVisibility;
309 mNavigationBarMode = mPhoneStatusBar.computeBarMode(0, mSystemUiVisibility,
310 View.NAVIGATION_BAR_TRANSIENT, View.NAVIGATION_BAR_TRANSLUCENT,
311 View.NAVIGATION_BAR_TRANSPARENT);
312 checkNavBarModes();
313 mPhoneStatusBar.touchAutoHide();
314 mLightBarController.onNavigationVisibilityChanged(mSystemUiVisibility, 0 /* mask */,
315 true /* nbModeChanged */, mNavigationBarMode);
316 }
317
318 @Override
319 public void setSystemUiVisibility(int vis, int fullscreenStackVis, int dockedStackVis,
320 int mask, Rect fullscreenStackBounds, Rect dockedStackBounds) {
321 final int oldVal = mSystemUiVisibility;
322 final int newVal = (oldVal & ~mask) | (vis & mask);
323 final int diff = newVal ^ oldVal;
324 boolean nbModeChanged = false;
325 if (diff != 0) {
326 mSystemUiVisibility = newVal;
327
328 // update navigation bar mode
329 final int nbMode = getView() == null
330 ? -1 : mPhoneStatusBar.computeBarMode(oldVal, newVal,
331 View.NAVIGATION_BAR_TRANSIENT, View.NAVIGATION_BAR_TRANSLUCENT,
332 View.NAVIGATION_BAR_TRANSPARENT);
333 nbModeChanged = nbMode != -1;
334 if (nbModeChanged) {
335 if (mNavigationBarMode != nbMode) {
336 mNavigationBarMode = nbMode;
337 checkNavBarModes();
338 }
339 mPhoneStatusBar.touchAutoHide();
340 }
341 }
342
343 mLightBarController.onNavigationVisibilityChanged(vis, mask, nbModeChanged,
344 mNavigationBarMode);
345 }
346
347 @Override
348 public void disable(int state1, int state2, boolean animate) {
349 // All navigation bar flags are in state1.
350 int masked = state1 & (StatusBarManager.DISABLE_HOME
351 | StatusBarManager.DISABLE_RECENT
352 | StatusBarManager.DISABLE_BACK
353 | StatusBarManager.DISABLE_SEARCH);
354 if (masked != mDisabledFlags1) {
355 mDisabledFlags1 = masked;
356 if (mNavigationBarView != null) mNavigationBarView.setDisabledFlags(state1);
357 }
358 }
359
360 // ----- Internal stuffz -----
361
362 private void refreshLayout(int layoutDirection) {
363 if (mNavigationBarView != null) {
364 mNavigationBarView.setLayoutDirection(layoutDirection);
365 }
366 }
367
368 private boolean shouldDisableNavbarGestures() {
369 return !mPhoneStatusBar.isDeviceProvisioned()
370 || (mDisabledFlags1 & StatusBarManager.DISABLE_SEARCH) != 0;
371 }
372
373 private void repositionNavigationBar() {
374 if (mNavigationBarView == null || !mNavigationBarView.isAttachedToWindow()) return;
375
376 prepareNavigationBarView();
377
378 mWindowManager.updateViewLayout((View) mNavigationBarView.getParent(),
379 ((View) mNavigationBarView.getParent()).getLayoutParams());
380 }
381
382 private void notifyNavigationBarScreenOn(boolean screenOn) {
383 mNavigationBarView.notifyScreenOn(screenOn);
384 }
385
386 private void prepareNavigationBarView() {
387 mNavigationBarView.reorient();
388
389 ButtonDispatcher recentsButton = mNavigationBarView.getRecentsButton();
390 recentsButton.setOnClickListener(this::onRecentsClick);
391 recentsButton.setOnTouchListener(this::onRecentsTouch);
392 recentsButton.setLongClickable(true);
393 recentsButton.setOnLongClickListener(this::onLongPressBackRecents);
394
395 ButtonDispatcher backButton = mNavigationBarView.getBackButton();
396 backButton.setLongClickable(true);
397 backButton.setOnLongClickListener(this::onLongPressBackRecents);
398
399 ButtonDispatcher homeButton = mNavigationBarView.getHomeButton();
400 homeButton.setOnTouchListener(this::onHomeTouch);
401 homeButton.setOnLongClickListener(this::onHomeLongClick);
402
403 if (mAssistManager != null) {
404 mAssistManager.onConfigurationChanged();
405 }
406 }
407
408 private boolean onHomeTouch(View v, MotionEvent event) {
409 if (mHomeBlockedThisTouch && event.getActionMasked() != MotionEvent.ACTION_DOWN) {
410 return true;
411 }
412 // If an incoming call is ringing, HOME is totally disabled.
413 // (The user is already on the InCallUI at this point,
414 // and his ONLY options are to answer or reject the call.)
415 switch (event.getAction()) {
416 case MotionEvent.ACTION_DOWN:
417 mHomeBlockedThisTouch = false;
418 TelecomManager telecomManager =
419 getContext().getSystemService(TelecomManager.class);
420 if (telecomManager != null && telecomManager.isRinging()) {
421 if (mPhoneStatusBar.isKeyguardShowing()) {
422 Log.i(TAG, "Ignoring HOME; there's a ringing incoming call. " +
423 "No heads up");
424 mHomeBlockedThisTouch = true;
425 return true;
426 }
427 }
428 break;
429 case MotionEvent.ACTION_UP:
430 case MotionEvent.ACTION_CANCEL:
431 mPhoneStatusBar.awakenDreams();
432 break;
433 }
434 return false;
435 }
436
437 private void onVerticalChanged(boolean isVertical) {
438 if (mAssistManager != null) {
439 // TODO: Clean this up.
440 mAssistManager.onConfigurationChanged();
441 }
442 mPhoneStatusBar.setQsScrimEnabled(!isVertical);
443 }
444
445 private boolean onNavigationTouch(View v, MotionEvent event) {
446 mPhoneStatusBar.checkUserAutohide(v, event);
447 return false;
448 }
449
450 private boolean onHomeLongClick(View v) {
451 if (shouldDisableNavbarGestures()) {
452 return false;
453 }
454 MetricsLogger.action(getContext(), MetricsEvent.ACTION_ASSIST_LONG_PRESS);
455 mAssistManager.startAssist(new Bundle() /* args */);
456 mPhoneStatusBar.awakenDreams();
457 if (mNavigationBarView != null) {
458 mNavigationBarView.abortCurrentGesture();
459 }
460 return true;
461 }
462
463 // additional optimization when we have software system buttons - start loading the recent
464 // tasks on touch down
465 private boolean onRecentsTouch(View v, MotionEvent event) {
466 int action = event.getAction() & MotionEvent.ACTION_MASK;
467 if (action == MotionEvent.ACTION_DOWN) {
468 mCommandQueue.preloadRecentApps();
469 } else if (action == MotionEvent.ACTION_CANCEL) {
470 mCommandQueue.cancelPreloadRecentApps();
471 } else if (action == MotionEvent.ACTION_UP) {
472 if (!v.isPressed()) {
473 mCommandQueue.cancelPreloadRecentApps();
474 }
475 }
476 return false;
477 }
478
479 private void onRecentsClick(View v) {
480 if (LatencyTracker.isEnabled(getContext())) {
481 LatencyTracker.getInstance(getContext()).onActionStart(
482 LatencyTracker.ACTION_TOGGLE_RECENTS);
483 }
484 mPhoneStatusBar.awakenDreams();
485 mCommandQueue.toggleRecentApps();
486 }
487
488 /**
489 * This handles long-press of both back and recents. They are
490 * handled together to capture them both being long-pressed
491 * at the same time to exit screen pinning (lock task).
492 *
493 * When accessibility mode is on, only a long-press from recents
494 * is required to exit.
495 *
496 * In all other circumstances we try to pass through long-press events
497 * for Back, so that apps can still use it. Which can be from two things.
498 * 1) Not currently in screen pinning (lock task).
499 * 2) Back is long-pressed without recents.
500 */
501 private boolean onLongPressBackRecents(View v) {
502 try {
503 boolean sendBackLongPress = false;
504 IActivityManager activityManager = ActivityManagerNative.getDefault();
505 boolean touchExplorationEnabled = mAccessibilityManager.isTouchExplorationEnabled();
506 boolean inLockTaskMode = activityManager.isInLockTaskMode();
507 if (inLockTaskMode && !touchExplorationEnabled) {
508 long time = System.currentTimeMillis();
509 // If we recently long-pressed the other button then they were
510 // long-pressed 'together'
511 if ((time - mLastLockToAppLongPress) < LOCK_TO_APP_GESTURE_TOLERENCE) {
512 activityManager.stopLockTaskMode();
513 // When exiting refresh disabled flags.
514 mNavigationBarView.setDisabledFlags(mDisabledFlags1, true);
515 return true;
516 } else if ((v.getId() == R.id.back)
517 && !mNavigationBarView.getRecentsButton().getCurrentView().isPressed()) {
518 // If we aren't pressing recents right now then they presses
519 // won't be together, so send the standard long-press action.
520 sendBackLongPress = true;
521 }
522 mLastLockToAppLongPress = time;
523 } else {
524 // If this is back still need to handle sending the long-press event.
525 if (v.getId() == R.id.back) {
526 sendBackLongPress = true;
527 } else if (touchExplorationEnabled && inLockTaskMode) {
528 // When in accessibility mode a long press that is recents (not back)
529 // should stop lock task.
530 activityManager.stopLockTaskMode();
531 // When exiting refresh disabled flags.
532 mNavigationBarView.setDisabledFlags(mDisabledFlags1, true);
533 return true;
534 } else if (v.getId() == R.id.recent_apps) {
535 return onLongPressRecents();
536 }
537 }
538 if (sendBackLongPress) {
539 KeyButtonView keyButtonView = (KeyButtonView) v;
540 keyButtonView.sendEvent(KeyEvent.ACTION_DOWN, KeyEvent.FLAG_LONG_PRESS);
541 keyButtonView.sendAccessibilityEvent(AccessibilityEvent.TYPE_VIEW_LONG_CLICKED);
542 return true;
543 }
544 } catch (RemoteException e) {
545 Log.d(TAG, "Unable to reach activity manager", e);
546 }
547 return false;
548 }
549
550 private boolean onLongPressRecents() {
551 if (mRecents == null || !ActivityManager.supportsMultiWindow()
552 || !mDivider.getView().getSnapAlgorithm()
553 .isSplitScreenFeasible()) {
554 return false;
555 }
556
557 return mPhoneStatusBar.toggleSplitScreenMode(MetricsEvent.ACTION_WINDOW_DOCK_LONGPRESS,
558 MetricsEvent.ACTION_WINDOW_UNDOCK_LONGPRESS);
559 }
560
561 // ----- Methods that PhoneStatusBar talks to (should be minimized) -----
562
563 public void setAssistManager(AssistManager assistManager) {
564 mAssistManager = assistManager;
565 }
566
567 public void setLightBarController(LightBarController lightBarController) {
568 mLightBarController = lightBarController;
569 mLightBarController.setNavigationBar(mNavigationBarView.getLightTransitionsController());
570 }
571
572 public boolean isSemiTransparent() {
573 return mNavigationBarMode == MODE_SEMI_TRANSPARENT;
574 }
575
576 public void onKeyguardOccludedChanged(boolean keyguardOccluded) {
577 mNavigationBarView.onKeyguardOccludedChanged(keyguardOccluded);
578 }
579
580 public void disableAnimationsDuringHide(long delay) {
581 mNavigationBarView.setLayoutTransitionsEnabled(false);
582 mNavigationBarView.postDelayed(() -> mNavigationBarView.setLayoutTransitionsEnabled(true),
583 delay + StackStateAnimator.ANIMATION_DURATION_GO_TO_FULL_SHADE);
584 }
585
586 public void setKeyguardGoingAway(boolean keyguardGoingAway) {
587 mKeyguardGoingAway = keyguardGoingAway;
588 }
589
590 public BarTransitions getBarTransitions() {
591 return mNavigationBarView.getBarTransitions();
592 }
593
594 public void checkNavBarModes() {
595 mPhoneStatusBar.checkBarMode(mNavigationBarMode,
596 mNavigationBarWindowState, mNavigationBarView.getBarTransitions());
597 }
598
599 public void finishBarAnimations() {
600 mNavigationBarView.getBarTransitions().finishAnimations();
601 }
602
603 private final Stub mRotationWatcher = new Stub() {
604 @Override
605 public void onRotationChanged(int rotation) throws RemoteException {
606 // We need this to be scheduled as early as possible to beat the redrawing of
607 // window in response to the orientation change.
608 Handler h = getView().getHandler();
609 Message msg = Message.obtain(h, () -> {
610 if (mNavigationBarView != null
611 && mNavigationBarView.needsReorient(rotation)) {
612 repositionNavigationBar();
613 }
614 });
615 msg.setAsynchronous(true);
616 h.sendMessageAtFrontOfQueue(msg);
617 }
618 };
619
620 private final BroadcastReceiver mBroadcastReceiver = new BroadcastReceiver() {
621 @Override
622 public void onReceive(Context context, Intent intent) {
623 String action = intent.getAction();
624 if (Intent.ACTION_SCREEN_OFF.equals(action)) {
625 notifyNavigationBarScreenOn(false);
626 } else if (Intent.ACTION_SCREEN_ON.equals(action)) {
627 notifyNavigationBarScreenOn(true);
628 }
629 }
630 };
631
632 public static View create(Context context, FragmentListener listener) {
633 WindowManager.LayoutParams lp = new WindowManager.LayoutParams(
634 LayoutParams.MATCH_PARENT, LayoutParams.MATCH_PARENT,
635 WindowManager.LayoutParams.TYPE_NAVIGATION_BAR,
636 WindowManager.LayoutParams.FLAG_TOUCHABLE_WHEN_WAKING
637 | WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE
638 | WindowManager.LayoutParams.FLAG_NOT_TOUCH_MODAL
639 | WindowManager.LayoutParams.FLAG_WATCH_OUTSIDE_TOUCH
640 | WindowManager.LayoutParams.FLAG_SPLIT_TOUCH
641 | WindowManager.LayoutParams.FLAG_SLIPPERY,
642 PixelFormat.TRANSLUCENT);
643 lp.token = new Binder();
644 // this will allow the navbar to run in an overlay on devices that support this
645 if (ActivityManager.isHighEndGfx()) {
646 lp.flags |= WindowManager.LayoutParams.FLAG_HARDWARE_ACCELERATED;
647 }
648 lp.setTitle("NavigationBar");
649 lp.windowAnimations = 0;
650
651 View navigationBarView = LayoutInflater.from(context).inflate(
652 R.layout.navigation_bar_window, null);
653
654 if (DEBUG) Log.v(TAG, "addNavigationBar: about to add " + navigationBarView);
655 if (navigationBarView == null) return null;
656
657 context.getSystemService(WindowManager.class).addView(navigationBarView, lp);
658 FragmentHostManager fragmentHost = FragmentHostManager.get(navigationBarView);
659 NavigationBarFragment fragment = new NavigationBarFragment();
660 fragmentHost.getFragmentManager().beginTransaction()
661 .replace(R.id.navigation_bar_frame, fragment, TAG)
662 .commit();
663 fragmentHost.addTagListener(TAG, listener);
664 return navigationBarView;
665 }
666}