/*
|
* Copyright (C) 2019 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.server.policy;
|
|
import android.content.Context;
|
import android.graphics.Rect;
|
import android.hardware.Sensor;
|
import android.hardware.SensorEvent;
|
import android.hardware.SensorEventListener;
|
import android.hardware.SensorManager;
|
import android.hardware.display.DisplayManagerInternal;
|
import android.os.Handler;
|
import android.os.RemoteCallbackList;
|
import android.os.RemoteException;
|
import android.view.DisplayInfo;
|
import android.view.IDisplayFoldListener;
|
|
import com.android.server.DisplayThread;
|
import com.android.server.LocalServices;
|
import com.android.server.wm.WindowManagerInternal;
|
|
/**
|
* Controls the behavior of foldable devices whose screen can literally bend and fold.
|
* TODO(b/126160895): Move DisplayFoldController from PhoneWindowManager to DisplayPolicy.
|
*/
|
class DisplayFoldController {
|
|
private static final String TAG = "DisplayFoldController";
|
|
private final WindowManagerInternal mWindowManagerInternal;
|
private final DisplayManagerInternal mDisplayManagerInternal;
|
private final int mDisplayId;
|
private final Handler mHandler;
|
|
/** The display area while device is folded. */
|
private final Rect mFoldedArea;
|
/** The display area to override the original folded area. */
|
private Rect mOverrideFoldedArea = new Rect();
|
|
private final DisplayInfo mNonOverrideDisplayInfo = new DisplayInfo();
|
private final RemoteCallbackList<IDisplayFoldListener> mListeners = new RemoteCallbackList<>();
|
private Boolean mFolded;
|
private String mFocusedApp;
|
private final DisplayFoldDurationLogger mDurationLogger = new DisplayFoldDurationLogger();
|
|
DisplayFoldController(WindowManagerInternal windowManagerInternal,
|
DisplayManagerInternal displayManagerInternal, int displayId, Rect foldedArea,
|
Handler handler) {
|
mWindowManagerInternal = windowManagerInternal;
|
mDisplayManagerInternal = displayManagerInternal;
|
mDisplayId = displayId;
|
mFoldedArea = new Rect(foldedArea);
|
mHandler = handler;
|
}
|
|
void finishedGoingToSleep() {
|
mDurationLogger.onFinishedGoingToSleep();
|
}
|
|
void finishedWakingUp() {
|
mDurationLogger.onFinishedWakingUp(mFolded);
|
}
|
|
void requestDeviceFolded(boolean folded) {
|
mHandler.post(() -> setDeviceFolded(folded));
|
}
|
|
void setDeviceFolded(boolean folded) {
|
if (mFolded != null && mFolded == folded) {
|
return;
|
}
|
if (folded) {
|
Rect foldedArea;
|
if (!mOverrideFoldedArea.isEmpty()) {
|
foldedArea = mOverrideFoldedArea;
|
} else if (!mFoldedArea.isEmpty()) {
|
foldedArea = mFoldedArea;
|
} else {
|
return;
|
}
|
|
mDisplayManagerInternal.getNonOverrideDisplayInfo(mDisplayId, mNonOverrideDisplayInfo);
|
final int dx = (mNonOverrideDisplayInfo.logicalWidth - foldedArea.width()) / 2
|
- foldedArea.left;
|
final int dy = (mNonOverrideDisplayInfo.logicalHeight - foldedArea.height()) / 2
|
- foldedArea.top;
|
|
// Bypass scaling otherwise LogicalDisplay will scale contents by default.
|
mDisplayManagerInternal.setDisplayScalingDisabled(mDisplayId, true);
|
mWindowManagerInternal.setForcedDisplaySize(mDisplayId,
|
foldedArea.width(), foldedArea.height());
|
mDisplayManagerInternal.setDisplayOffsets(mDisplayId, -dx, -dy);
|
} else {
|
mDisplayManagerInternal.setDisplayScalingDisabled(mDisplayId, false);
|
mWindowManagerInternal.clearForcedDisplaySize(mDisplayId);
|
mDisplayManagerInternal.setDisplayOffsets(mDisplayId, 0, 0);
|
}
|
mDurationLogger.setDeviceFolded(folded);
|
mDurationLogger.logFocusedAppWithFoldState(folded, mFocusedApp);
|
mFolded = folded;
|
|
final int n = mListeners.beginBroadcast();
|
for (int i = 0; i < n; i++) {
|
try {
|
mListeners.getBroadcastItem(i).onDisplayFoldChanged(mDisplayId, folded);
|
} catch (RemoteException e) {
|
// Listener died.
|
}
|
}
|
mListeners.finishBroadcast();
|
}
|
|
void registerDisplayFoldListener(IDisplayFoldListener listener) {
|
mListeners.register(listener);
|
if (mFolded == null) {
|
return;
|
}
|
mHandler.post(() -> {
|
try {
|
listener.onDisplayFoldChanged(mDisplayId, mFolded);
|
} catch (RemoteException e) {
|
// Listener died.
|
}
|
});
|
}
|
|
void unregisterDisplayFoldListener(IDisplayFoldListener listener) {
|
mListeners.unregister(listener);
|
}
|
|
void setOverrideFoldedArea(Rect area) {
|
mOverrideFoldedArea.set(area);
|
}
|
|
Rect getFoldedArea() {
|
if (!mOverrideFoldedArea.isEmpty()) {
|
return mOverrideFoldedArea;
|
} else {
|
return mFoldedArea;
|
}
|
}
|
|
/**
|
* Only used for the case that persist.debug.force_foldable is set.
|
* This is using proximity sensor to simulate the fold state switch.
|
*/
|
static DisplayFoldController createWithProxSensor(Context context, int displayId) {
|
final SensorManager sensorManager = context.getSystemService(SensorManager.class);
|
final Sensor proxSensor = sensorManager.getDefaultSensor(Sensor.TYPE_PROXIMITY);
|
if (proxSensor == null) {
|
return null;
|
}
|
|
final DisplayFoldController result = create(context, displayId);
|
sensorManager.registerListener(new SensorEventListener() {
|
@Override
|
public void onSensorChanged(SensorEvent event) {
|
result.requestDeviceFolded(event.values[0] < 1f);
|
}
|
|
@Override
|
public void onAccuracyChanged(Sensor sensor, int accuracy) {
|
// Ignore.
|
}
|
}, proxSensor, SensorManager.SENSOR_DELAY_NORMAL);
|
|
return result;
|
}
|
|
void onDefaultDisplayFocusChanged(String pkg) {
|
mFocusedApp = pkg;
|
}
|
|
static DisplayFoldController create(Context context, int displayId) {
|
final DisplayManagerInternal displayService =
|
LocalServices.getService(DisplayManagerInternal.class);
|
final String configFoldedArea = context.getResources().getString(
|
com.android.internal.R.string.config_foldedArea);
|
final Rect foldedArea;
|
if (configFoldedArea == null || configFoldedArea.isEmpty()) {
|
foldedArea = new Rect();
|
} else {
|
foldedArea = Rect.unflattenFromString(configFoldedArea);
|
}
|
|
return new DisplayFoldController(LocalServices.getService(WindowManagerInternal.class),
|
displayService, displayId, foldedArea, DisplayThread.getHandler());
|
}
|
}
|