为什么在拇指前面显示了seekbar tick标记?

时间:2017-01-05 16:45:16

标签: android android-styles android-seekbar

我的seekBar样式是android Widget.AppCompat.SeekBar.Discrete。 我有自己的tickMarker,但你可以看到它显示在拇指标记前面,但我不想看到拇指背后的刻度。

我想要的: enter image description here

和我拥有的:

enter image description here

我的XML:

 <android.support.v7.widget.AppCompatSeekBar
    style="@style/seekbarStyle"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:max="4"
    android:padding="4dp"
    android:progress="0"/>

我的风格:

<style name="seekbarStyle" parent="Widget.AppCompat.SeekBar.Discrete">
    <item name="tickMark">@drawable/seekbar_tickmark</item>
    <item name="android:thumb">@drawable/circle</item>
</style>

4 个答案:

答案 0 :(得分:8)

这是AppCompatSeekBar的错误。 我使用扩展AppCompatSeekBar的自定义类解决了这个问题:

public class CustomSeekBar extends AppCompatSeekBar {

    private Drawable mTickMark;

    public CustomSeekBar(Context context) {
        this(context, null);
    }

    public CustomSeekBar(Context context, AttributeSet attrs) {
        this(context, attrs, android.support.v7.appcompat.R.attr.seekBarStyle);
    }

    public CustomSeekBar(Context context, AttributeSet attrs, int defStyleAttr) {
        super(context, attrs, defStyleAttr);
        applyAttributes(attrs, defStyleAttr);
    }

    private void applyAttributes(AttributeSet rawAttrs, int defStyleAttr)
    {
        TypedArray attrs = getContext().obtainStyledAttributes(rawAttrs, R.styleable.CustomSeekBar, defStyleAttr, 0);
        try {
            mTickMark = attrs.getDrawable(R.styleable.CustomSeekBar_tickMarkFixed);
        } finally {
            attrs.recycle();
        }
    }

    @Override
    protected synchronized void onDraw(Canvas canvas) {
        super.onDraw(canvas);
        drawTickMarks(canvas);
    }

    @Override
    public int getThumbOffset() {
        return super.getThumbOffset();
    }

    void drawTickMarks(Canvas canvas) {
        if (mTickMark != null) {
            final int count = getMax();
            if (count > 1) {
                final int w = mTickMark.getIntrinsicWidth();
                final int h = mTickMark.getIntrinsicHeight();
                final int halfThumbW = getThumb().getIntrinsicWidth() / 2;
                final int halfW = w >= 0 ? w / 2 : 1;
                final int halfH = h >= 0 ? h / 2 : 1;
                mTickMark.setBounds(-halfW, -halfH, halfW, halfH);
                final float spacing = (getWidth() - getPaddingLeft() - getPaddingRight() + getThumbOffset() * 2 - halfThumbW * 2) / (float) count;
                final int saveCount = canvas.save();
                canvas.translate(getPaddingLeft() - getThumbOffset() + halfThumbW, getHeight() / 2);
                for (int i = 0; i <= count; i++) {
                    if(i!=getProgress())
                        mTickMark.draw(canvas);
                    canvas.translate(spacing, 0);
                }
                canvas.restoreToCount(saveCount);
            }
        }
    }
}

使用attrs.xml:

<resources>
    <declare-styleable name="CustomSeekBar">
        <attr name="tickMarkFixed" format="reference"/>
    </declare-styleable>
</resources>

在布局中,您必须使用 tickMarkFixed 而不是 tickMark

答案 1 :(得分:1)

似乎是问题所在,AppCompatSeekBar小部件调用了super,它绘制了一个拇指,然后在其上绘制了刻度。

这里是一个Kotlin类,它通过在画布上重画拇指来解决此问题(此时,拇指先有拇指,然后在其上绘制刻度线:

class SeekBar @JvmOverloads constructor(
    context: Context,
    attrs: AttributeSet? = null,
    defStyleAttr: Int = android.R.attr.seekBarStyle
) : AppCompatSeekBar(context, attrs, defStyleAttr) {
    override fun onDraw(canvas: Canvas) {
        super.onDraw(canvas)
        drawThumb(canvas)
    }

    private fun drawThumb(canvas: Canvas) {
        if (thumb != null) {
            val saveCount = canvas.save()
            canvas.translate((paddingLeft - thumbOffset).toFloat(), paddingTop.toFloat())
            thumb.draw(canvas)
            canvas.restoreToCount(saveCount)
        }
    }
}

注意:此解决方案可能会留下一些伪像,因为拇指被绘制了两次。一种解决方法是定义一个辅助拇指,并在隐藏原始拇指的同时使用它。

答案 2 :(得分:0)

我在这里找到了一个好的图书馆:

IndicatorSeekBar

答案 3 :(得分:0)

我将在 kotlin 中共享此修复程序,以寻求更好的解决方案并避免在拇指使用前打勾:

    if (i > progress)

完整代码:

    class CustomSeekBar @JvmOverloads constructor(
    context: Context,
    attrs: AttributeSet? = null,
    defStyle: Int = 0) : AppCompatSeekBar(context, attrs, defStyle) {
private var mTickMark: Drawable? = null

init {
    applyAttributes(attrs, defStyle)
}

private fun applyAttributes(rawAttrs: AttributeSet?, defStyleAttr: Int) {
    val attrs = context.obtainStyledAttributes(rawAttrs, R.styleable.CustomSeekBar, defStyleAttr, 0)
    try {
        mTickMark = attrs.getDrawable(R.styleable.CustomSeekBar_customTickMark)
    } finally {
        attrs.recycle()
    }
}


override fun onDraw(canvas: Canvas) {
    super.onDraw(canvas)
    drawTickMarks(canvas)
}


private fun drawTickMarks(canvas: Canvas) {
    if (mTickMark != null) {
        val count = max

        val w = mTickMark!!.intrinsicWidth
        val h = mTickMark!!.intrinsicHeight
        val halfThumbW = thumb.intrinsicWidth / 2
        val halfW = if (w >= 0) w / 2 else 1
        val halfH = if (h >= 0) h / 2 else 1
        mTickMark!!.setBounds(-halfW, -halfH, halfW, halfH)
        val spacing = (width - paddingLeft - paddingRight + thumbOffset * 2 - halfThumbW * 2) / count.toFloat()
        val saveCount = canvas.save()
        canvas.translate((paddingLeft - thumbOffset + halfThumbW).toFloat(), (height / 2).toFloat())
        for (i in 0..count) {
            if (i > progress)
                mTickMark!!.draw(canvas)
            canvas.translate(spacing, 0F)
        }
        canvas.restoreToCount(saveCount)
    }
}
}

记住要在 res / values 中的 attrs.xml 文件中创建属性,如下所示:

<?xml version="1.0" encoding="utf-8"?>
<resources>
    <declare-styleable name="CustomSeekBar">
        <attr name="customTickMark" format="integer" />
    </declare-styleable>
</resources>