1 /* <lambda>null2 * 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.controls 18 19 import android.annotation.StringRes 20 import android.content.Context 21 import android.graphics.CornerPathEffect 22 import android.graphics.drawable.ShapeDrawable 23 import android.util.TypedValue 24 import android.view.LayoutInflater 25 import android.view.View 26 import android.view.ViewGroup 27 import android.view.animation.AccelerateInterpolator 28 import android.view.animation.DecelerateInterpolator 29 import android.widget.TextView 30 import com.android.systemui.Prefs 31 import com.android.systemui.res.R 32 import com.android.systemui.recents.TriangleShape 33 34 /** 35 * Manager for showing an onboarding tooltip on screen. 36 * 37 * The tooltip can be made to appear below or above a point. The number of times it will appear 38 * is determined by an shared preference (defined in [Prefs]). 39 * 40 * @property context A context to use to inflate the views and retrieve shared preferences from 41 * @property preferenceName name of the preference to use to track the number of times the tooltip 42 * has been shown. 43 * @property maxTimesShown the maximum number of times to show the tooltip 44 * @property below whether the tooltip should appear below (with up pointing arrow) or above (down 45 * pointing arrow) the specified point. 46 * @see [TooltipManager.show] 47 */ 48 class TooltipManager( 49 context: Context, 50 private val preferenceName: String, 51 private val maxTimesShown: Int = 2, 52 private val below: Boolean = true 53 ) { 54 55 companion object { 56 private const val SHOW_DELAY_MS: Long = 500 57 private const val SHOW_DURATION_MS: Long = 300 58 private const val HIDE_DURATION_MS: Long = 100 59 } 60 61 private var shown = Prefs.getInt(context, preferenceName, 0) 62 63 val layout: ViewGroup = 64 LayoutInflater.from(context).inflate(R.layout.controls_onboarding, null) as ViewGroup 65 val preferenceStorer = { num: Int -> 66 Prefs.putInt(context, preferenceName, num) 67 } 68 69 init { 70 layout.alpha = 0f 71 } 72 73 private val textView = layout.requireViewById<TextView>(R.id.onboarding_text) 74 private val dismissView = layout.requireViewById<View>(R.id.dismiss).apply { 75 setOnClickListener { 76 hide(true) 77 } 78 } 79 80 private val arrowView = layout.requireViewById<View>(R.id.arrow).apply { 81 val typedValue = TypedValue() 82 context.theme.resolveAttribute(android.R.attr.colorAccent, typedValue, true) 83 val toastColor = context.resources.getColor(typedValue.resourceId, context.theme) 84 val arrowRadius = context.resources.getDimensionPixelSize( 85 R.dimen.recents_onboarding_toast_arrow_corner_radius) 86 val arrowLp = layoutParams 87 val arrowDrawable = ShapeDrawable(TriangleShape.create( 88 arrowLp.width.toFloat(), arrowLp.height.toFloat(), below)) 89 val arrowPaint = arrowDrawable.paint 90 arrowPaint.color = toastColor 91 // The corner path effect won't be reflected in the shadow, but shouldn't be noticeable. 92 arrowPaint.pathEffect = CornerPathEffect(arrowRadius.toFloat()) 93 setBackground(arrowDrawable) 94 } 95 96 init { 97 if (!below) { 98 layout.removeView(arrowView) 99 layout.addView(arrowView) 100 (arrowView.layoutParams as ViewGroup.MarginLayoutParams).apply { 101 bottomMargin = topMargin 102 topMargin = 0 103 } 104 } 105 } 106 107 /** 108 * Show the tooltip 109 * 110 * @param stringRes the id of the string to show in the tooltip 111 * @param x horizontal position (w.r.t. screen) for the arrow point 112 * @param y vertical position (w.r.t. screen) for the arrow point 113 */ 114 fun show(@StringRes stringRes: Int, x: Int, y: Int) { 115 if (!shouldShow()) return 116 textView.setText(stringRes) 117 shown++ 118 preferenceStorer(shown) 119 layout.post { 120 val p = IntArray(2) 121 layout.getLocationOnScreen(p) 122 layout.translationX = (x - p[0] - layout.width / 2).toFloat() 123 layout.translationY = (y - p[1]).toFloat() - if (!below) layout.height else 0 124 if (layout.alpha == 0f) { 125 layout.animate() 126 .alpha(1f) 127 .withLayer() 128 .setStartDelay(SHOW_DELAY_MS) 129 .setDuration(SHOW_DURATION_MS) 130 .setInterpolator(DecelerateInterpolator()) 131 .start() 132 } 133 } 134 } 135 136 /** 137 * Hide the tooltip 138 * 139 * @param animate whether to animate the fade out 140 */ 141 fun hide(animate: Boolean = false) { 142 if (layout.alpha == 0f) return 143 layout.post { 144 if (animate) { 145 layout.animate() 146 .alpha(0f) 147 .withLayer() 148 .setStartDelay(0) 149 .setDuration(HIDE_DURATION_MS) 150 .setInterpolator(AccelerateInterpolator()) 151 .start() 152 } else { 153 layout.animate().cancel() 154 layout.alpha = 0f 155 } 156 } 157 } 158 159 private fun shouldShow() = shown < maxTimesShown 160 }