如何在 Android 中获得可用的垂直 SeekBar?

问题描述 投票:0回答:10

我已经在here实现了常见的VerticalSeekBar帖子。实际上,SeekBar 的运行方式有点古怪。这是我根据该示例稍微改编的

onTouchEvent()

public boolean onTouchEvent(MotionEvent event)
    {
            xPos = event.getX();
            yPos = event.getY();
            oOffset = this.getThumbOffset();
            oProgress = this.getProgress();

            //Code from example - Not working
            //this.setThumbOffset( progress * (this.getBottom()-this.getTop()) );

            this.setProgress((int)(29*yPos/this.getBottom()));
            return true;
    }

我已经成功实现了一个 VerticalSeekBar,其中进度按预期更新并且功能齐全,但拇指不跟随。这只是一个图形故障,所以我现在忽略它。但是,如果能做到这一点就太好了。此 SeekBar 有 max = 20


但是,我尝试使用

max = 1000

实现另一个 VerticalSeekBar。显然,它使用相同的代码,因此您会假设相同的行为。即使我的手指滑过 SeekBar 并最终离开屏幕,我也只能取得 0~35 的进度。如果我只是点击进度条末端附近(

应该
progress ~ 900),它会返回大约 35 的进度,黄色进度条通过停留在顶部附近来反映该值。

我的问题是:是否有人有一个工作垂直 SeekBar 的链接,

知道如何适应这个特定的示例?

android seekbar
10个回答
154
投票

package android.widget; import android.content.Context; import android.graphics.Canvas; import android.util.AttributeSet; import android.view.MotionEvent; public class VerticalSeekBar extends SeekBar { public VerticalSeekBar(Context context) { super(context); } public VerticalSeekBar(Context context, AttributeSet attrs, int defStyle) { super(context, attrs, defStyle); } public VerticalSeekBar(Context context, AttributeSet attrs) { super(context, attrs); } protected void onSizeChanged(int w, int h, int oldw, int oldh) { super.onSizeChanged(h, w, oldh, oldw); } @Override protected synchronized void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { super.onMeasure(heightMeasureSpec, widthMeasureSpec); setMeasuredDimension(getMeasuredHeight(), getMeasuredWidth()); } protected void onDraw(Canvas c) { c.rotate(-90); c.translate(-getHeight(), 0); super.onDraw(c); } @Override public boolean onTouchEvent(MotionEvent event) { if (!isEnabled()) { return false; } switch (event.getAction()) { case MotionEvent.ACTION_DOWN: case MotionEvent.ACTION_MOVE: case MotionEvent.ACTION_UP: setProgress(getMax() - (int) (getMax() * event.getY() / getHeight())); onSizeChanged(getWidth(), getHeight(), 0, 0); break; case MotionEvent.ACTION_CANCEL: break; } return true; } }

要实现它,请在项目中创建一个新类,选择正确的包:

在那里粘贴代码并保存。现在在您的 XML 布局中使用它:

<android.widget.VerticalSeekBar android:id="@+id/seekBar1" android:layout_width="wrap_content" android:layout_height="200dp" />



31
投票

这是我的代码:

public class VerticalSeekBar extends SeekBar { private OnSeekBarChangeListener myListener; public VerticalSeekBar(Context context) { super(context); } public VerticalSeekBar(Context context, AttributeSet attrs, int defStyle) { super(context, attrs, defStyle); } public VerticalSeekBar(Context context, AttributeSet attrs) { super(context, attrs); } protected void onSizeChanged(int w, int h, int oldw, int oldh) { super.onSizeChanged(h, w, oldh, oldw); } @Override protected synchronized void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { super.onMeasure(heightMeasureSpec, widthMeasureSpec); setMeasuredDimension(getMeasuredHeight(), getMeasuredWidth()); } @Override public void setOnSeekBarChangeListener(OnSeekBarChangeListener mListener){ this.myListener = mListener; } protected void onDraw(Canvas c) { c.rotate(-90); c.translate(-getHeight(), 0); super.onDraw(c); } @Override public boolean onTouchEvent(MotionEvent event) { if (!isEnabled()) { return false; } switch (event.getAction()) { case MotionEvent.ACTION_DOWN: if(myListener!=null) myListener.onStartTrackingTouch(this); break; case MotionEvent.ACTION_MOVE: setProgress(getMax() - (int) (getMax() * event.getY() / getHeight())); onSizeChanged(getWidth(), getHeight(), 0, 0); myListener.onProgressChanged(this, getMax() - (int) (getMax() * event.getY() / getHeight()), true); break; case MotionEvent.ACTION_UP: myListener.onStopTrackingTouch(this); break; case MotionEvent.ACTION_CANCEL: break; } return true; } }



