Android自定义滚动选择器实例代码

Android自定义滚动选择器

实现图片的效果

代码如下

package com.linzihui.widget;

import android.annotation.SuppressLint;
import android.content.Context;
import android.graphics.Canvas;
import android.graphics.Paint;
import android.graphics.Paint.Align;
import android.graphics.Paint.FontMetricsInt;
import android.graphics.Paint.Style;
import android.os.Handler;
import android.os.Message;
import android.util.AttributeSet;
import android.view.MotionEvent;
import android.view.View;

import java.util.ArrayList;
import java.util.List;
import java.util.Timer;
import java.util.TimerTask;

/**
 * 自定义滚动选择器
 *
 * @author cuiran
 *
 */
@SuppressLint({ "HandlerLeak", "ClickableViewAccessibility" })
public class PickerScrollView extends View {

  public static final String TAG = "PickerView";
  /**
   * text之间间距和minTextSize之比
   */
  public static final float MARGIN_ALPHA = 2.8f;
  /**
   * 自动回滚到中间的速度
   */
  public static final float SPEED = 2;

  private List<Pickers> mDataList;
  /**
   * 选中的位置,这个位置是mDataList的中心位置,一直不变
   */
  private int mCurrentSelected;
  private Paint mPaint;

  private float mMaxTextSize = 20;
  private float mMinTextSize = 10;

  private float mMaxTextAlpha = 255;
  private float mMinTextAlpha = 120;

  private int mColorText = 0x333333;

  private int mViewHeight;
  private int mViewWidth;

  private float mLastDownY;
  /**
   * 滑动的距离
   */
  private float mMoveLen = 0;
  private boolean isInit = false;
  private onSelectListener mSelectListener;
  private Timer timer;
  private MyTimerTask mTask;

  Handler updateHandler = new Handler() {

    @Override
    public void handleMessage(Message msg) {
      if (Math.abs(mMoveLen) < SPEED) {
        mMoveLen = 0;
        if (mTask != null) {
          mTask.cancel();
          mTask = null;
          performSelect();
        }
      } else
        // 这里mMoveLen / Math.abs(mMoveLen)是为了保有mMoveLen的正负号,以实现上滚或下滚
        mMoveLen = mMoveLen - mMoveLen / Math.abs(mMoveLen) * SPEED;
      invalidate();
    }

  };

  public PickerScrollView(Context context) {
    super(context);
    init();
  }

  public PickerScrollView(Context context, AttributeSet attrs) {
    super(context, attrs);
    init();
  }

  public void setOnSelectListener(onSelectListener listener) {
    mSelectListener = listener;
  }

  private void performSelect() {
    if (mSelectListener != null)
      mSelectListener.onSelect(mDataList.get(mCurrentSelected));
  }

  public void setData(List<Pickers> datas) {
    mDataList = datas;
    mCurrentSelected = datas.size() / 2;
    invalidate();
  }

  /**
   * 选择选中的item的index
   *
   * @param selected
   */
  public void setSelected(int selected) {
    mCurrentSelected = selected;
    int distance = mDataList.size() / 2 - mCurrentSelected;
    if (distance < 0)
      for (int i = 0; i < -distance; i++) {
        moveHeadToTail();
        mCurrentSelected--;
      }
    else if (distance > 0)
      for (int i = 0; i < distance; i++) {
        moveTailToHead();
        mCurrentSelected++;
      }
    invalidate();
  }

  /**
   * 选择选中的内容
   *
   * @param mSelectItem
   */
  public void setSelected(String mSelectItem) {
    for (int i = 0; i < mDataList.size(); i++)
      if (mDataList.get(i).equals(mSelectItem)) {
        setSelected(i);
        break;
      }
  }

  private void moveHeadToTail() {
    Pickers head = mDataList.get(0);
    mDataList.remove(0);
    mDataList.add(head);
  }

  private void moveTailToHead() {
    Pickers tail = mDataList.get(mDataList.size() - 1);
    mDataList.remove(mDataList.size() - 1);
    mDataList.add(0, tail);
  }

