Donate to e Foundation | Murena handsets with /e/OS | Own a part of Murena! Learn more

Commit 21aeaf4b authored by Jason Monk's avatar Jason Monk Committed by gitbuildkicker
Browse files

Fix leak in nav bar

Test: runtest systemui
Change-Id: Ib082ef216f1541911acbed84942ddfd5dd065a7f
Fixes: 37220220
(cherry picked from commit 91e587eb)
parent 2778583a
Loading
Loading
Loading
Loading
+4 −0
Original line number Original line Diff line number Diff line
@@ -42,6 +42,7 @@ import com.android.systemui.statusbar.phone.StatusBarWindowManager;
import com.android.systemui.statusbar.phone.StatusBarIconController;
import com.android.systemui.statusbar.phone.StatusBarIconController;
import com.android.systemui.statusbar.phone.StatusBarIconControllerImpl;
import com.android.systemui.statusbar.phone.StatusBarIconControllerImpl;
import com.android.systemui.statusbar.policy.AccessibilityController;
import com.android.systemui.statusbar.policy.AccessibilityController;
import com.android.systemui.statusbar.policy.AccessibilityManagerWrapper;
import com.android.systemui.statusbar.policy.BatteryController;
import com.android.systemui.statusbar.policy.BatteryController;
import com.android.systemui.statusbar.policy.BatteryControllerImpl;
import com.android.systemui.statusbar.policy.BatteryControllerImpl;
import com.android.systemui.statusbar.policy.BluetoothController;
import com.android.systemui.statusbar.policy.BluetoothController;
@@ -260,6 +261,9 @@ public class Dependency extends SystemUI {


        mProviders.put(MetricsLogger.class, () -> new MetricsLogger());
        mProviders.put(MetricsLogger.class, () -> new MetricsLogger());


        mProviders.put(AccessibilityManagerWrapper.class,
                () -> new AccessibilityManagerWrapper(mContext));

        // Put all dependencies above here so the factory can override them if it wants.
        // Put all dependencies above here so the factory can override them if it wants.
        SystemUIFactory.getInstance().injectDependencies(mProviders, mContext);
        SystemUIFactory.getInstance().injectDependencies(mProviders, mContext);
    }
    }
+9 −4
Original line number Original line Diff line number Diff line
@@ -64,6 +64,7 @@ import android.view.WindowManager.LayoutParams;
import android.view.WindowManagerGlobal;
import android.view.WindowManagerGlobal;
import android.view.accessibility.AccessibilityEvent;
import android.view.accessibility.AccessibilityEvent;
import android.view.accessibility.AccessibilityManager;
import android.view.accessibility.AccessibilityManager;
import android.view.accessibility.AccessibilityManager.AccessibilityServicesStateChangeListener;


import com.android.internal.logging.MetricsLogger;
import com.android.internal.logging.MetricsLogger;
import com.android.internal.logging.nano.MetricsProto.MetricsEvent;
import com.android.internal.logging.nano.MetricsProto.MetricsEvent;
@@ -78,6 +79,7 @@ import com.android.systemui.recents.Recents;
import com.android.systemui.stackdivider.Divider;
import com.android.systemui.stackdivider.Divider;
import com.android.systemui.statusbar.CommandQueue;
import com.android.systemui.statusbar.CommandQueue;
import com.android.systemui.statusbar.CommandQueue.Callbacks;
import com.android.systemui.statusbar.CommandQueue.Callbacks;
import com.android.systemui.statusbar.policy.AccessibilityManagerWrapper;
import com.android.systemui.statusbar.policy.KeyButtonView;
import com.android.systemui.statusbar.policy.KeyButtonView;
import com.android.systemui.statusbar.stack.StackStateAnimator;
import com.android.systemui.statusbar.stack.StackStateAnimator;


