selector 的 enterfadeduration 不生效,因其本质是 statelistdrawable,不支持该属性;需改用 animatedstatelistdrawable(api 21+)或 transitiondrawable 实现淡入淡出效果。

selector 的 enterFadeDuration 不生效?先确认是否用了 StateListDrawable
Android 的 selector XML 本质是 StateListDrawable,而 enterFadeDuration 是 TransitionDrawable 才支持的属性——它根本不在 StateListDrawable 的能力范围内。所以你写进去不会报错,但完全没效果。
常见错误现象:enterFadeDuration 写在 selector 根标签里,状态切换仍是瞬时跳变;Logcat 无报错,IDE 也不提示。
- 只有
TransitionDrawable(即用<transition></transition>标签定义的 drawable)才支持enterFadeDuration和exitFadeDuration -
selector只支持android:enterFadeDuration这个属性名,但仅当它被用作View的背景且该 View 启用了状态动画(如android:drawingCacheQuality="auto"并配合View.setState()触发)时,部分系统版本(API 21+)才可能间接起效——但这不是标准行为,不可依赖 - 如果你真需要淡入淡出切换,得换方案:要么用
TransitionDrawable+ 代码控制,要么用AnimatedStateListDrawable(API 21+)
想让 selector 状态切换带淡入效果,该用 AnimatedStateListDrawable
AnimatedStateListDrawable 是 Android 5.0(API 21)引入的、专为状态过渡动画设计的 drawable 类型。它允许你为不同状态对(比如 pressed → default)指定独立的过渡动画,包括淡入淡出。
使用场景:按钮按压反馈、开关 toggle、Tab 选中态变化等需要平滑状态过渡的 UI 元素。
- XML 文件需放在
res/drawable-v21/目录下(避免低版本崩溃),或运行时判断 API 版本兜底 - 每个
<item></item>必须有android:id,<transition></transition>通过android:fromId和android:toId关联状态 - 过渡动画本身是另一个 drawable(如
alpha动画或TransitionDrawable),不能直接写时长;时长由动画资源决定
示例片段(res/drawable-v21/btn_bg.xml):
<animated-selector xmlns:android="http://schemas.android.com/apk/res/android">
<item android:id="@+id/pressed" android:drawable="@drawable/btn_pressed" android:state_pressed="true" />
<item android:id="@+id/default" android:drawable="@drawable/btn_default" />
<transition android:fromId="@id/default" android:toId="@id/pressed">
<animation-list>
<item android:duration="150">
<alpha android:fromAlpha="1.0" android:toAlpha="0.3" />
</item>
</animation-list>
</transition>
</animated-selector>
低版本兼容:用代码 + TransitionDrawable 模拟 selector 淡入
如果必须支持 API TransitionDrawable,并监听 View 状态变化手动触发。
关键点在于:你无法让系统自动把 selector 替换成 TransitionDrawable,只能自己接管状态逻辑。
- 准备两个 drawable(如
default_bg和pressed_bg),构造TransitionDrawable:new TransitionDrawable(new Drawable[]{default_bg, pressed_bg}) - 设置淡入时长:
transition.setCrossFade(true)+transition.startTransition(200) - 监听
View.OnTouchListener或重写onCreateDrawableState()来感知状态变化,再调用startTransition() - 注意内存和重复触发:多次快速点击可能导致
startTransition()被反复调用,建议加标志位防抖
enterFadeDuration 在 TransitionDrawable XML 中怎么写才对
这是唯一真正支持 enterFadeDuration 的地方。但它不是写在 selector 里,而是写在 <transition></transition> 标签内,且只影响「进入」的淡入时间。
容易踩的坑:很多人把时长单位当成毫秒以外的值,或者误以为它能控制整个过渡周期。
-
enterFadeDuration和exitFadeDuration都是整数,单位是毫秒,必须写在<transition></transition>开始标签内,例如:<transition android:enterfadeduration="150" android:exitfadeduration="100"></transition> - 这两个值只控制 alpha 渐变节奏,不控制 drawable 切换时机;实际过渡是否“可见”,还取决于你是否调用了
startTransition() - 若未调用
startTransition(),哪怕设了时长也完全不动;若调用时传入的 duration 小于 XML 中的enterFadeDuration,则以代码参数为准
示例(res/drawable/transition_bg.xml):
<transition xmlns:android="http://schemas.android.com/apk/res/android"
android:enterFadeDuration="150"
android:exitFadeDuration="100">
<item android:drawable="@drawable/bg_default" />
<item android:drawable="@drawable/bg_active" />
</transition>
细节上最麻烦的其实是状态同步——XML 定义的淡入时长再准,只要代码里没在正确时机调用 startTransition(),或者没处理好按下/抬起/取消的边界,视觉效果就还是硬切。这点比写对标签难得多。










