blob: bed8278bc7b827c7a6f9daab7dfeb634ca0fe4f3 [file] [log] [blame]
Sunny Goyal740ac7f2016-09-28 16:47:32 -07001/*
2 * Copyright (C) 2016 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 com.android.launcher3;
18
Sunny Goyalde753212018-05-15 13:55:57 -070019import static android.view.accessibility.AccessibilityEvent.TYPE_WINDOW_CONTENT_CHANGED;
20import static android.view.accessibility.AccessibilityEvent.TYPE_WINDOW_STATE_CHANGED;
Tony Wickham9791bd12019-04-05 13:52:35 -070021
Sunny Goyalde753212018-05-15 13:55:57 -070022import static com.android.launcher3.compat.AccessibilityManagerCompat.isAccessibilityEnabled;
23import static com.android.launcher3.compat.AccessibilityManagerCompat.sendCustomAccessibilityEvent;
24
Tony Wickham50e51652017-03-20 17:12:24 -070025import android.annotation.SuppressLint;
Sunny Goyal740ac7f2016-09-28 16:47:32 -070026import android.content.Context;
Sunny Goyal740ac7f2016-09-28 16:47:32 -070027import android.util.AttributeSet;
Sunny Goyalde753212018-05-15 13:55:57 -070028import android.util.Pair;
Tony Wickham50e51652017-03-20 17:12:24 -070029import android.view.MotionEvent;
Sunny Goyal740ac7f2016-09-28 16:47:32 -070030import android.view.View;
Pinyao Ting52d02522019-09-27 16:29:32 -070031import android.view.accessibility.AccessibilityNodeInfo;
Sunny Goyalf3ac7032020-03-13 13:01:33 -070032import android.view.animation.Interpolator;
Sunny Goyal740ac7f2016-09-28 16:47:32 -070033import android.widget.LinearLayout;
34
Tony Wickham9791bd12019-04-05 13:52:35 -070035import androidx.annotation.IntDef;
Tony Wickham9791bd12019-04-05 13:52:35 -070036
Sunny Goyalf3ac7032020-03-13 13:01:33 -070037import com.android.launcher3.anim.PendingAnimation;
Sunny Goyal37920962017-09-28 13:43:24 -070038import com.android.launcher3.userevent.nano.LauncherLogProto.Action;
Tony Wickham9791bd12019-04-05 13:52:35 -070039import com.android.launcher3.userevent.nano.LauncherLogProto.ContainerType;
Sunny Goyal37920962017-09-28 13:43:24 -070040import com.android.launcher3.util.TouchController;
Sunny Goyal87b5eb62018-07-03 15:53:39 -070041import com.android.launcher3.views.ActivityContext;
Sunny Goyal0b0847b2018-03-14 12:30:11 -070042import com.android.launcher3.views.BaseDragLayer;
Sunny Goyal740ac7f2016-09-28 16:47:32 -070043
44import java.lang.annotation.Retention;
45import java.lang.annotation.RetentionPolicy;
46
47/**
48 * Base class for a View which shows a floating UI on top of the launcher UI.
49 */
Sunny Goyal37920962017-09-28 13:43:24 -070050public abstract class AbstractFloatingView extends LinearLayout implements TouchController {
Sunny Goyal740ac7f2016-09-28 16:47:32 -070051
Tony Wickham50e51652017-03-20 17:12:24 -070052 @IntDef(flag = true, value = {
53 TYPE_FOLDER,
Sunny Goyal10a1bd02017-10-09 14:56:21 -070054 TYPE_ACTION_POPUP,
Sunny Goyal37920962017-09-28 13:43:24 -070055 TYPE_WIDGETS_BOTTOM_SHEET,
Sunny Goyalf1fbc3f2017-10-10 15:21:15 -070056 TYPE_WIDGET_RESIZE_FRAME,
Sunny Goyalf8088ee2017-11-10 14:52:00 -080057 TYPE_WIDGETS_FULL_SHEET,
Tony Wickham2fae2a02017-12-14 18:38:25 -080058 TYPE_ON_BOARD_POPUP,
Vadim Tryshev17839d52018-05-23 14:45:56 -070059 TYPE_DISCOVERY_BOUNCE,
Tony Wickham6a71a5b2018-08-21 11:40:23 -070060 TYPE_SNACKBAR,
Jon Mirandade0093d2019-04-16 20:53:24 -070061 TYPE_LISTENER,
Sunny Goyal02424b22018-01-19 11:24:32 -080062
Sunny Goyal02424b22018-01-19 11:24:32 -080063 TYPE_TASK_MENU,
64 TYPE_OPTIONS_POPUP
Tony Wickham50e51652017-03-20 17:12:24 -070065 })
Sunny Goyal740ac7f2016-09-28 16:47:32 -070066 @Retention(RetentionPolicy.SOURCE)
67 public @interface FloatingViewType {}
68 public static final int TYPE_FOLDER = 1 << 0;
Sunny Goyal10a1bd02017-10-09 14:56:21 -070069 public static final int TYPE_ACTION_POPUP = 1 << 1;
Tony Wickham343a77e2017-04-12 18:31:09 -070070 public static final int TYPE_WIDGETS_BOTTOM_SHEET = 1 << 2;
Sunny Goyal37920962017-09-28 13:43:24 -070071 public static final int TYPE_WIDGET_RESIZE_FRAME = 1 << 3;
Sunny Goyalf1fbc3f2017-10-10 15:21:15 -070072 public static final int TYPE_WIDGETS_FULL_SHEET = 1 << 4;
Sunny Goyal02424b22018-01-19 11:24:32 -080073 public static final int TYPE_ON_BOARD_POPUP = 1 << 5;
Vadim Tryshev17839d52018-05-23 14:45:56 -070074 public static final int TYPE_DISCOVERY_BOUNCE = 1 << 6;
Tony Wickham6a71a5b2018-08-21 11:40:23 -070075 public static final int TYPE_SNACKBAR = 1 << 7;
Jon Mirandade0093d2019-04-16 20:53:24 -070076 public static final int TYPE_LISTENER = 1 << 8;
Sunny Goyal02424b22018-01-19 11:24:32 -080077
78 // Popups related to quickstep UI
Jon Mirandade0093d2019-04-16 20:53:24 -070079 public static final int TYPE_TASK_MENU = 1 << 9;
80 public static final int TYPE_OPTIONS_POPUP = 1 << 10;
Sunny Goyalf1fbc3f2017-10-10 15:21:15 -070081
82 public static final int TYPE_ALL = TYPE_FOLDER | TYPE_ACTION_POPUP
Sunny Goyalf8088ee2017-11-10 14:52:00 -080083 | TYPE_WIDGETS_BOTTOM_SHEET | TYPE_WIDGET_RESIZE_FRAME | TYPE_WIDGETS_FULL_SHEET
Sunny Goyal462551b2019-02-18 14:42:47 -080084 | TYPE_ON_BOARD_POPUP | TYPE_DISCOVERY_BOUNCE | TYPE_TASK_MENU
Jon Mirandade0093d2019-04-16 20:53:24 -070085 | TYPE_OPTIONS_POPUP | TYPE_SNACKBAR | TYPE_LISTENER;
Sunny Goyal740ac7f2016-09-28 16:47:32 -070086
Sunny Goyal7ede6112017-12-05 15:11:21 -080087 // Type of popups which should be kept open during launcher rebind
88 public static final int TYPE_REBIND_SAFE = TYPE_WIDGETS_FULL_SHEET
Pinyao Tinga74b63a2019-07-17 23:26:06 -070089 | TYPE_WIDGETS_BOTTOM_SHEET | TYPE_ON_BOARD_POPUP | TYPE_DISCOVERY_BOUNCE;
Sunny Goyal7ede6112017-12-05 15:11:21 -080090
Tony Wickhamdf1eb8b2018-04-12 17:26:18 -070091 // Usually we show the back button when a floating view is open. Instead, hide for these types.
Tony Wickham6a71a5b2018-08-21 11:40:23 -070092 public static final int TYPE_HIDE_BACK_BUTTON = TYPE_ON_BOARD_POPUP | TYPE_DISCOVERY_BOUNCE
Tonyf941f842019-05-13 11:15:54 -050093 | TYPE_SNACKBAR | TYPE_WIDGET_RESIZE_FRAME | TYPE_LISTENER;
Vadim Tryshev17839d52018-05-23 14:45:56 -070094
Jon Mirandade0093d2019-04-16 20:53:24 -070095 public static final int TYPE_ACCESSIBLE = TYPE_ALL & ~TYPE_DISCOVERY_BOUNCE & ~TYPE_LISTENER;
Tony Wickhamdf1eb8b2018-04-12 17:26:18 -070096
Hyunyoung Songf58cf5e2018-09-19 16:06:16 -070097 // These view all have particular operation associated with swipe down interaction.
98 public static final int TYPE_STATUS_BAR_SWIPE_DOWN_DISALLOW = TYPE_WIDGETS_BOTTOM_SHEET |
99 TYPE_WIDGETS_FULL_SHEET | TYPE_WIDGET_RESIZE_FRAME | TYPE_ON_BOARD_POPUP |
100 TYPE_DISCOVERY_BOUNCE | TYPE_TASK_MENU ;
101
Sunny Goyal740ac7f2016-09-28 16:47:32 -0700102 protected boolean mIsOpen;
103
104 public AbstractFloatingView(Context context, AttributeSet attrs) {
105 super(context, attrs);
106 }
107
108 public AbstractFloatingView(Context context, AttributeSet attrs, int defStyleAttr) {
109 super(context, attrs, defStyleAttr);
110 }
111
Tony Wickham50e51652017-03-20 17:12:24 -0700112 /**
113 * We need to handle touch events to prevent them from falling through to the workspace below.
114 */
115 @SuppressLint("ClickableViewAccessibility")
116 @Override
117 public boolean onTouchEvent(MotionEvent ev) {
118 return true;
119 }
120
Sunny Goyal740ac7f2016-09-28 16:47:32 -0700121 public final void close(boolean animate) {
Sunny Goyal7368fa42019-04-22 09:58:14 -0700122 animate &= Utilities.areAnimationsEnabled(getContext());
Hyunyoung Songbd6fba92018-05-16 15:54:31 -0700123 if (mIsOpen) {
124 BaseActivity.fromContext(getContext()).getUserEventDispatcher()
125 .resetElapsedContainerMillis("container closed");
126 }
Sunny Goyal740ac7f2016-09-28 16:47:32 -0700127 handleClose(animate);
Jon Miranda83337f92018-04-24 12:21:28 -0700128 mIsOpen = false;
Sunny Goyal740ac7f2016-09-28 16:47:32 -0700129 }
130
131 protected abstract void handleClose(boolean animate);
132
Tony Wickham9791bd12019-04-05 13:52:35 -0700133 /**
134 * Creates a user-controlled animation to hint that the view will be closed if completed.
135 * @param distanceToMove The max distance that elements should move from their starting point.
136 */
Sunny Goyalf3ac7032020-03-13 13:01:33 -0700137 public void addHintCloseAnim(
138 float distanceToMove, Interpolator interpolator, PendingAnimation target) { }
Tony Wickham9791bd12019-04-05 13:52:35 -0700139
Sunny Goyal37920962017-09-28 13:43:24 -0700140 public abstract void logActionCommand(int command);
Sunny Goyal740ac7f2016-09-28 16:47:32 -0700141
Tony Wickham9791bd12019-04-05 13:52:35 -0700142 public int getLogContainerType() {
143 return ContainerType.DEFAULT_CONTAINERTYPE;
144 }
145
Sunny Goyal740ac7f2016-09-28 16:47:32 -0700146 public final boolean isOpen() {
147 return mIsOpen;
148 }
149
150 protected abstract boolean isOfType(@FloatingViewType int type);
151
Tony Wickham52c1b662018-05-21 13:13:58 -0700152 /** @return Whether the back is consumed. If false, Launcher will handle the back as well. */
153 public boolean onBackPressed() {
Sunny Goyal37920962017-09-28 13:43:24 -0700154 logActionCommand(Action.Command.BACK);
155 close(true);
Tony Wickham52c1b662018-05-21 13:13:58 -0700156 return true;
Sunny Goyal37920962017-09-28 13:43:24 -0700157 }
158
159 @Override
160 public boolean onControllerTouchEvent(MotionEvent ev) {
161 return false;
162 }
163
Sunny Goyalde753212018-05-15 13:55:57 -0700164 protected void announceAccessibilityChanges() {
165 Pair<View, String> targetInfo = getAccessibilityTarget();
166 if (targetInfo == null || !isAccessibilityEnabled(getContext())) {
167 return;
168 }
169 sendCustomAccessibilityEvent(
170 targetInfo.first, TYPE_WINDOW_STATE_CHANGED, targetInfo.second);
171
172 if (mIsOpen) {
Pinyao Ting52d02522019-09-27 16:29:32 -0700173 performAccessibilityAction(AccessibilityNodeInfo.ACTION_ACCESSIBILITY_FOCUS, null);
Sunny Goyalde753212018-05-15 13:55:57 -0700174 }
Sunny Goyal87b5eb62018-07-03 15:53:39 -0700175 ActivityContext.lookupContext(getContext()).getDragLayer()
Sunny Goyalde753212018-05-15 13:55:57 -0700176 .sendAccessibilityEvent(TYPE_WINDOW_CONTENT_CHANGED);
177 }
178
179 protected Pair<View, String> getAccessibilityTarget() {
180 return null;
181 }
182
Samuel Fufa82bbdac2020-03-09 18:24:47 -0700183 /**
184 * Returns a view matching FloatingViewType
185 */
186 public static <T extends AbstractFloatingView> T getOpenView(
Sunny Goyal87b5eb62018-07-03 15:53:39 -0700187 ActivityContext activity, @FloatingViewType int type) {
Sunny Goyal0b0847b2018-03-14 12:30:11 -0700188 BaseDragLayer dragLayer = activity.getDragLayer();
vadimt5a22ef72019-04-17 18:39:00 -0700189 if (dragLayer == null) return null;
Sunny Goyal740ac7f2016-09-28 16:47:32 -0700190 // Iterate in reverse order. AbstractFloatingView is added later to the dragLayer,
191 // and will be one of the last views.
192 for (int i = dragLayer.getChildCount() - 1; i >= 0; i--) {
193 View child = dragLayer.getChildAt(i);
194 if (child instanceof AbstractFloatingView) {
195 AbstractFloatingView view = (AbstractFloatingView) child;
196 if (view.isOfType(type) && view.isOpen()) {
197 return (T) view;
198 }
199 }
200 }
201 return null;
202 }
203
Sunny Goyal87b5eb62018-07-03 15:53:39 -0700204 public static void closeOpenContainer(ActivityContext activity,
Sunny Goyal0b0847b2018-03-14 12:30:11 -0700205 @FloatingViewType int type) {
206 AbstractFloatingView view = getOpenView(activity, type);
Sunny Goyal740ac7f2016-09-28 16:47:32 -0700207 if (view != null) {
208 view.close(true);
209 }
210 }
211
Sunny Goyal87b5eb62018-07-03 15:53:39 -0700212 public static void closeOpenViews(ActivityContext activity, boolean animate,
Sunny Goyalf1fbc3f2017-10-10 15:21:15 -0700213 @FloatingViewType int type) {
Sunny Goyal0b0847b2018-03-14 12:30:11 -0700214 BaseDragLayer dragLayer = activity.getDragLayer();
Sunny Goyal740ac7f2016-09-28 16:47:32 -0700215 // Iterate in reverse order. AbstractFloatingView is added later to the dragLayer,
216 // and will be one of the last views.
217 for (int i = dragLayer.getChildCount() - 1; i >= 0; i--) {
218 View child = dragLayer.getChildAt(i);
219 if (child instanceof AbstractFloatingView) {
Sunny Goyalf1fbc3f2017-10-10 15:21:15 -0700220 AbstractFloatingView abs = (AbstractFloatingView) child;
221 if (abs.isOfType(type)) {
222 abs.close(animate);
223 }
Sunny Goyal740ac7f2016-09-28 16:47:32 -0700224 }
225 }
226 }
227
Sunny Goyal87b5eb62018-07-03 15:53:39 -0700228 public static void closeAllOpenViews(ActivityContext activity, boolean animate) {
Sunny Goyal0b0847b2018-03-14 12:30:11 -0700229 closeOpenViews(activity, animate, TYPE_ALL);
230 activity.finishAutoCancelActionMode();
Sunny Goyalf1fbc3f2017-10-10 15:21:15 -0700231 }
232
Sunny Goyal87b5eb62018-07-03 15:53:39 -0700233 public static void closeAllOpenViews(ActivityContext activity) {
Sunny Goyal0b0847b2018-03-14 12:30:11 -0700234 closeAllOpenViews(activity, true);
Sunny Goyal740ac7f2016-09-28 16:47:32 -0700235 }
236
Tracy Zhoud43e7c22018-10-16 14:49:55 -0700237 public static void closeAllOpenViewsExcept(ActivityContext activity, boolean animate,
238 @FloatingViewType int type) {
239 closeOpenViews(activity, animate, TYPE_ALL & ~type);
240 activity.finishAutoCancelActionMode();
241 }
242
243 public static void closeAllOpenViewsExcept(ActivityContext activity,
244 @FloatingViewType int type) {
245 closeAllOpenViewsExcept(activity, true, type);
246 }
247
Sunny Goyal87b5eb62018-07-03 15:53:39 -0700248 public static AbstractFloatingView getTopOpenView(ActivityContext activity) {
Tony Wickhamdf1eb8b2018-04-12 17:26:18 -0700249 return getTopOpenViewWithType(activity, TYPE_ALL);
250 }
251
Sunny Goyal87b5eb62018-07-03 15:53:39 -0700252 public static AbstractFloatingView getTopOpenViewWithType(ActivityContext activity,
Tony Wickhamdf1eb8b2018-04-12 17:26:18 -0700253 @FloatingViewType int type) {
254 return getOpenView(activity, type);
Sunny Goyal740ac7f2016-09-28 16:47:32 -0700255 }
Sunny Goyal740ac7f2016-09-28 16:47:32 -0700256}