@@ -138,8 +140,8 @@ public class NavigationBarFragment extends Fragment implements Callbacks {
        mDivider = SysUiServiceProvider.getComponent(getContext(), Divider.class);
        mDivider = SysUiServiceProvider.getComponent(getContext(), Divider.class);
        mWindowManager = getContext().getSystemService(WindowManager.class);
        mWindowManager = getContext().getSystemService(WindowManager.class);
        mAccessibilityManager = getContext().getSystemService(AccessibilityManager.class);
        mAccessibilityManager = getContext().getSystemService(AccessibilityManager.class);
        mAccessibilityManager.addAccessibilityServicesStateChangeListener(
        Dependency.get(AccessibilityManagerWrapper.class).addCallback(
                this::updateAccessibilityServicesState);
                mAccessibilityListener);
        mContentResolver = getContext().getContentResolver();
        mContentResolver = getContext().getContentResolver();
        mMagnificationObserver = new MagnificationContentObserver(
        mMagnificationObserver = new MagnificationContentObserver(
                getContext().getMainThreadHandler());
                getContext().getMainThreadHandler());
@@ -164,8 +166,8 @@ public class NavigationBarFragment extends Fragment implements Callbacks {
    public void onDestroy() {
    public void onDestroy() {
        super.onDestroy();
        super.onDestroy();
        mCommandQueue.removeCallbacks(this);
        mCommandQueue.removeCallbacks(this);
        mAccessibilityManager.removeAccessibilityServicesStateChangeListener(
        Dependency.get(AccessibilityManagerWrapper.class).removeCallback(
                this::updateAccessibilityServicesState);
                mAccessibilityListener);
        mContentResolver.unregisterContentObserver(mMagnificationObserver);
        mContentResolver.unregisterContentObserver(mMagnificationObserver);
        try {
        try {
            WindowManagerGlobal.getWindowManagerService()
            WindowManagerGlobal.getWindowManagerService()
@@ -625,6 +627,9 @@ public class NavigationBarFragment extends Fragment implements Callbacks {
        mNavigationBarView.getBarTransitions().finishAnimations();
        mNavigationBarView.getBarTransitions().finishAnimations();
    }
    }


    private final AccessibilityServicesStateChangeListener mAccessibilityListener =
            this::updateAccessibilityServicesState;

    private class MagnificationContentObserver extends ContentObserver {
    private class MagnificationContentObserver extends ContentObserver {


        public MagnificationContentObserver(Handler handler) {
        public MagnificationContentObserver(Handler handler) {
+42 −0
Original line number Original line Diff line number Diff line
/*
 * Copyright (C) 2017 The Android Open Source Project
 *
 * Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file
 * except in compliance with the License. You may obtain a copy of the License at
 *
 *      http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software distributed under the
 * License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
 * KIND, either express or implied. See the License for the specific language governing
 * permissions and limitations under the License.
 */

package com.android.systemui.statusbar.policy;

import android.content.Context;
import android.view.accessibility.AccessibilityManager;
import android.view.accessibility.AccessibilityManager.AccessibilityServicesStateChangeListener;

/**
 * For mocking because AccessibilyManager is final for some reason...
 */
public class AccessibilityManagerWrapper implements
        CallbackController<AccessibilityServicesStateChangeListener> {

    private final AccessibilityManager mAccessibilityManager;

    public AccessibilityManagerWrapper(Context context) {
        mAccessibilityManager = context.getSystemService(AccessibilityManager.class);
    }

    @Override
    public void addCallback(AccessibilityServicesStateChangeListener listener) {
        mAccessibilityManager.addAccessibilityServicesStateChangeListener(listener);
    }

    @Override
    public void removeCallback(AccessibilityServicesStateChangeListener listener) {
        mAccessibilityManager.removeAccessibilityServicesStateChangeListener(listener);
    }
}
+23 −0
Original line number Original line Diff line number Diff line
@@ -14,12 +14,14 @@


package com.android.systemui.statusbar.phone;
package com.android.systemui.statusbar.phone;


import static org.mockito.ArgumentMatchers.any;
import static org.mockito.Mockito.mock;
import static org.mockito.Mockito.mock;
import static org.mockito.Mockito.when;
import static org.mockito.Mockito.when;


import android.content.Context;
import android.content.Context;
import android.os.Looper;
import android.os.Looper;
import android.testing.AndroidTestingRunner;
import android.testing.AndroidTestingRunner;
import android.testing.LeakCheck.Tracker;
import android.view.Display;
import android.view.Display;
import android.view.WindowManager;
import android.view.WindowManager;


@@ -29,11 +31,18 @@ import com.android.systemui.SysuiBaseFragmentTest;
import com.android.systemui.recents.Recents;
import com.android.systemui.recents.Recents;
import com.android.systemui.stackdivider.Divider;
import com.android.systemui.stackdivider.Divider;
import com.android.systemui.statusbar.CommandQueue;
import com.android.systemui.statusbar.CommandQueue;
import com.android.systemui.statusbar.policy.AccessibilityManagerWrapper;
import com.android.systemui.utils.leaks.BaseLeakChecker;

import android.testing.TestableLooper.RunWithLooper;
import android.testing.TestableLooper.RunWithLooper;
import android.view.accessibility.AccessibilityManager;
import android.view.accessibility.AccessibilityManager.AccessibilityServicesStateChangeListener;


import org.junit.Before;
import org.junit.Before;
import org.junit.Test;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.junit.runner.RunWith;
import org.mockito.invocation.InvocationOnMock;
import org.mockito.stubbing.Answer;


@RunWith(AndroidTestingRunner.class)
@RunWith(AndroidTestingRunner.class)
@RunWithLooper(setAsMainLooper = true)
@RunWithLooper(setAsMainLooper = true)
@@ -56,6 +65,20 @@ public class NavigationBarFragmentTest extends SysuiBaseFragmentTest {
        when(windowManager.getDefaultDisplay()).thenReturn(
        when(windowManager.getDefaultDisplay()).thenReturn(
                defaultDisplay);
                defaultDisplay);
        mContext.addMockSystemService(Context.WINDOW_SERVICE, windowManager);
        mContext.addMockSystemService(Context.WINDOW_SERVICE, windowManager);

        Tracker tracker = mLeakCheck.getTracker("accessibility_manager");
        AccessibilityManagerWrapper wrapper = new AccessibilityManagerWrapper(mContext) {
            @Override
            public void addCallback(AccessibilityServicesStateChangeListener listener) {
                tracker.getLeakInfo(listener).addAllocation(new Throwable());
            }

            @Override
            public void removeCallback(AccessibilityServicesStateChangeListener listener) {
                tracker.getLeakInfo(listener).clearAllocations();
            }
        };
        mDependency.injectTestDependency(AccessibilityManagerWrapper.class, wrapper);
    }
    }


    @Test
    @Test