更改 TimePicker、DatePicker 的 Spinner 样式默认分隔线颜色

3,158 阅读1分钟

从源码看,DatePicker 和 TimePicker 都是一个 FrameLayout,有两种布局,一种是 Material Design 的时钟样式,一种是下图的 Spinner 样式,如图(DatePicker):

在 xml 文件中(date_picker_legacy、time_picker_legacy)可以看到 DatePicker 和 TimePicker 是多个 NumberPicker 构成的,Spinner 中间都是蓝色的分割线,但并没有提供方法可以进行颜色的修改,因此可以考虑使用反射的方法来修改分割线颜色。下面通过 TimePicker 为例说明一下修改流程:

1. 获取系统资源,拿到 TimePicker 中的 NumberPicker 资源 id:

Resources systemResources = Resources.getSystem();
int hourNumberPickerId = systemResources.getIdentifier("hour", "id", "android");
int minuteNumberPickerId = systemResources.getIdentifier("minute", "id", "android");
// 12小时格式还有一条上下午的分割线。
int amPmId = systemResources.getIdentifier("amPm", "id", "android");

注意:这里如果是 DatePicker,则 hour、minute 改为 month、day、year

2. 通过资源 id 找到 TimePicker 中的 NumberPicker:

NumberPicker hourNumberPicker = (NumberPicker) timePicker.findViewById(hourNumberPickerId);
NumberPicker minuteNumberPicker = (NumberPicker) timePicker.findViewById(minuteNumberPickerId);

3. 利用反射修改 NumberPicker 中的颜色,分割线颜色是由 NumberPicker 中的变量 mSelectionDivider 保存的

private void setNumberPickerDivider(NumberPicker numberPicker) {
    final int count = numberPicker.getChildCount();
    for(int i = 0; i < count; i++){
        try{
            Field dividerField = numberPicker.getClass().getDeclaredField("mSelectionDivider");
            dividerField.setAccessible(true);
            ColorDrawable colorDrawable = new ColorDrawable(
                    ContextCompat.getColor(this, android.R.color.holo_red_light));
            dividerField.set(numberPicker,colorDrawable);
            numberPicker.invalidate();
         }
        catch(NoSuchFieldException | IllegalAccessException | IllegalArgumentException e){
            Log.w("setNumberPickerTxtClr", e);
        }
    }
}

调用上面方法,将获得的numberPicker作为参数传入即可。