@@ -20,11 +20,18 @@ import androidx.compose.runtime.setValue
20
20
import androidx.compose.ui.Modifier
21
21
import androidx.compose.ui.draw.clipToBounds
22
22
import androidx.compose.ui.draw.shadow
23
+ import androidx.compose.ui.geometry.Offset
24
+ import androidx.compose.ui.geometry.Size
25
+ import androidx.compose.ui.graphics.Brush
26
+ import androidx.compose.ui.graphics.ClipOp
23
27
import androidx.compose.ui.graphics.Color
24
28
import androidx.compose.ui.graphics.Path
25
29
import androidx.compose.ui.graphics.PathEffect
26
30
import androidx.compose.ui.graphics.PathOperation
31
+ import androidx.compose.ui.graphics.drawscope.Fill
27
32
import androidx.compose.ui.graphics.drawscope.Stroke
33
+ import androidx.compose.ui.graphics.drawscope.clipPath
34
+ import androidx.compose.ui.graphics.drawscope.clipRect
28
35
import androidx.compose.ui.tooling.preview.Preview
29
36
import androidx.compose.ui.unit.dp
30
37
import org.zhiwei.compose.ui.widget.Title_Desc_Text
@@ -41,21 +48,26 @@ internal fun CanvasPathOperations_Screen(modifier: Modifier = Modifier) {
41
48
) {
42
49
Title_Text (title = " Clip Path/Rect" )
43
50
Title_Sub_Text (title = " 示例演示path的操作符,DrawScope的clipPath/clipRect使用不同的交互模式而形成不同的UI显示效果" )
44
- Title_Desc_Text (desc = " path.op Stroke" )
45
- PathOpStroke ()
51
+ Title_Desc_Text (desc = " path.op Stroke Fill,就是演示数学集合中交集、并集、差集、补集等" )
52
+ PathOpStrokeFill ()
53
+ PathOpStrokeFill (true )
54
+ Title_Desc_Text (desc = " clipPath 根据路径path裁剪" )
55
+ ClipPath ()
56
+ Title_Desc_Text (desc = " clipRect 根据rect矩形裁剪" )
57
+ ClipRect ()
46
58
}
47
59
48
60
}
49
61
50
62
@Composable
51
- private fun PathOpStroke ( ) {
63
+ private fun PathOpStrokeFill ( fill : Boolean = false ) {
52
64
53
65
var sides1 by remember { mutableFloatStateOf(5f ) }
54
66
var radius1 by remember { mutableFloatStateOf(300f ) }
55
67
56
68
var sides2 by remember { mutableFloatStateOf(7f ) }
57
69
var radius2 by remember { mutableFloatStateOf(300f ) }
58
-
70
+ // 两图形的交互层叠方式
59
71
var operation by remember { mutableStateOf(PathOperation .Difference ) }
60
72
61
73
val newPath = remember { Path () }
@@ -69,21 +81,21 @@ private fun PathOpStroke() {
69
81
val cy = canvasHeight / 2
70
82
71
83
72
- val path1 = createPolygonPath(cx1, cy, sides1.roundToInt(), radius1)
73
- val path2 = createPolygonPath(cx2, cy, sides2.roundToInt(), radius2)
74
-
84
+ val pathA = createPolygonPath(cx1, cy, sides1.roundToInt(), radius1)
85
+ val pathB = createPolygonPath(cx2, cy, sides2.roundToInt(), radius2)
86
+ // 绘制A图形
75
87
drawPath(
76
88
color = Color .Red ,
77
- path = path1 ,
89
+ path = pathA ,
78
90
style = Stroke (
79
91
width = 2 .dp.toPx(),
80
92
pathEffect = PathEffect .dashPathEffect(floatArrayOf(10f , 10f ))
81
93
)
82
94
)
83
-
95
+ // 绘制B图形
84
96
drawPath(
85
97
color = Color .Blue ,
86
- path = path2 ,
98
+ path = pathB ,
87
99
style = Stroke (
88
100
width = 2 .dp.toPx(),
89
101
pathEffect = PathEffect .dashPathEffect(floatArrayOf(10f , 10f ))
@@ -97,14 +109,16 @@ private fun PathOpStroke() {
97
109
The curve order is reduced where possible so that cubics may be turned into quadratics,
98
110
and quadratics maybe turned into lines.
99
111
*/
100
- newPath.op(path1, path2 , operation = operation)
101
-
112
+ newPath.op(pathA, pathB , operation = operation)
113
+ // 触发层叠方式计算后的部分,对应的路径path的绘制
102
114
drawPath(
103
115
color = Color .Green ,
104
116
path = newPath,
105
- style = Stroke (
117
+ // style可以是描边,可以是填充
118
+ style = if (fill) Fill else Stroke (
106
119
width = 4 .dp.toPx(),
107
120
)
121
+
108
122
)
109
123
}
110
124
@@ -118,7 +132,13 @@ private fun PathOpStroke() {
118
132
PathOperation .Xor -> 3
119
133
else -> 4
120
134
},
121
- options = listOf (" Difference" , " Intersect" , " Union" , " Xor" , " ReverseDifference" ),
135
+ options = listOf (
136
+ " Difference A与B的差集" ,
137
+ " Intersect B与A的交集" ,
138
+ " Union 并集" ,
139
+ " Xor 交集在并集的补集" ,
140
+ " ReverseDifference B与A的差集"
141
+ ),
122
142
onSelected = {
123
143
operation = when (it) {
124
144
0 -> PathOperation .Difference
@@ -160,6 +180,197 @@ private fun PathOpStroke() {
160
180
}
161
181
}
162
182
183
+ @Composable
184
+ private fun ClipPath () {
185
+
186
+ var sides1 by remember { mutableFloatStateOf(5f ) }
187
+ var radius1 by remember { mutableFloatStateOf(400f ) }
188
+
189
+ var sides2 by remember { mutableFloatStateOf(7f ) }
190
+ var radius2 by remember { mutableFloatStateOf(300f ) }
191
+
192
+ var clipOp by remember { mutableStateOf(ClipOp .Difference ) }
193
+
194
+ Canvas (modifier = canvasModifier) {
195
+ val canvasWidth = size.width
196
+ val canvasHeight = size.height
197
+
198
+ val cx1 = canvasWidth / 3
199
+ val cx2 = canvasWidth * 2 / 3
200
+ val cy = canvasHeight / 2
201
+
202
+ val path1 = createPolygonPath(cx1, cy, sides1.roundToInt(), radius1)
203
+ val path2 = createPolygonPath(cx2, cy, sides2.roundToInt(), radius2)
204
+
205
+
206
+ // Draw path1 to display it as reference, it's for demonstration
207
+ drawPath(
208
+ color = Color .Red ,
209
+ path = path1,
210
+ style = Stroke (
211
+ width = 2 .dp.toPx(),
212
+ pathEffect = PathEffect .dashPathEffect(floatArrayOf(40f , 20f ))
213
+ )
214
+ )
215
+
216
+ // We apply clipPath operation to pah1 and draw after this operation
217
+ /*
218
+ Reduces the clip region to the intersection of the current clip and the given path.
219
+ This method provides a callback to issue drawing commands within the region defined
220
+ by the clipped path. After this method is invoked, this clip is no longer applied
221
+ */
222
+ clipPath(path = path1, clipOp = clipOp) {
223
+
224
+ // Draw path1 to display it as reference, it's for demonstration
225
+ drawPath(
226
+ color = Color .Green ,
227
+ path = path1,
228
+ style = Stroke (
229
+ width = 2 .dp.toPx(),
230
+ pathEffect = PathEffect .dashPathEffect(floatArrayOf(20f , 40f ))
231
+ )
232
+ )
233
+
234
+
235
+ // Anything inside this scope will be clipped according to path1 shape
236
+ drawRect(
237
+ color = Color .Yellow ,
238
+ topLeft = Offset (100f , 100f ),
239
+ size = Size (canvasWidth - 300f , canvasHeight - 300f )
240
+ )
241
+
242
+ drawPath(
243
+ color = Color .Blue ,
244
+ path = path2
245
+ )
246
+
247
+ drawCircle(
248
+ brush = Brush .sweepGradient(
249
+ colors = listOf (Color .Red , Color .Green , Color .Magenta , Color .Cyan , Color .Yellow )
250
+ ),
251
+ radius = 200f
252
+ )
253
+
254
+ drawLine(
255
+ color = Color .Black ,
256
+ start = Offset (0f , 0f ),
257
+ end = Offset (canvasWidth, canvasHeight),
258
+ strokeWidth = 10f
259
+ )
260
+ }
261
+ }
262
+
263
+ Column (modifier = Modifier .padding(horizontal = 20 .dp, vertical = 4 .dp)) {
264
+
265
+ ExposedSelectionMenu (title = " Clip Operation" ,
266
+ index = when (clipOp) {
267
+ ClipOp .Difference -> 0
268
+
269
+ else -> 1
270
+ },
271
+ options = listOf (" Difference" , " Intersect" ),
272
+ onSelected = {
273
+ clipOp = when (it) {
274
+ 0 -> ClipOp .Difference
275
+ else -> ClipOp .Intersect
276
+ }
277
+ }
278
+ )
279
+
280
+ Text (text = " Sides left: ${sides1.roundToInt()} " )
281
+ Slider (
282
+ value = sides1,
283
+ onValueChange = { sides1 = it },
284
+ valueRange = 3f .. 12f ,
285
+ steps = 10
286
+ )
287
+ Text (text = " radius left: ${radius1.roundToInt()} " )
288
+ Slider (
289
+ value = radius1,
290
+ onValueChange = { radius1 = it },
291
+ valueRange = 100f .. 500f
292
+ )
293
+
294
+ Text (text = " Sides right: ${sides2.roundToInt()} " )
295
+ Slider (
296
+ value = sides2,
297
+ onValueChange = { sides2 = it },
298
+ valueRange = 3f .. 12f ,
299
+ steps = 10
300
+ )
301
+ Text (text = " radius right: ${radius2.roundToInt()} " )
302
+ Slider (
303
+ value = radius2,
304
+ onValueChange = { radius2 = it },
305
+ valueRange = 100f .. 500f
306
+ )
307
+ }
308
+ }
309
+
310
+ @Composable
311
+ private fun ClipRect () {
312
+
313
+ var clipOp by remember { mutableStateOf(ClipOp .Difference ) }
314
+
315
+ Canvas (modifier = canvasModifier) {
316
+ val canvasWidth = size.width
317
+ val canvasHeight = size.height
318
+
319
+
320
+ drawRect(
321
+ color = Color .Red ,
322
+ topLeft = Offset (100f , 80f ),
323
+ size = Size (600f , 320f ),
324
+ style = Stroke (
325
+ width = 2 .dp.toPx(),
326
+ pathEffect = PathEffect .dashPathEffect(floatArrayOf(10f , 10f ))
327
+ )
328
+ )
329
+
330
+ /*
331
+ Reduces the clip region to the intersection of the current clip and the
332
+ given rectangle indicated by the given left, top, right and bottom bounds.
333
+ This provides a callback to issue drawing commands within the clipped region.
334
+ After this method is invoked, this clip is no longer applied.
335
+ */
336
+ clipRect(left = 100f , top = 80f , right = 700f , bottom = 400f , clipOp = clipOp) {
337
+
338
+ drawCircle(
339
+ center = Offset (canvasWidth / 2 + 100 , + canvasHeight / 2 + 50 ),
340
+ brush = Brush .sweepGradient(
341
+ center = Offset (canvasWidth / 2 + 100 , + canvasHeight / 2 + 50 ),
342
+ colors = listOf (Color .Red , Color .Green , Color .Magenta , Color .Cyan , Color .Yellow )
343
+ ),
344
+ radius = 300f
345
+ )
346
+
347
+ drawLine(
348
+ color = Color .Black ,
349
+ start = Offset (0f , 0f ),
350
+ end = Offset (canvasWidth, canvasHeight),
351
+ strokeWidth = 10f
352
+ )
353
+ }
354
+ }
355
+
356
+ Column (modifier = Modifier .padding(horizontal = 20 .dp, vertical = 4 .dp)) {
357
+
358
+ ExposedSelectionMenu (title = " Clip Operation" ,
359
+ index = when (clipOp) {
360
+ ClipOp .Difference -> 0
361
+
362
+ else -> 1
363
+ },
364
+ options = listOf (" Difference" , " Intersect" ),
365
+ onSelected = {
366
+ clipOp = when (it) {
367
+ 0 -> ClipOp .Difference
368
+ else -> ClipOp .Intersect
369
+ }
370
+ }
371
+ )
372
+ }
373
+ }
163
374
164
375
private val canvasModifier = Modifier
165
376
.padding(8 .dp)
0 commit comments