我正在创建一些日历视图,我想做的是为ClickAbe的LinearLayout创建一个背景。
因此,我创建了一个带有两个图像的stateListDrawable:
背景图像
按下项目时的图像
到目前为止,这段代码的作用是:

    NinePatchDrawable background = (NinePatchDrawable) context.getResources().getDrawable(R.drawable.calendar_item);
    Drawable backgroundFocus = context.getResources().getDrawable(R.drawable.calendar_focus);

    int stateFocused = android.R.attr.state_focused;
    int statePressed = android.R.attr.state_pressed;

    StateListDrawable sld = new StateListDrawable();
    sld.addState(new int[]{ stateFocused,  statePressed}, backgroundFocus);
    sld.addState(new int[]{-stateFocused,  statePressed}, backgroundFocus);
    sld.addState(new int[]{-stateFocused}, background);
    return sld;

但我想做些额外的事。我希望用户能够传递一种他想用来显示背景的颜色。因此背景变量必须是可变的,但它必须基于九个可绘制的补丁。
所以我想我可以这样做:
background.setColorFilter(Color.RED, PorterDuff.Mode.DST_IN);

其中color.red必须替换为用户选择的颜色。
但这似乎行不通。九个补丁创建得很完美,但没有应用颜色过滤器。
我还尝试了其他的poterduff模式:
SRC
斯科塔托普
德斯汀

如果你知道我做错了什么,或者我还能做什么来解决我的问题,请告诉我!-)
Kr
德克

最佳答案

我不认为您可以为stateListDrawable中的每个Drawable分配colorFilters。原因:当stateListDrawable更改状态时,ColorFilter将被删除/替换。要看到这一点,请更改语句的顺序,以便:

background.setColorFilter(Color.RED, PorterDuff.Mode.DST_IN);

在创建stateListDrawable之后。您将看到应用了colorfilter。但是,一旦状态改变(单击,然后释放),colorfilter就不再存在了。
stateListDrawables允许您设置colorFilter:StateListDrawable#setColorFilter(ColorFilter)。以下是使用提供的(或空)colorfilter的方法:
stateListDrawable onStateChange(int[]):
@Override
protected boolean onStateChange(int[] stateSet) {
    ....
    if (selectDrawable(idx)) {    // DrawableContainer#selectDrawable(int)
        return true;
    }
    ....
}

可提取容器可选择可提取(int):
public boolean selectDrawable(int idx) {
    ....
    if (idx >= 0 && idx < mDrawableContainerState.mNumChildren) {
        Drawable d = mDrawableContainerState.mDrawables[idx];
        mCurrDrawable = d;
        mCurIndex = idx;

        if (d != null) {
            ....

            // So, at this stage, any ColorFilter you might have supplied
            // to `d` will be replaced by the ColorFilter you
            // supplied to the StateListDrawable, or `null`
            // if you didn't supply any.
            d.setColorFilter(mColorFilter);

            ....
        }
    } else {
        ....
    }
}

解决办法:
如果可能的话,可以使用imageview(将父视图与维度匹配)进行视觉交流。设置您创建的stateListDrawable作为ImageView的背景。为覆盖创建另一个StateListDrawable:
StateListDrawable sldOverlay = new StateListDrawable();

// Match Colors with states (and ultimately, Drawables)
sldOverlay.addState(new int[] { statePressed },
                         new ColorDrawable(Color.TRANSPARENT));

sldOverlay.addState(new int[] { -statePressed },
                         new ColorDrawable(Color.parseColor("#50000000")));

// Drawable that you already have
iv1.setBackground(sld);

// Drawable that you just created
iv1.setImageDrawable(sldOverlay);

另一种可能性:使用框架布局代替线性布局。LinearLayouts没有前台属性。
// StateListDrawable
frameLayout.setBackground(sld);

// For tint
frameLayout.setForeground(sldOverlay);

它确实涉及透支,使其成为次优解决方案/解决方案。也许您可以看看扩展stateListDrawable和DrawableContainer。由于您没有为stateListDrawable使用colorFilter,因此可以从overridedd.setColorFilter(mColorFilter);中删除DrawableContainer#selectDrawable(int)

07-26 02:02