13
投票


11
投票

private int x,y,z,w; protected void onSizeChanged(int w, int h, int oldw, int oldh) { super.onSizeChanged(h, w, oldh, oldw); this.x=w; this.y=h; this.z=oldw; this.w=oldh; } @Override public synchronized void setProgress(int progress) { super.setProgress(progress); onSizeChanged(x, y, z, w); }

通过添加以下代码来执行选定的悬停操作:

1.setPressed(true);setSelected(true);//在ACTION_DOWN中添加这个

2.setPressed(false);setSelected(false);//在ACTION_UP中添加这个

并在 xml 中为选定的悬停选项编写代码。

<selector xmlns:android="http://schemas.android.com/apk/res/android"> <item android:state_pressed="true" android:state_window_focused="true" android:drawable="@drawable/thumb_h" /> <item android:state_selected="true" android:state_window_focused="true" android:drawable="@drawable/thumb_h" /> <item android:drawable="@drawable/thumb" /> </selector>

这对我有用......


9
投票

就我个人而言,我想要一个与给定代码相比是颠倒的垂直滑块。换句话说,拇指越低,该值就会增加而不是减少。改变四行似乎已经解决了这个问题。

首先,我更改了 Paul 最初给出的

onDraw()

方法。

rotate()
translate()
调用现在具有以下参数:

c.rotate(90); c.translate(0, -getWidth());

然后我对 Fatal1ty2787 给出的 
ACTION_MOVE

中的

onTouchEvent()
案例进行了两处更改。对
setProgress()
的调用现在如下所示:

setProgress((int) (getMax() * event.getY() / getHeight()));

最后,对 
onProgressChanged()

的调用如下所示:


myListener.onProgressChanged(this, (int) (getMax() * event.getY() / getHeight()), true);

现在,如果 Google 能分享我们对这个功能的兴趣就好了……


9
投票
API 11 and later

,可以使用

seekbar's XML attributes
(
android:rotation="270"
)来实现垂直效果。

<SeekBar android:id="@+id/seekBar1" android:layout_width="match_parent" android:layout_height="wrap_content" android:rotation="270"/>

对于较旧的 API 级别(例如 API10),请使用:
https://github.com/AndroSelva/Vertical-SeekBar-Android


2
投票

public class VerticalSeekBar extends SeekBar { public VerticalSeekBar(Context context) { super(context); } public VerticalSeekBar(Context context, AttributeSet attrs, int defStyle) { super(context, attrs, defStyle); } public VerticalSeekBar(Context context, AttributeSet attrs) { super(context, attrs); } @Override public boolean onTouchEvent(MotionEvent event) { if (!isEnabled()) { return false; } float x = (getHeight() - event.getY()) * getWidth() / getHeight(); float y = event.getX(); MotionEvent verticalEvent = MotionEvent .obtain(event.getDownTime(), event.getEventTime(), event.getAction(), x, y, event.getPressure(), event.getSize(), event.getMetaState(), event.getYPrecision(), event.getXPrecision(), event.getDeviceId(), event.getEdgeFlags()); return super.onTouchEvent(verticalEvent); } protected void onDraw(Canvas c) { c.rotate(-90); c.translate(-getHeight(), 0); super.onDraw(c); } @Override protected synchronized void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { super.onMeasure(heightMeasureSpec, widthMeasureSpec); setMeasuredDimension(getMeasuredHeight(), getMeasuredWidth()); } protected void onSizeChanged(int w, int h, int oldw, int oldh) { super.onSizeChanged(h, w, oldh, oldw); } }

在这种情况下,无需调用 setProgress(int) 方法,因此您可以在 OnSeekBarChangeListener.onProgressChanged() 中使用布尔标志“fromUser”来确定查找是否由用户交互产生。


2
投票
AppCompatVerticalSeekBar


