Skip to content

Commit d1dc865

Browse files
author
iOrchid
committedMay 20, 2024·
简单的裁剪clip操作
1 parent 517edc3 commit d1dc865

File tree

2 files changed

+234
-30
lines changed

2 files changed

+234
-30
lines changed
 

‎compose/src/main/java/org/zhiwei/compose/screen/graphics/CanvasPathOperations.kt

+225-14
Original file line numberDiff line numberDiff line change
@@ -20,11 +20,18 @@ import androidx.compose.runtime.setValue
2020
import androidx.compose.ui.Modifier
2121
import androidx.compose.ui.draw.clipToBounds
2222
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
2327
import androidx.compose.ui.graphics.Color
2428
import androidx.compose.ui.graphics.Path
2529
import androidx.compose.ui.graphics.PathEffect
2630
import androidx.compose.ui.graphics.PathOperation
31+
import androidx.compose.ui.graphics.drawscope.Fill
2732
import androidx.compose.ui.graphics.drawscope.Stroke
33+
import androidx.compose.ui.graphics.drawscope.clipPath
34+
import androidx.compose.ui.graphics.drawscope.clipRect
2835
import androidx.compose.ui.tooling.preview.Preview
2936
import androidx.compose.ui.unit.dp
3037
import org.zhiwei.compose.ui.widget.Title_Desc_Text
@@ -41,21 +48,26 @@ internal fun CanvasPathOperations_Screen(modifier: Modifier = Modifier) {
4148
) {
4249
Title_Text(title = "Clip Path/Rect")
4350
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()
4658
}
4759

4860
}
4961

