Android selector enterFadeDuration XML设置状态切换淡入时间

1次阅读

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

Android selector enterFadeDuration XML设置状态切换淡入时间

selector 的 enterFadeDuration 不生效?先确认是否用了 StateListDrawable

androidselector xml 本质是 StateListDrawable,而 enterFadeDurationTransitionDrawable 才支持的属性——它根本不在 StateListDrawable 的能力范围内。所以你写进去不会报错,但完全没效果。

常见错误现象:enterFadeDuration 写在 selector 根标签里,状态切换仍是瞬时跳变;Logcat 无报错,ide 也不提示。

  • 只有 TransitionDrawable(即用 <transition></transition> 标签定义的 drawable)才支持 enterFadeDurationexitFadeDuration
  • 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:fromIdandroid: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_bgpressed_bg),构造 TransitionDrawablenew TransitionDrawable(new Drawable[]{default_bg, pressed_bg})
  • 设置淡入时长:transition.setCrossFade(true) + transition.startTransition(200)
  • 监听 View.OnTouchListener 或重写 onCreateDrawableState() 来感知状态变化,再调用 startTransition()
  • 注意内存和重复触发:多次快速点击可能导致 startTransition() 被反复调用,建议加标志位防抖

enterFadeDurationTransitionDrawable XML 中怎么写才对

这是唯一真正支持 enterFadeDuration 的地方。但它不是写在 selector 里,而是写在 <transition></transition> 标签内,且只影响「进入」的淡入时间。

容易踩的坑:很多人把时长单位当成毫秒以外的值,或者误以为它能控制整个过渡周期。

  • enterFadeDurationexitFadeDuration 都是整数,单位是毫秒,必须写在 <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(),或者没处理好按下/抬起/取消的边界,视觉效果就还是硬切。这点比写对标签难得多。

text=ZqhQzanResources