1 /*
2  * Copyright (C) 2020 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 package com.android.systemui.accessibility;
18 
19 import static android.view.WindowManager.LayoutParams;
20 
21 import android.annotation.NonNull;
22 import android.annotation.Nullable;
23 import android.content.Context;
24 import android.graphics.PixelFormat;
25 import android.graphics.Point;
26 import android.graphics.Rect;
27 import android.util.Log;
28 import android.util.MathUtils;
29 import android.view.Gravity;
30 import android.view.LayoutInflater;
31 import android.view.View;
32 import android.view.WindowManager;
33 
34 import com.android.systemui.res.R;
35 
36 /**
37  * Contains a movable control UI to manipulate mirrored window's position, size and scale. The
38  * window type of the UI is {@link LayoutParams#TYPE_ACCESSIBILITY_MAGNIFICATION_OVERLAY} to
39  * ensure it won't be magnified. It is not movable to the navigation bar.
40  */
41 public abstract class MirrorWindowControl {
42     private static final String TAG = "MirrorWindowControl";
43     private static final boolean DBG = Log.isLoggable(TAG, Log.DEBUG) | false;
44 
45     /**
46      * A delegate handling a mirrored window's offset.
47      */
48     public interface MirrorWindowDelegate {
49         /**
50          * Moves the window with specified offset.
51          *
52          * @param xOffset the amount in pixels to offset the window in the X coordinate, in current
53          *                display pixels.
54          * @param yOffset the amount in pixels to offset the window in the Y coordinate, in current
55          *                display pixels.
56          */
move(int xOffset, int yOffset)57         void move(int xOffset, int yOffset);
58     }
59 
60     protected final Context mContext;
61     private final Rect mDraggableBound = new Rect();
62     final Point mTmpPoint = new Point();
63 
64     @Nullable
65     protected MirrorWindowDelegate mMirrorWindowDelegate;
66     protected View mControlsView;
67     /**
68      * The left top position of the control UI. Initialized when the control UI is visible.
69      *
70      * @see #setDefaultPosition(LayoutParams)
71      */
72     private final Point mControlPosition = new Point();
73     private final WindowManager mWindowManager;
74 
MirrorWindowControl(Context context)75     MirrorWindowControl(Context context) {
76         mContext = context;
77         mWindowManager = (WindowManager) mContext.getSystemService(Context.WINDOW_SERVICE);
78     }
79 
setWindowDelegate(@ullable MirrorWindowDelegate windowDelegate)80     public void setWindowDelegate(@Nullable MirrorWindowDelegate windowDelegate) {
81         mMirrorWindowDelegate = windowDelegate;
82     }
83 
84     /**
85      * Shows the control UI.
86      *
87      * {@link LayoutParams#TYPE_ACCESSIBILITY_MAGNIFICATION_OVERLAY}  window.
88      */
showControl()89     public final void showControl() {
90         if (mControlsView != null) {
91             Log.w(TAG, "control view is visible");
92             return;
93         }
94         final Point  viewSize = mTmpPoint;
95         mControlsView = onCreateView(LayoutInflater.from(mContext), viewSize);
96 
97         final LayoutParams lp = new LayoutParams();
98         final int defaultSize = mContext.getResources().getDimensionPixelSize(
99                 R.dimen.magnification_controls_size);
100         lp.width = viewSize.x <= 0 ? defaultSize : viewSize.x;
101         lp.height = viewSize.y <= 0 ? defaultSize : viewSize.y;
102         setDefaultParams(lp);
103         setDefaultPosition(lp);
104         mWindowManager.addView(mControlsView, lp);
105         updateDraggableBound(lp.width, lp.height);
106     }
107 
setDefaultParams(LayoutParams lp)108     private void setDefaultParams(LayoutParams lp) {
109         lp.gravity = Gravity.TOP | Gravity.LEFT;
110         lp.flags = LayoutParams.FLAG_NOT_TOUCH_MODAL
111                 | LayoutParams.FLAG_NOT_FOCUSABLE;
112         lp.type = LayoutParams.TYPE_ACCESSIBILITY_MAGNIFICATION_OVERLAY;
113         lp.format = PixelFormat.RGBA_8888;
114         lp.setTitle(getWindowTitle());
115     }
116 
setDefaultPosition(LayoutParams layoutParams)117     private void setDefaultPosition(LayoutParams layoutParams) {
118         final Point displaySize = mTmpPoint;
119         mContext.getDisplay().getSize(displaySize);
120         layoutParams.x = displaySize.x - layoutParams.width;
121         layoutParams.y = displaySize.y - layoutParams.height;
122         mControlPosition.set(layoutParams.x, layoutParams.y);
123     }
124 
125     /**
126      * Removes the UI from the scene.
127      */
destroyControl()128     public final void destroyControl() {
129         if (mControlsView != null) {
130             mWindowManager.removeView(mControlsView);
131             mControlsView = null;
132         }
133     }
134 
135     /**
136      * Moves the control view with specified offset.
137      *
138      * @param xOffset the amount in pixels to offset the UI in the X coordinate, in current
139      *                display pixels.
140      * @param yOffset the amount in pixels to offset the UI in the Y coordinate, in current
141      *                display pixels.
142      */
move(int xOffset, int yOffset)143     public void move(int xOffset, int yOffset) {
144         if (mControlsView == null) {
145             Log.w(TAG, "control view is not available yet or destroyed");
146             return;
147         }
148         final Point nextPosition = mTmpPoint;
149         nextPosition.set(mControlPosition.x, mControlPosition.y);
150         mTmpPoint.offset(xOffset, yOffset);
151         setPosition(mTmpPoint);
152     }
153 
setPosition(Point point)154     private void setPosition(Point point) {
155         constrainFrameToDraggableBound(point);
156         if (point.equals(mControlPosition)) {
157             return;
158         }
159         mControlPosition.set(point.x, point.y);
160         LayoutParams lp = (LayoutParams) mControlsView.getLayoutParams();
161         lp.x = mControlPosition.x;
162         lp.y = mControlPosition.y;
163         mWindowManager.updateViewLayout(mControlsView, lp);
164     }
165 
constrainFrameToDraggableBound(Point point)166     private void constrainFrameToDraggableBound(Point point) {
167         point.x = MathUtils.constrain(point.x, mDraggableBound.left, mDraggableBound.right);
168         point.y = MathUtils.constrain(point.y, mDraggableBound.top, mDraggableBound.bottom);
169     }
170 
updateDraggableBound(int viewWidth, int viewHeight)171     private void updateDraggableBound(int viewWidth, int viewHeight) {
172         final Point size = mTmpPoint;
173         mContext.getDisplay().getSize(size);
174         mDraggableBound.set(0, 0, size.x - viewWidth, size.y - viewHeight);
175         if (DBG) {
176             Log.d(TAG, "updateDraggableBound :" + mDraggableBound);
177         }
178     }
179 
getWindowTitle()180     abstract String getWindowTitle();
181 
182     /**
183      * Called when the UI is going to show.
184      *
185      * @param inflater The LayoutInflater object used to inflate the view.
186      * @param viewSize The {@link Point} to specify view's width with {@link Point#x)} and height
187      *                with {@link Point#y)} .The value should be greater than 0, otherwise will
188      *                 fall back to the default size.
189      * @return the View for the control's UI.
190      */
191     @NonNull
onCreateView(@onNull LayoutInflater inflater, @NonNull Point viewSize)192     abstract View onCreateView(@NonNull LayoutInflater inflater, @NonNull Point viewSize);
193 }
194