  @Override
  protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
    super.onMeasure(widthMeasureSpec, heightMeasureSpec);
    mViewHeight = getMeasuredHeight();
    mViewWidth = getMeasuredWidth();
    // 按照View的高度计算字体大小
    mMaxTextSize = mViewHeight / 8.0f;
    mMinTextSize = mMaxTextSize / 2f;
    isInit = true;
    invalidate();
  }

  private void init() {
    timer = new Timer();
    mDataList = new ArrayList<Pickers>();
    mPaint = new Paint(Paint.ANTI_ALIAS_FLAG);
    mPaint.setStyle(Style.FILL);
    mPaint.setTextAlign(Align.CENTER);
    mPaint.setColor(mColorText);
  }

  @Override
  protected void onDraw(Canvas canvas) {
    super.onDraw(canvas);
    // 根据index绘制view
    if (isInit)
      drawData(canvas);
  }

  private void drawData(Canvas canvas) {
    // 先绘制选中的text再往上往下绘制其余的text
    float scale = parabola(mViewHeight / 4.0f, mMoveLen);
    float size = (mMaxTextSize - mMinTextSize) * scale + mMinTextSize;
    mPaint.setTextSize(size);
    mPaint.setAlpha((int) ((mMaxTextAlpha - mMinTextAlpha) * scale + mMinTextAlpha));
    // text居中绘制,注意baseline的计算才能达到居中,y值是text中心坐标
    float x = (float) (mViewWidth / 2.0);
    float y = (float) (mViewHeight / 2.0 + mMoveLen);
    FontMetricsInt fmi = mPaint.getFontMetricsInt();
    float baseline = (float) (y - (fmi.bottom / 2.0 + fmi.top / 2.0));

    int indexs = mCurrentSelected;
    String textData = mDataList.get(indexs).getShowConetnt();
    canvas.drawText(textData, x, baseline, mPaint);

    // 绘制上方data
    for (int i = 1; (mCurrentSelected - i) >= 0; i++) {
      drawOtherText(canvas, i, -1);
    }
    // 绘制下方data
    for (int i = 1; (mCurrentSelected + i) < mDataList.size(); i++) {
      drawOtherText(canvas, i, 1);
    }
  }

  /**
   * @param canvas
   * @param position
   *      距离mCurrentSelected的差值
   * @param type
   *      1表示向下绘制,-1表示向上绘制
   */
  private void drawOtherText(Canvas canvas, int position, int type) {
    float d = (float) (MARGIN_ALPHA * mMinTextSize * position + type
        * mMoveLen);
    float scale = parabola(mViewHeight / 4.0f, d);
    float size = (mMaxTextSize - mMinTextSize) * scale + mMinTextSize;
    mPaint.setTextSize(size);
    mPaint.setAlpha((int) ((mMaxTextAlpha - mMinTextAlpha) * scale + mMinTextAlpha));
    float y = (float) (mViewHeight / 2.0 + type * d);
    FontMetricsInt fmi = mPaint.getFontMetricsInt();
    float baseline = (float) (y - (fmi.bottom / 2.0 + fmi.top / 2.0));

    int indexs = mCurrentSelected + type * position;
    String textData = mDataList.get(indexs).getShowConetnt();
    canvas.drawText(textData, (float) (mViewWidth / 2.0), baseline, mPaint);
  }

  /**
   * 抛物线
   *
   * @param zero
   *      零点坐标
   * @param x
   *      偏移量
   * @return scale
   */
  private float parabola(float zero, float x) {
    float f = (float) (1 - Math.pow(x / zero, 2));
    return f < 0 ? 0 : f;
  }

  @Override
  public boolean onTouchEvent(MotionEvent event) {
    switch (event.getActionMasked()) {
    case MotionEvent.ACTION_DOWN:
      doDown(event);
      break;
    case MotionEvent.ACTION_MOVE:
      doMove(event);
      break;
    case MotionEvent.ACTION_UP:
      doUp(event);
      break;
    }
    return true;
  }

  private void doDown(MotionEvent event) {
    if (mTask != null) {
      mTask.cancel();
      mTask = null;
    }
    mLastDownY = event.getY();
  }

  private void doMove(MotionEvent event) {

    mMoveLen += (event.getY() - mLastDownY);

    if (mMoveLen > MARGIN_ALPHA * mMinTextSize / 2) {
      // 往下滑超过离开距离
      moveTailToHead();
      mMoveLen = mMoveLen - MARGIN_ALPHA * mMinTextSize;
    } else if (mMoveLen < -MARGIN_ALPHA * mMinTextSize / 2) {
      // 往上滑超过离开距离
      moveHeadToTail();
      mMoveLen = mMoveLen + MARGIN_ALPHA * mMinTextSize;
    }

    mLastDownY = event.getY();
    invalidate();
  }

  private void doUp(MotionEvent event) {
    // 抬起手后mCurrentSelected的位置由当前位置move到中间选中位置
    if (Math.abs(mMoveLen) < 0.0001) {
      mMoveLen = 0;
      return;
    }
    if (mTask != null) {
      mTask.cancel();
      mTask = null;
    }
    mTask = new MyTimerTask(updateHandler);
    timer.schedule(mTask, 0, 10);
  }

  class MyTimerTask extends TimerTask {
    Handler handler;

    public MyTimerTask(Handler handler) {
      this.handler = handler;
    }

    @Override
    public void run() {
      handler.sendMessage(handler.obtainMessage());
    }

  }

  public interface onSelectListener {
    void onSelect(Pickers pickers);
  }
}
package com.linzihui.widget;

