• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
<lambda>null2  * Copyright (C) 2024 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 package com.android.wm.shell.shared.bubbles
17 
18 import android.annotation.ColorInt
19 import android.graphics.Canvas
20 import android.graphics.ColorFilter
21 import android.graphics.Matrix
22 import android.graphics.Outline
23 import android.graphics.Paint
24 import android.graphics.Path
25 import android.graphics.Rect
26 import android.graphics.RectF
27 import android.graphics.drawable.Drawable
28 import kotlin.math.atan
29 import kotlin.math.cos
30 import kotlin.math.sin
31 import kotlin.properties.Delegates
32 
33 /** A drawable for the [BubblePopupView] that draws a popup background with a directional arrow */
34 class BubblePopupDrawable(val config: Config) : Drawable() {
35     /** The direction of the arrow in the popup drawable */
36     enum class ArrowDirection {
37         UP,
38         DOWN
39     }
40 
41     /** The arrow position on the side of the popup bubble */
42     sealed class ArrowPosition {
43         object Start : ArrowPosition()
44         object Center : ArrowPosition()
45         object End : ArrowPosition()
46         class Custom(val value: Float) : ArrowPosition()
47     }
48 
49     /** The configuration for drawable features */
50     data class Config(
51         @ColorInt val color: Int,
52         val cornerRadius: Float,
53         val contentPadding: Int,
54         val arrowWidth: Float,
55         val arrowHeight: Float,
56         val arrowRadius: Float
57     )
58 
59     /**
60      * The direction of the arrow in the popup drawable. It affects the content padding and requires
61      * it to be updated in the view.
62      */
63     var arrowDirection: ArrowDirection by
64         Delegates.observable(ArrowDirection.UP) { _, _, _ -> requestPathUpdate() }
65 
66     /**
67      * Arrow position along the X axis and its direction. The position is adjusted to the content
68      * corner radius when applied so it doesn't go into rounded corner area
69      */
70     var arrowPosition: ArrowPosition by
71         Delegates.observable(ArrowPosition.Center) { _, _, _ -> requestPathUpdate() }
72 
73     private val path = Path()
74     private val paint = Paint()
75     private var shouldUpdatePath = true
76 
77     init {
78         paint.color = config.color
79         paint.style = Paint.Style.FILL
80         paint.isAntiAlias = true
81     }
82 
83     override fun draw(canvas: Canvas) {
84         updatePathIfNeeded()
85         canvas.drawPath(path, paint)
86     }
87 
88     override fun onBoundsChange(bounds: Rect) {
89         requestPathUpdate()
90     }
91 
92     /** Should be applied to the view padding if arrow direction changes */
93     override fun getPadding(padding: Rect): Boolean {
94         padding.set(
95             config.contentPadding,
96             config.contentPadding,
97             config.contentPadding,
98             config.contentPadding
99         )
100         when (arrowDirection) {
101             ArrowDirection.UP -> padding.top += config.arrowHeight.toInt()
102             ArrowDirection.DOWN -> padding.bottom += config.arrowHeight.toInt()
103         }
104         return true
105     }
106 
107     override fun getOutline(outline: Outline) {
108         updatePathIfNeeded()
109         outline.setPath(path)
110     }
111 
112     override fun getOpacity(): Int {
113         return paint.alpha
114     }
115 
116     override fun setAlpha(alpha: Int) {
117         paint.alpha = alpha
118     }
119 
120     override fun setColorFilter(colorFilter: ColorFilter?) {
121         paint.colorFilter = colorFilter
122     }
123 
124     /** Schedules path update for the next redraw */
125     private fun requestPathUpdate() {
126         shouldUpdatePath = true
127     }
128 
129     /** Updates the path if required, when bounds or arrow direction/position changes */
130     private fun updatePathIfNeeded() {
131         if (shouldUpdatePath) {
132             updatePath()
133             shouldUpdatePath = false
134         }
135     }
136 
137     /** Updates the path value using the current bounds, config, arrow direction and position */
138     private fun updatePath() {
139         if (bounds.isEmpty) return
140         // Reset the path state
141         path.reset()
142         // The content rect where the filled rounded rect will be drawn
143         val contentRect = RectF(bounds)
144         when (arrowDirection) {
145             ArrowDirection.UP -> {
146                 // Add rounded arrow pointing up to the path
147                 addRoundedArrowPositioned(path, arrowPosition)
148                 // Inset content rect by the arrow size from the top
149                 contentRect.top += config.arrowHeight
150             }
151             ArrowDirection.DOWN -> {
152                 val matrix = Matrix()
153                 // Flip the path with the matrix to draw arrow pointing down
154                 matrix.setScale(1f, -1f, bounds.width() / 2f, bounds.height() / 2f)
155                 path.transform(matrix)
156                 // Add rounded arrow with the flipped matrix applied, will point down
157                 addRoundedArrowPositioned(path, arrowPosition)
158                 // Restore the path matrix to the original state with inverted matrix
159                 matrix.invert(matrix)
160                 path.transform(matrix)
161                 // Inset content rect by the arrow size from the bottom
162                 contentRect.bottom -= config.arrowHeight
163             }
164         }
165         // Add the content area rounded rect
166         path.addRoundRect(contentRect, config.cornerRadius, config.cornerRadius, Path.Direction.CW)
167     }
168 
169     /** Add a rounded arrow pointing up in the horizontal position on the canvas */
170     private fun addRoundedArrowPositioned(path: Path, position: ArrowPosition) {
171         val matrix = Matrix()
172         var translationX = positionValue(position) - config.arrowWidth / 2
173         // Offset to position between rounded corners of the content view
174         translationX = translationX.coerceIn(config.cornerRadius,
175                 bounds.width() - config.cornerRadius - config.arrowWidth)
176         // Translate to add the arrow in the center horizontally
177         matrix.setTranslate(-translationX, 0f)
178         path.transform(matrix)
179         // Add rounded arrow
180         addRoundedArrow(path)
181         // Restore the path matrix to the original state with inverted matrix
182         matrix.invert(matrix)
183         path.transform(matrix)
184     }
185 
186     /** Adds a rounded arrow pointing up to the path, can be flipped if needed */
187     private fun addRoundedArrow(path: Path) {
188         // Theta is half of the angle inside the triangle tip
189         val thetaTan = config.arrowWidth / (config.arrowHeight * 2f)
190         val theta = atan(thetaTan)
191         val thetaDeg = Math.toDegrees(theta.toDouble()).toFloat()
192         // The center Y value of the circle for the triangle tip
193         val tipCircleCenterY = config.arrowRadius / sin(theta)
194         // The length from triangle tip to intersection point with the circle
195         val tipIntersectionSideLength = config.arrowRadius / thetaTan
196         // The offset from the top to the point of intersection
197         val intersectionTopOffset = tipIntersectionSideLength * cos(theta)
198         // The offset from the center to the point of intersection
199         val intersectionCenterOffset = tipIntersectionSideLength * sin(theta)
200         // The center X of the triangle
201         val arrowCenterX = config.arrowWidth / 2f
202 
203         // Set initial position in bottom left of the arrow
204         path.moveTo(0f, config.arrowHeight)
205         // Add the left side of the triangle
206         path.lineTo(arrowCenterX - intersectionCenterOffset, intersectionTopOffset)
207         // Add the arc from the left to the right side of the triangle
208         path.arcTo(
209             /* left = */ arrowCenterX - config.arrowRadius,
210             /* top = */ tipCircleCenterY - config.arrowRadius,
211             /* right = */ arrowCenterX + config.arrowRadius,
212             /* bottom = */ tipCircleCenterY + config.arrowRadius,
213             /* startAngle = */ 180 + thetaDeg,
214             /* sweepAngle = */ 180 - (2 * thetaDeg),
215             /* forceMoveTo = */ false
216         )
217         // Add the right side of the triangle
218         path.lineTo(config.arrowWidth, config.arrowHeight)
219         // Close the path
220         path.close()
221     }
222 
223     /** The value of the arrow position provided the position and current bounds */
224     private fun positionValue(position: ArrowPosition): Float {
225         return when (position) {
226             is ArrowPosition.Start -> 0f
227             is ArrowPosition.Center -> bounds.width().toFloat() / 2f
228             is ArrowPosition.End -> bounds.width().toFloat()
229             is ArrowPosition.Custom -> position.value
230         }
231     }
232 }
233