package com.my.apppackage; import android.annotation.SuppressLint; import android.content.Context; import android.graphics.Canvas; import android.util.AttributeSet; import android.view.MotionEvent; import androidx.appcompat.widget.AppCompatSeekBar; public class AppCompatVerticalSeekBar extends AppCompatSeekBar { public AppCompatVerticalSeekBar(Context context) { super(context); } public AppCompatVerticalSeekBar(Context context, AttributeSet attrs) { super(context, attrs); } public AppCompatVerticalSeekBar(Context context, AttributeSet attrs, int defStyleAttr) { super(context, attrs, defStyleAttr); } @Override protected void onSizeChanged(int w, int h, int oldw, int oldh) { super.onSizeChanged(h, w, oldh, oldw); } @Override protected synchronized void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { super.onMeasure(heightMeasureSpec, widthMeasureSpec); setMeasuredDimension(getMeasuredHeight(), getMeasuredWidth()); } @Override protected synchronized void onDraw(Canvas canvas) { canvas.rotate(-90); canvas.translate(-getHeight(), 0); super.onDraw(canvas); } @SuppressLint("ClickableViewAccessibility") @Override public boolean onTouchEvent(MotionEvent event) { if (!isEnabled()) { return false; } switch (event.getAction()) { case MotionEvent.ACTION_DOWN: case MotionEvent.ACTION_MOVE: case MotionEvent.ACTION_UP: { setProgress(getMax() - (int) (getMax() * event.getY() / getHeight())); onSizeChanged(getWidth(), getHeight(), 0, 0); break; } case MotionEvent.ACTION_CANCEL: { break; } } return true; } }

在布局文件中使用它:

<com.my.apppackage.AppCompatVerticalSeekBar android:id="@+id/verticalSeekBar1" android:layout_width="wrap_content" android:layout_height="200dp" />



0
投票
目标平台

来自 Android 2.3.x(姜饼) 至 Android 7.x(牛轧糖)

开始使用

该库发布在jCenter上。只需将这些行添加到 build.gradle 即可。

dependencies { compile 'com.h6ah4i.android.widget.verticalseekbar:verticalseekbar:0.7.2' }

用法

布局 XML

<!-- This library requires pair of the VerticalSeekBar and VerticalSeekBarWrapper classes --> <com.h6ah4i.android.widget.verticalseekbar.VerticalSeekBarWrapper android:layout_width="wrap_content" android:layout_height="150dp"> <com.h6ah4i.android.widget.verticalseekbar.VerticalSeekBar android:id="@+id/mySeekBar" android:layout_width="0dp" android:layout_height="0dp" android:max="100" android:progress="0" android:splitTrack="false" app:seekBarRotation="CW90" /> <!-- Rotation: CW90 or CW270 --> </com.h6ah4i.android.widget.verticalseekbar.VerticalSeekBarWrapper>

注意:Android N+ 需要
android:splitTrack="false"


Java代码

public class TestVerticalSeekbar extends AppCompatActivity { private SeekBar volumeControl = null; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_test_vertical_seekbar); volumeControl = (SeekBar) findViewById(R.id.mySeekBar); volumeControl.setOnSeekBarChangeListener(new SeekBar.OnSeekBarChangeListener() { int progressChanged = 0; public void onProgressChanged(SeekBar seekBar, int progress, boolean fromUser) { progressChanged = progress; } public void onStartTrackingTouch(SeekBar seekBar) { // TODO Auto-generated method stub } public void onStopTrackingTouch(SeekBar seekBar) { Toast.makeText(getApplicationContext(), "seek bar progress:" + progressChanged, Toast.LENGTH_SHORT).show(); } }); } }



0
投票
SeekBar

,而是在寻找垂直

Slider
。这是我在 Kotlin 中实现的垂直滑块。
import android.content.Context
import android.graphics.Canvas
import android.util.AttributeSet
import android.view.MotionEvent
import com.google.android.material.slider.Slider

class MySlider : Slider {

    constructor(context: Context) : super(context)
    constructor (context: Context, attrs: AttributeSet?) : super(context, attrs) 

    override fun onSizeChanged(w: Int, h: Int, oldw: Int, oldh: Int) {
        super.onSizeChanged(h, w, oldh, oldw)
    }

    @Synchronized
    override fun onMeasure(widthMeasureSpec: Int, heightMeasureSpec: Int) {
        super.onMeasure(heightMeasureSpec, widthMeasureSpec)
        setMeasuredDimension(measuredHeight, measuredWidth)
    }

    override fun setValue(value: Float) {
        super.setValue(value)
        onSizeChanged(width, height, 0, 0)
    }

    override fun onDraw(canvas: Canvas) {
        canvas.rotate(-90f)
        canvas.translate(-height * 1f, 0f)
        super.onDraw(canvas)
    }

    override fun onTouchEvent(event: MotionEvent): Boolean {
        if (!isEnabled) {
            return false
        }
        when (event.action) {
            MotionEvent.ACTION_DOWN, MotionEvent.ACTION_MOVE, MotionEvent.ACTION_UP -> {
                setValue((valueTo - (valueTo * event.y / height)).coerceIn(valueFrom, valueTo))
                onSizeChanged(width, height, 0, 0)
            }

            MotionEvent.ACTION_CANCEL -> {}
        }
        return true
    }
}

© www.soinside.com 2019 - 2024. All rights reserved.