import java.io.Serializable;

public class Pickers implements Serializable {

  private static final long serialVersionUID = 1L;

  private String showConetnt;
  private String showId;

  public String getShowConetnt() {
    return showConetnt;
  }

  public String getShowId() {
    return showId;
  }

  public Pickers(String showConetnt, String showId) {
    super();
    this.showConetnt = showConetnt;
    this.showId = showId;
  }

  public Pickers() {
    super();
  }

}

使用的时候 新建一个Activity 里面对View进行处理

 pickerscrlllview=(PickerScrollView)findViewById(R.id.pickerscrlllview);
    pickerscrlllview.setOnSelectListener(pickerListener);

 PickerScrollView.onSelectListener pickerListener0 = new PickerScrollView.onSelectListener() {

    @Override
    public void onSelect(Pickers pickers) {
      s1=pickers.getShowConetnt();
    }
  };

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • Android自定义可循环的滚动选择器CycleWheelView

    最近碰到个项目要使用到滚动选择器,原生的NumberPicker可定制性太差,不大符合UI要求. 网上开源的WheelView是用ScrollView写的,不能循环滚动,而且当数据量很大时要加载的Item太多,性能非常低. 然后,还是自己写一个比较靠谱,用的是ListView实现的.写完自己体验了一下,性能不错,再大的数据也不怕了. 感觉不错,重新封装了一下,提供了一些接口可以直接按照自己的需求定制,调用方法在MainActivity中. 补个图片: 不多说了,直接上代码: CycleWheel

  • Android PickerView滚动选择器的使用方法

    手机里设置闹钟需要选择时间,那个选择时间的控件就是滚动选择器,前几天用手机刷了MIUI,发现自带的那个时间选择器效果挺好看的,于是就自己仿写了一个,权当练手.先来看效果: 效果还行吧?实现思路就是自定义一个PickerView,单独滚动的是一个PickerView,显然上图中有分和秒的选择所以在布局里用了两个PickerView.由于这里不涉及到text的点击事件,所以只需要继承View就行了,直接把text用canvas画上去.PickerView的实现的主要难点: 难点1: 字体随距离的渐变

  • Android自定义滚动选择器实例代码

    Android自定义滚动选择器 实现图片的效果 代码如下 package com.linzihui.widget; import android.annotation.SuppressLint; import android.content.Context; import android.graphics.Canvas; import android.graphics.Paint; import android.graphics.Paint.Align; import android.graph

  • Android自定义扇形倒计时实例代码

    一.概述 严格来说,我是Android小白,写的目的只是想作为知识储备而已-.但是想到别人或许会不小心搜到我的这篇,如果我只是简单的描述,别人有可能看不懂,说不定还被吐槽,那岂不是很冤吗? 所以,我还是把问题及过程描述清楚,这也是对自己的一个交代,同时,这也是我的第一篇,我应该做好它; 先说一下需求: 最近工作中需要做一个倒计时,是那种一个圆,慢慢的被吃掉的动画倒计时,由于自己知识不是很足,只知道要用Canvas来画,在网上搜了一圈,发现要么是静态的画了一个扇形,要么是不能控制控件的位置大小-.

  • Android仿iphone自定义滚动选择器

    本文实例为大家分享了Android仿iphone自定义滚动选择器的具体代码,供大家参考,具体内容如下 一.多的不说,效果图,先走起 二.实例源码 (1)自定义控件 package com.pickerscrollview.views; import java.util.ArrayList; import java.util.List; import java.util.Timer; import java.util.TimerTask; import android.annotation.Sup

  • Android 日期选择器实例代码

    废话不多说了,直接给大家贴代码了,具体代码如下所示: //出生年月设置 private void birthSetting() { Calendar calendar = Calendar.getInstance(); calendar.add(Calendar.DAY_OF_YEAR, 1); new DatePickerDialog(mContext, new DatePickerDialog.OnDateSetListener() { @Override public void onDat

  • android自定义滚动上下回弹scollView

    本文实例为大家分享了android自定义滚动上下回弹scollView的具体代码,供大家参考,具体内容如下 这是一个自定义view,在xml布局中用这个view嵌套要使之可以上下回弹的view 就能实现布局可以滚动上下回弹了,自定义view代码如下: package com.loopfire.meitaotao.view.scrollView;   import android.content.Context; import android.graphics.Rect; import andro

  • 微信小程序自定义滚动选择器

    本文实例为大家分享了微信小程序自定义滚动选择器的具体代码,供大家参考,具体内容如下 最近项目里有个需求要做个滚动选择器,在网上找了半天也没找到合适的demo,没办法只能发挥我的聪明才智创造一个,上代码. js: // pages/xuanzeqi/xuanzeqi.js Page({   /**    * 页面的初始数据    */   data: {     list: ['0分', '1分', '2分', '3分', '4分', '5分', '6分', '7分', '8分', '9分', '

  • 微信公众号测试账号自定义菜单的实例代码

    自定义菜单接口可实现多种类型按钮,如下: 1.click:点击推事件 用户点击click类型按钮后,微信服务器会通过消息接口推送消息类型为event 的结构给开发者(参考消息接口指南),并且带上按钮中开发者填写的key值,开发者可以通过自定义的key值与用户进行交互: 2.view:跳转URL 用户点击view类型按钮后,微信客户端将会打开开发者在按钮中填写的网页URL,可与网页授权获取用户基本信息接口结合,获得用户基本信息. 3.scancode_push:扫码推事件 用户点击按钮后,微信客户

  • iOS UIScrollView滚动视图/无限循环滚动/自动滚动的实例代码

    我们都知道UIScrollView有一种很流畅的切换效果,结合UIPageControl的辅助展示效果,就可以完成一个很不错的产品介绍功能页面.下面给大家分享iOS UIScrollView滚动视图/无限循环滚动/自动滚动功能,具体代码如下所示: <UIScrollViewDelegate> #define WIDTH [[UIScreen mainScreen] bounds].size.width #define HEIGHT [[UIScreen mainScreen] bounds].

  • Android 自定义ContentProvider简单实例

    Android 自定义ContentProvider简单实例 Android允许我们定义自己的的ContentProvider对象来共享数据,练练手,简单来实现一下. 要使用ContentProvider来操作数据,必须要有保存数据的场所.可以使用文件或SQLite数据库的方式来保存数据,通常使用SQLite数据库. 1,创建一个数据库帮助类,归根结底都是它在操作数据库.代码如下: package com.njue; import android.content.Context; import

  • Android 自定义标题栏的实例详解

     Android 自定义标题栏的实例详解 开发 Android APP 经常会用到自定义标题栏,而有多级页面的情况下还需要给自定义标题栏传递数据. 本文要点: 自定义标题填充不完整 自定义标题栏返回按钮的点击事件 一.代码 这里先介绍一下流程: 1. 创建一个标题栏布局文件 mytitlebar.xml 2. 在style.xml中创建 mytitlestyle 主题 3. 创建类 CustomTitleBar 4. 在需要自定义标题栏的Activity的OnCreate方法中实例化 Custo

随机推荐