XML Shape/Selector → Kotlin 动态创建

XML Shape/Selector → Kotlin 动态创建

老写法(XML drawable)

res/drawable/bg_rounded_button.xml

xml 复制代码
<shape xmlns:android="http://schemas.android.com/apk/res/android">
    <solid android:color="#4CAF50"/>
    <corners android:radius="8dp"/>
    <stroke android:width="1dp" android:color="#388E3C"/>
</shape>

res/drawable/selector_button.xml

xml 复制代码
<selector xmlns:android="http://schemas.android.com/apk/res/android">
    <item android:state_pressed="true">
        <shape>
            <solid android:color="#388E3C"/>
            <corners android:radius="8dp"/>
        </shape>
    </item>
    <item>
        <shape>
            <solid android:color="#4CAF50"/>
            <corners android:radius="8dp"/>
        </shape>
    </item>
</selector>

Java 中使用:

java 复制代码
view.setBackgroundResource(R.drawable.selector_button);

问题在哪里

XML drawable 有三个局限:颜色写死不能动态改、不同状态下要写多份 shape 重复代码、需要在 res/drawable 里维护一堆小文件。

新写法(Kotlin 动态创建)

kotlin 复制代码
fun roundedButtonBg(
    context: Context,
    normalColor: Int,
    pressedColor: Int,
    cornerRadiusDp: Int,
    strokeColor: Int,
    strokeWidthDp: Int = 1
): StateListDrawable {
    val density = context.resources.displayMetrics.density
    val radius = cornerRadiusDp * density
    val strokeWidth = strokeWidthDp * density

    val normal = GradientDrawable().apply {
        shape = GradientDrawable.RECTANGLE
        setColor(normalColor)
        cornerRadius = radius
        setStroke(strokeWidth.toInt(), strokeColor)
    }

    val pressed = GradientDrawable().apply {
        shape = GradientDrawable.RECTANGLE
        setColor(pressedColor)
        cornerRadius = radius
        setStroke(strokeWidth.toInt(), strokeColor)
    }

    return StateListDrawable().apply {
        addState(intArrayOf(android.R.attr.state_pressed), pressed)
        addState(intArrayOf(), normal)
    }
}

使用:

kotlin 复制代码
view.background = roundedButtonBg(
    context = context,
    normalColor = Color.parseColor("#4CAF50"),
    pressedColor = Color.parseColor("#388E3C"),
    cornerRadiusDp = 8,
    strokeColor = Color.parseColor("#388E3C")
)

一句话注意

GradientDrawable.cornerRadiussetStroke 的参数单位是像素,不是 dp。传 dp 值必须乘以 density,否则不同分辨率手机上圆角大小不一致。StateListDrawable.addState 的状态数组顺序很重要,越具体的状态(如 state_pressed)要放在前面,默认状态(空数组)放最后。


Java Android 老项目迁移系列,持续更新中。

相关推荐
用户128526116021 小时前
我把祖传Java项目重构后,接口响应从3s砍到了200ms,只改了这几行代码
java
通玄1 小时前
Jetpack Compose 入门系列(六):Navigation 3 页面导航
android
Linsk1 小时前
组件 = 模板 + 业务逻辑
java·前端·vue.js
星沉远浦2 小时前
用Gemini高效解决Java代码报错难以定位的问题
java
rocpp4 小时前
Android 多语言切换实战:从 Context 到 Android 13 应用语言适配
android·kotlin
释然小师弟4 小时前
Android开发十年:反思与回顾
android·后端·嵌入式
用户298698530145 小时前
Word 文档字符级格式化:Java 实现方案详解
java·后端
笨鸟飞不快6 小时前
从单个服务到集群:一次完整的性能排查复盘
java·前端
荣码6 小时前
用Streamlit给AI应用套个界面,10行代码出Web页面
java·python
SamDeepThinking6 小时前
Java微服务练习方式
java·后端·微服务