Android4.0以上的DatePicker,时间选择器,想必大家不会陌生,如下所示效果

今天我们仿照DatePicker的效果,来实现自定义的WheelView,它可以实现选择年龄、身高、体重。实现后的效果如下

 

下面是代Android竖直滑动选择器WheelView的实现_第1张图片码实现的过程

1、MainActivity的xml布局文件

<?xml version="1.0" encoding="utf-8"?>            

2、主要方法放在MainActivity中

package test.qywang.com.wheelviewtest;import android.app.AlertDialog;import android.os.Bundle;import android.support.v7.app.AppCompatActivity;import android.view.LayoutInflater;import android.view.View;import android.widget.TextView;import java.util.ArrayList;public class MainActivity extends AppCompatActivity {    private TextView tvAge;    private TextView tvHeight;    private TextView tvWeight;    private String selectText = "";    private ArrayList heightList = new ArrayList<>();    private ArrayList weightList = new ArrayList<>();    private ArrayList ageList = new ArrayList<>();    @Override    protected void onCreate(Bundle savedInstanceState) {        super.onCreate(savedInstanceState);        setContentView(R.layout.activity_main);        initView();        initData();        initListeners();    }    private void initData() {        // 填充列表        ageList.clear();        heightList.clear();        weightList.clear();        for (int i = 10; i <= 50; i++) {            ageList.add(String.format("%d岁", i));        }        for (int i = 120; i <= 200; i++) {            heightList.add(String.format("%dcm", i));        }        for (int i = 40; i <= 110; i++) {            weightList.add(String.format("%d公斤", i));        }    }    private void initView() {        tvAge = findViewById(R.id.tv_age);        tvHeight = findViewById(R.id.tv_height);        tvWeight = findViewById(R.id.tv_weight);    }    private void initListeners(){        findViewById(R.id.tv_age).setOnClickListener(view ->showDialog(tvAge, ageList, 8)        );        findViewById(R.id.tv_height).setOnClickListener(view -> showDialog(tvHeight,heightList,40));        findViewById(R.id.tv_weight).setOnClickListener(view -> showDialog(tvWeight,weightList,10));    }    private void showDialog(TextView textView, ArrayList list, int selected){        showChoiceDialog(list, textView, selected,                new WheelView.OnWheelViewListener() {                    @Override                    public void onSelected(int selectedIndex, String item) {                        selectText = item;                    }                });    }    private void showChoiceDialog(ArrayList dataList,final TextView textView,int selected,                                  WheelView.OnWheelViewListener listener){        selectText = "";        View outerView = LayoutInflater.from(this).inflate(R.layout.dialog_wheelview,null);        final WheelView wheelView = outerView.findViewById(R.id.wheel_view);        wheelView.setOffset(2);// 对话框中当前项上面和下面的项数        wheelView.setItems(dataList);// 设置数据源        wheelView.setSeletion(selected);// 默认选中第三项        wheelView.setOnWheelViewListener(listener);        // 显示对话框,点击确认后将所选项的值显示到Button上        AlertDialog alertDialog = new AlertDialog.Builder(this)                .setView(outerView)                .setPositiveButton("确认",                        (dialogInterface, i) -> {                          textView.setText(selectText);                          textView.setTextColor(this.getResources().getColor(R.color.color_333333));                        })                .setNegativeButton("取消",null).create();        alertDialog.show();        int green = this.getResources().getColor(R.color.green);        alertDialog.getButton(AlertDialog.BUTTON_POSITIVE).setTextColor(green);        alertDialog.getButton(AlertDialog.BUTTON_NEGATIVE).setTextColor(green);    }}

3、自定义WheelView类

package test.qywang.com.wheelviewtest;import android.content.Context;import android.graphics.Canvas;import android.graphics.Color;import android.graphics.ColorFilter;import android.graphics.Paint;import android.graphics.PixelFormat;import android.graphics.drawable.Drawable;import android.util.AttributeSet;import android.util.Log;import android.util.TypedValue;import android.view.Gravity;import android.view.MotionEvent;import android.view.View;import android.view.ViewGroup;import android.widget.LinearLayout;import android.widget.ScrollView;import android.widget.TextView;import java.util.ArrayList;import java.util.List;public class WheelView extends ScrollView {    public static final String TAG = WheelView.class.getSimpleName();    public static class OnWheelViewListener {        public void onSelected(int selectedIndex, String item) {        }    }    private Context context;    private LinearLayout views;    public WheelView(Context context) {        super(context);        init(context);    }    public WheelView(Context context, AttributeSet attrs) {        super(context, attrs);        init(context);    }    public WheelView(Context context, AttributeSet attrs, int defStyle) {        super(context, attrs, defStyle);        init(context);    }    List items;    private List getItems() {        return items;    }    public void setItems(List list) {        if (null == items) {            items = new ArrayList();        }        items.clear();        items.addAll(list);        // 前面和后面补全        for (int i = 0; i < offset; i++) {            items.add(0, "");            items.add("");        }        initData();    }    public static final int OFF_SET_DEFAULT = 1;    int offset = OFF_SET_DEFAULT; // 偏移量(需要在最前面和最后面补全)    public int getOffset() {        return offset;    }    public void setOffset(int offset) {        this.offset = offset;    }    int displayItemCount; // 每页显示的数量    int selectedIndex = 1;    private void init(Context context) {        this.context = context;        Log.d(TAG, "parent: " + this.getParent());        this.setVerticalScrollBarEnabled(false);        views = new LinearLayout(context);        views.setOrientation(LinearLayout.VERTICAL);        this.addView(views);        scrollerTask = new Runnable() {            public void run() {                int newY = getScrollY();                if (initialY - newY == 0) { // stopped                    final int remainder = initialY % itemHeight;                    final int divided = initialY / itemHeight;                    if (remainder == 0) {                        selectedIndex = divided + offset;                        onSeletedCallBack();                    } else {                        if (remainder > itemHeight / 2) {                            WheelView.this.post(new Runnable() {                                @Override                                public void run() {                                    WheelView.this.smoothScrollTo(0, initialY - remainder + itemHeight);                                    selectedIndex = divided + offset + 1;                                    onSeletedCallBack();                                }                            });                        } else {                            WheelView.this.post(new Runnable() {                                @Override                                public void run() {                                    WheelView.this.smoothScrollTo(0, initialY - remainder);                                    selectedIndex = divided + offset;                                    onSeletedCallBack();                                }                            });                        }                    }                } else {                    initialY = getScrollY();                    WheelView.this.postDelayed(scrollerTask, newCheck);                }            }        };        // 默认初始不滑动时执行一次回调        if (null != onWheelViewListener) {            onWheelViewListener.onSelected(selectedIndex, items.get(selectedIndex));        }    }    int initialY;    Runnable scrollerTask;    int newCheck = 50;    public void startScrollerTask() {        initialY = getScrollY();        this.postDelayed(scrollerTask, newCheck);    }    private void initData() {        displayItemCount = offset * 2 + 1;        views.removeAllViews();        for (String item : items) {            views.addView(createView(item));        }        refreshItemView(0);    }    int itemHeight = 0;    private TextView createView(String item) {        TextView tv = new TextView(context);        tv.setLayoutParams(new LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT, ViewGroup.LayoutParams.WRAP_CONTENT));        tv.setSingleLine(true);        tv.setTextSize(TypedValue.COMPLEX_UNIT_SP, 20);        tv.setText(item);        tv.setGravity(Gravity.CENTER);        int padding = dip2px(15);        tv.setPadding(padding, padding, padding, padding);        if (0 == itemHeight) {            itemHeight = getViewMeasuredHeight(tv);            Log.d(TAG, "itemHeight: " + itemHeight);            views.setLayoutParams(new LayoutParams(ViewGroup.LayoutParams.WRAP_CONTENT, itemHeight * displayItemCount, Gravity.CENTER_HORIZONTAL));            views.setGravity(Gravity.CENTER);            LinearLayout.LayoutParams lp = (LinearLayout.LayoutParams) this.getLayoutParams();            this.setLayoutParams(new LinearLayout.LayoutParams(lp.width, itemHeight * displayItemCount));        }        return tv;    }    @Override    protected void onScrollChanged(int l, int t, int oldl, int oldt) {        super.onScrollChanged(l, t, oldl, oldt);        refreshItemView(t);        if (t > oldt) {//            Log.d(TAG, "向下滚动");            scrollDirection = SCROLL_DIRECTION_DOWN;        } else {//            Log.d(TAG, "向上滚动");            scrollDirection = SCROLL_DIRECTION_UP;        }    }    private void refreshItemView(int y) {        int position = y / itemHeight + offset;        int remainder = y % itemHeight;        int divided = y / itemHeight;        if (remainder == 0) {            position = divided + offset;        } else {            if (remainder > itemHeight / 2) {                position = divided + offset + 1;            }        }        int childSize = views.getChildCount();        for (int i = 0; i < childSize; i++) {            TextView itemView = (TextView) views.getChildAt(i);            if (null == itemView) {                return;            }            if (position == i) {                itemView.setTextColor(context.getResources().getColor(R.color.green));            } else {                itemView.setTextColor(context.getResources().getColor(R.color.color_999999));            }        }    }    /**     * 获取选中区域的边界     */    int[] selectedAreaBorder;    private int[] obtainSelectedAreaBorder() {        if (null == selectedAreaBorder) {            selectedAreaBorder = new int[2];            selectedAreaBorder[0] = itemHeight * offset;            selectedAreaBorder[1] = itemHeight * (offset + 1);        }        return selectedAreaBorder;    }    private int scrollDirection = -1;    private static final int SCROLL_DIRECTION_UP = 0;    private static final int SCROLL_DIRECTION_DOWN = 1;    Paint paint;    int viewWidth;    @Override    public void setBackgroundDrawable(Drawable background) {        if (viewWidth == 0) {//            viewWidth = ((Activity) context).getWindowManager().getDefaultDisplay().getWidth();            viewWidth = views.getWidth();            Log.d(TAG, "viewWidth: " + viewWidth);        }        if (null == paint) {            paint = new Paint();            paint.setColor(Color.parseColor("#83cde6"));            paint.setStrokeWidth(dip2px(1f));        }        background = new Drawable() {            @Override            public void draw(Canvas canvas) {                canvas.drawLine(viewWidth * 1 / 6, obtainSelectedAreaBorder()[0], viewWidth * 5 / 6, obtainSelectedAreaBorder()[0], paint);                canvas.drawLine(viewWidth * 1 / 6, obtainSelectedAreaBorder()[1], viewWidth * 5 / 6, obtainSelectedAreaBorder()[1], paint);            }            @Override            public void setAlpha(int alpha) {            }            @Override            public void setColorFilter(ColorFilter cf) {            }            @Override            public int getOpacity() {                return PixelFormat.UNKNOWN;            }        };        super.setBackgroundDrawable(background);    }    @Override    protected void onSizeChanged(int w, int h, int oldw, int oldh) {        super.onSizeChanged(w, h, oldw, oldh);        Log.d(TAG, "w: " + w + ", h: " + h + ", oldw: " + oldw + ", oldh: " + oldh);        viewWidth = w;        setBackgroundDrawable(null);    }    /**     * 选中回调     */    private void onSeletedCallBack() {        if (null != onWheelViewListener) {            onWheelViewListener.onSelected(selectedIndex, items.get(selectedIndex));        }    }    public void setSeletion(int position) {        final int p = position;        selectedIndex = p + offset;        this.post(new Runnable() {            @Override            public void run() {                WheelView.this.smoothScrollTo(0, p * itemHeight);            }        });    }    public String getSeletedItem() {        return items.get(selectedIndex);    }    public int getSeletedIndex() {        return selectedIndex - offset;    }    @Override    public void fling(int velocityY) {        super.fling(velocityY / 3);    }    @Override    public boolean onTouchEvent(MotionEvent ev) {        if (ev.getAction() == MotionEvent.ACTION_UP) {            startScrollerTask();        }        return super.onTouchEvent(ev);    }    private OnWheelViewListener onWheelViewListener;    public OnWheelViewListener getOnWheelViewListener() {        return onWheelViewListener;    }    public void setOnWheelViewListener(OnWheelViewListener onWheelViewListener) {        this.onWheelViewListener = onWheelViewListener;        // 默认初始不滑动时执行一次回调        onWheelViewListener.onSelected(selectedIndex, items.get(selectedIndex));    }    private int dip2px(float dpValue) {        final float scale = context.getResources().getDisplayMetrics().density;        return (int) (dpValue * scale + 0.5f);    }    private int getViewMeasuredHeight(View view) {        int width = View.MeasureSpec.makeMeasureSpec(0, View.MeasureSpec.UNSPECIFIED);        int expandSpec = View.MeasureSpec.makeMeasureSpec(Integer.MAX_VALUE >> 2,                View.MeasureSpec.AT_MOST);        view.measure(width, expandSpec);        return view.getMeasuredHeight();    }}

dialog_wheelview.xml

<?xml version="1.0" encoding="utf-8"?>    

最后放上完整代码下载链接点我下载

更多相关文章

  1. android 手势屏幕平移图片转换效果(也可以平移文本)
  2. Android关于Activity切换效果的实现加强
  3. android学习之ToggleButton实现开关效果
  4. Android动画效果translate、scale、alpha、rotate详解
  5. Android5.0水波纹效果适配4.X
  6. 自定义ProgressDialog(无遮罩效果)
  7. android下使用Fragment实现左侧3级菜单+动画效果

随机推荐

  1. android 和 java 调色板
  2. Testing和Instrumentation
  3. Android 3.0细节曝光:Google程序更耀眼
  4. OpenGL学习资料和记录
  5. Android Support 包:Android Support v4、
  6. Android文件系统的提取方法(一)
  7. 最常见的猜拳小游戏Android代码实现
  8. Android水波纹点击效果
  9. Android(安卓)Shape自定义纯色圆角按钮
  10. Android(安卓)L中的RecyclerView 、CardV