5062
@Composable
51-
private fun PathOpStroke() {
63+
private fun PathOpStrokeFill(fill: Boolean = false) {
5264

5365
var sides1 by remember { mutableFloatStateOf(5f) }
5466
var radius1 by remember { mutableFloatStateOf(300f) }
5567

5668
var sides2 by remember { mutableFloatStateOf(7f) }
5769
var radius2 by remember { mutableFloatStateOf(300f) }
58-
70+
//两图形的交互层叠方式
5971
var operation by remember { mutableStateOf(PathOperation.Difference) }
6072

6173
val newPath = remember { Path() }
@@ -69,21 +81,21 @@ private fun PathOpStroke() {
6981
val cy = canvasHeight / 2
7082

7183

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图形
7587
drawPath(
7688
color = Color.Red,
77-
path = path1,
89+
path = pathA,
7890
style = Stroke(
7991
width = 2.dp.toPx(),
8092
pathEffect = PathEffect.dashPathEffect(floatArrayOf(10f, 10f))
8193
)
8294
)
83-
95+
//绘制B图形
8496
drawPath(
8597
color = Color.Blue,
86-
path = path2,
98+
path = pathB,
8799
style = Stroke(
88100
width = 2.dp.toPx(),
89101
pathEffect = PathEffect.dashPathEffect(floatArrayOf(10f, 10f))
@@ -97,14 +109,16 @@ private fun PathOpStroke() {
97109
The curve order is reduced where possible so that cubics may be turned into quadratics,
98110
and quadratics maybe turned into lines.
99111
*/
100-
newPath.op(path1, path2, operation = operation)
101-
112+
newPath.op(pathA, pathB, operation = operation)
113+
//触发层叠方式计算后的部分,对应的路径path的绘制
102114
drawPath(
103115
color = Color.Green,
104116
path = newPath,
105-
style = Stroke(
117+
//style可以是描边,可以是填充
118+
style = if (fill) Fill else Stroke(
106119
width = 4.dp.toPx(),
107120
)
121+
108122
)
109123
}
110124

@@ -118,7 +132,13 @@ private fun PathOpStroke() {
118132
PathOperation.Xor -> 3
119133
else -> 4
120134
},
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+
),
122142
onSelected = {
123143
operation = when (it) {
124144
0 -> PathOperation.Difference
@@ -160,6 +180,197 @@ private fun PathOpStroke() {
160180
}
161181
}
162182

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+
}
163374

164375
private val canvasModifier = Modifier
165376
.padding(8.dp)

‎compose/src/main/java/org/zhiwei/compose/screen/graphics/Ui.kt

+9-16
Original file line numberDiff line numberDiff line change
@@ -15,24 +15,24 @@ import androidx.compose.foundation.layout.height
1515
import androidx.compose.foundation.layout.padding
1616
import androidx.compose.foundation.layout.width
1717
import androidx.compose.foundation.shape.RoundedCornerShape
18-
import androidx.compose.material.DropdownMenuItem
1918
import androidx.compose.material.ExperimentalMaterialApi
2019
import androidx.compose.material.ExposedDropdownMenuBox
21-
import androidx.compose.material.ExposedDropdownMenuDefaults
22-
import androidx.compose.material.TextField
2320
import androidx.compose.material.icons.Icons
2421
import androidx.compose.material.icons.filled.BorderColor
2522
import androidx.compose.material.icons.filled.Brush
2623
import androidx.compose.material.icons.filled.ColorLens
2724
import androidx.compose.material.icons.filled.TouchApp
2825
import androidx.compose.material3.Card
2926
import androidx.compose.material3.CardDefaults
27+
import androidx.compose.material3.DropdownMenuItem
3028
import androidx.compose.material3.ExperimentalMaterial3Api
29+
import androidx.compose.material3.ExposedDropdownMenuDefaults
3130
import androidx.compose.material3.Icon
3231
import androidx.compose.material3.IconButton
3332
import androidx.compose.material3.Slider
3433
import androidx.compose.material3.Text
3534
import androidx.compose.material3.TextButton
35+
import androidx.compose.material3.TextField
3636
import androidx.compose.runtime.Composable
3737
import androidx.compose.runtime.getValue
3838
import androidx.compose.runtime.mutableFloatStateOf
@@ -77,7 +77,7 @@ fun ExposedSelectionMenu(
7777

7878
var expanded by remember { mutableStateOf(false) }
7979
var selectedOptionText by remember { mutableStateOf(options[index]) }
80-
80+
//目前material3的ExposedDropdownMenuBox使用有Bug无法弹窗,所以暂时用material 1.6.7的
8181
ExposedDropdownMenuBox(
8282
expanded = expanded,
8383
onExpandedChange = { expanded = expanded.not() },
@@ -86,39 +86,32 @@ fun ExposedSelectionMenu(
8686
.padding(vertical = 4.dp),
8787
) {
8888
TextField(
89+
value = selectedOptionText,
90+
onValueChange = {},
8991
modifier = Modifier.fillMaxWidth(),
9092
readOnly = true,
91-
value = selectedOptionText,
92-
onValueChange = { },
93-
label = { Text(title) },
9493
trailingIcon = {
9594
ExposedDropdownMenuDefaults.TrailingIcon(
9695
expanded = expanded
9796
)
9897
},
99-
colors = ExposedDropdownMenuDefaults.textFieldColors(
100-
backgroundColor = Color.White,
101-
focusedIndicatorColor = Color.Transparent,
102-
unfocusedIndicatorColor = Color.Transparent,
103-
disabledIndicatorColor = Color.Transparent,
104-
)
10598
)
99+
106100
ExposedDropdownMenu(
107101
expanded = expanded,
108102
onDismissRequest = { expanded = false },
109103
modifier = Modifier.fillMaxWidth(),
110104
) {
111105
options.forEachIndexed { index: Int, selectionOption: String ->
112106
DropdownMenuItem(
107+
text = { Text(text = selectionOption) },
113108
onClick = {
114109
selectedOptionText = selectionOption
115110
expanded = false
116111
onSelected(index)
117112
},
118113
modifier = Modifier.fillMaxWidth(),
119-
) {
120-
Text(text = selectionOption)
121-
}
114+
)
122115
}
123116
}
124117
}

0 commit comments

Comments
 (0)
Please sign in to comment.