/* * Copyright (C) 2022 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.settings.accessibility; import android.content.Context; import android.content.res.Resources; import android.graphics.drawable.ColorDrawable; import android.os.Bundle; import android.os.Handler; import android.view.Gravity; import android.view.LayoutInflater; import android.view.View; import android.view.View.AccessibilityDelegate; import android.view.accessibility.AccessibilityNodeInfo; import android.view.accessibility.AccessibilityNodeInfo.AccessibilityAction; import android.widget.ImageView; import android.widget.LinearLayout; import android.widget.PopupWindow; import android.widget.TextView; import androidx.annotation.DrawableRes; import androidx.annotation.VisibleForTesting; import com.android.settings.R; /** * UI container for the accessibility quick settings tooltip. * *

The popup window shows the information about the operation of the quick settings. In * addition, the arrow is pointing to the top center of the device to display one-off menu within * {@code mCloseDelayTimeMillis} time.

*/ public class AccessibilityQuickSettingsTooltipWindow extends PopupWindow { private final Context mContext; private Handler mHandler; private long mCloseDelayTimeMillis; public AccessibilityQuickSettingsTooltipWindow(Context context) { super(context); this.mContext = context; } private final AccessibilityDelegate mAccessibilityDelegate = new AccessibilityDelegate() { @Override public void onInitializeAccessibilityNodeInfo(View host, AccessibilityNodeInfo info) { super.onInitializeAccessibilityNodeInfo(host, info); final AccessibilityAction clickAction = new AccessibilityAction( AccessibilityNodeInfo.ACTION_CLICK, mContext.getString(R.string.accessibility_quick_settings_tooltip_dismiss)); info.addAction(clickAction); } @Override public boolean performAccessibilityAction(View host, int action, Bundle args) { if (action == AccessibilityNodeInfo.ACTION_CLICK) { dismiss(); return true; } return super.performAccessibilityAction(host, action, args); } }; /** * Sets up {@link #AccessibilityQuickSettingsTooltipWindow}'s layout and content. * * @param text text to be displayed * @param imageResId the resource ID of the image drawable */ public void setup(CharSequence text, @DrawableRes int imageResId) { this.setup(text, imageResId, /* closeDelayTimeMillis= */ 0); } /** * Sets up {@link #AccessibilityQuickSettingsTooltipWindow}'s layout and content. * *

The system will attempt to close popup window to the target duration of the threads if * close delay time is positive number.

* * @param text text to be displayed * @param imageResId the resource ID of the image drawable * @param closeDelayTimeMillis how long the popup window be auto-closed */ public void setup(CharSequence text, @DrawableRes int imageResId, long closeDelayTimeMillis) { this.mCloseDelayTimeMillis = closeDelayTimeMillis; setBackgroundDrawable(new ColorDrawable(mContext.getColor(android.R.color.transparent))); final LayoutInflater inflater = mContext.getSystemService(LayoutInflater.class); final View popupView = inflater.inflate(R.layout.accessibility_qs_tooltip, /* root= */ null); popupView.setFocusable(/* focusable= */ true); popupView.setAccessibilityDelegate(mAccessibilityDelegate); setContentView(popupView); final ImageView imageView = getContentView().findViewById(R.id.qs_illustration); imageView.setImageResource(imageResId); final TextView textView = getContentView().findViewById(R.id.qs_content); textView.setText(text); setWidth(getWindowWidthWith(textView)); setHeight(LinearLayout.LayoutParams.WRAP_CONTENT); setFocusable(/* focusable= */ true); setOutsideTouchable(/* touchable= */ true); } /** * Displays the content view in a popup window at the top and center position. * * @param targetView a target view to get the {@link View#getWindowToken()} token from. */ public void showAtTopCenter(View targetView) { showAtLocation(targetView, Gravity.TOP | Gravity.CENTER_HORIZONTAL, 0, 0); } /** * Disposes of the popup window. * *

Remove any pending posts of callbacks and sent messages for closing popup window.

*/ @Override public void dismiss() { super.dismiss(); if (mHandler != null) { mHandler.removeCallbacksAndMessages(/* token= */ null); } } /** * Displays the content view in a popup window at the specified location. * *

The system will attempt to close popup window to the target duration of the threads if * close delay time is positive number.

* * @param parent a parent view to get the {@link android.view.View#getWindowToken()} token from * @param gravity the gravity which controls the placement of the popup window * @param x the popup's x location offset * @param y the popup's y location offset */ @Override public void showAtLocation(View parent, int gravity, int x, int y) { super.showAtLocation(parent, gravity, x, y); scheduleAutoCloseAction(); } private void scheduleAutoCloseAction() { if (mCloseDelayTimeMillis <= 0) { return; } if (mHandler == null) { mHandler = new Handler(mContext.getMainLooper()); } mHandler.removeCallbacksAndMessages(/* token= */ null); mHandler.postDelayed(this::dismiss, mCloseDelayTimeMillis); } private int getWindowWidthWith(TextView textView) { final int availableWindowWidth = getAvailableWindowWidth(); final int widthSpec = View.MeasureSpec.makeMeasureSpec(availableWindowWidth, View.MeasureSpec.AT_MOST); final int heightSpec = View.MeasureSpec.makeMeasureSpec(0, View.MeasureSpec.UNSPECIFIED); textView.measure(widthSpec, heightSpec); return textView.getMeasuredWidth(); } @VisibleForTesting int getAvailableWindowWidth() { final Resources res = mContext.getResources(); final int padding = res.getDimensionPixelSize(R.dimen.accessibility_qs_tooltip_margin); final int screenWidth = res.getDisplayMetrics().widthPixels; return screenWidth - padding * 2; } }