我的seekBar样式是android Widget.AppCompat.SeekBar.Discrete。 我有自己的tickMarker,但你可以看到它显示在拇指标记前面,但我不想看到拇指背后的刻度。
和我拥有的:
我的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>
答案 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)
我在这里找到了一个好的图书馆:
答案 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>