Android自定义View仿IOS圆盘时间选择器

通过自定义view实现仿iOS实现滑动两端的点选择时间的效果

效果图

自定义的view代码

public class Ring_Slide2 extends View {
 private static final double RADIAN = 180 / Math.PI;
 private int max_progress; // 设置最大进度
 private int cur_progress; //设置锚点1当前进度
 private int cur_progress2; //设置锚点2进度
 private int bottom_color;//设置底色
 private int circle_color; //设置圆的颜色(锚点)
 private int slide_color; //设置滑动过的颜色
 private float ring_width; //圆环的宽度
 private double cur_Angle; //当前锚点1旋转角度
 private double cur_Angle2; //当前锚点2的旋转角度
 private float ring_Radius;//圆环的半径
 private final int[] arrColorCircle = new int[]{0xFFFFde37, 0xFFFFa400};
 private int main_width; //圆的宽度
 private float mWheelCurX, mWheelCurY; //圆的位置
 private float mWheelCurX2, mWheelCurY2; //圆2的位置
 private Paint circle_Paint; //圆环的画笔
 private Paint select_Paint;//选中的画笔
 private Paint dot1; //圆点1
 private Paint dot2; //圆点2
 private Context context;
 private OnSeekBarChangeListener changeListener,changeListener2;
 public Ring_Slide2(Context context) {
  this(context,null);
 }
 public Ring_Slide2(Context context, AttributeSet attrs) {
  this(context, attrs,0);
 }
 public Ring_Slide2(Context context, AttributeSet attrs, int defStyleAttr) {
  super(context, attrs, defStyleAttr);
  this.context=context;
  initAttrs(attrs,defStyleAttr);
  initPadding();
  //初始化画笔
  initPaints();
 }
 //初始化属性
 private void initAttrs(AttributeSet attrs, int defStyle){
  TypedArray typedArray = getContext().obtainStyledAttributes(attrs, R.styleable.Cricle_slide, defStyle, 0);
  max_progress=typedArray.getInt(R.styleable.Cricle_slide_max_progress,720);
  cur_progress=typedArray.getInt(R.styleable.Cricle_slide_cur_progress,420);
  cur_progress2=typedArray.getInt(R.styleable.Cricle_slide_cur_progress2,540);
  if (cur_progress > max_progress) cur_progress = max_progress;
  if (cur_progress2 > max_progress) cur_progress2 = max_progress;
  Bitmap bitmap = BitmapFactory.decodeResource(context.getResources(), R.mipmap.select_sun_bg2);
  main_width= bitmap.getWidth();
  ring_width=typedArray.getFloat(R.styleable.Cricle_slide_Ring_Width,main_width);
  bottom_color=typedArray.getColor(R.styleable.Cricle_slide_bottom_color,getColor(R.color.select_main_bg_color));
  circle_color=typedArray.getColor(R.styleable.Cricle_slide_circle_color,getColor(R.color.duration));
  slide_color=typedArray.getColor(R.styleable.Cricle_slide_slide_color,getColor(R.color.time));
  typedArray.recycle();
 }
 //初始化边距
 private void initPadding(){
  int paddingLeft = getPaddingLeft();
  int paddingTop = getPaddingTop();
  int paddingRight = getPaddingRight();
  int paddingBottom = getPaddingBottom();
  int paddingStart = 0, paddingEnd = 0;
  if (Build.VERSION.SDK_INT >= 17) {
   paddingStart = getPaddingStart();
   paddingEnd = getPaddingEnd();
  }
  int maxPadding = Math.max(paddingLeft, Math.max(paddingTop,
    Math.max(paddingRight, Math.max(paddingBottom, Math.max(paddingStart, paddingEnd)))));
  setPadding(maxPadding, maxPadding, maxPadding, maxPadding);
 }
 private void initPaints(){
  /*
  圆环的画笔
   */
  circle_Paint=new Paint(Paint.ANTI_ALIAS_FLAG);
  circle_Paint.setAntiAlias(true);
  circle_Paint.setColor(bottom_color);
  circle_Paint.setStyle(Paint.Style.STROKE);
  circle_Paint.setStrokeWidth(ring_width);
  /*
  选中区域的画笔
   */
  select_Paint=new Paint(Paint.ANTI_ALIAS_FLAG);
  select_Paint.setShader(new SweepGradient(0, 0, arrColorCircle, null));
  /*select_Paint.setColor(circle_color);*/
  select_Paint.setAntiAlias(true);
  select_Paint.setStyle(Paint.Style.STROKE);
  select_Paint.setStrokeWidth(ring_width);
  // 画锚点
  dot1 = new Paint(Paint.ANTI_ALIAS_FLAG);
  dot1.setColor(circle_color);
  dot1.setAntiAlias(true);
  dot1.setStyle(Paint.Style.FILL);
  // 画锚点2
  dot2 = new Paint(Paint.ANTI_ALIAS_FLAG);
  dot2.setColor(slide_color);
  dot2.setAntiAlias(true);
  dot2.setStyle(Paint.Style.FILL);
 }
 //获取宽度
 private float getDimen(int dimenId) {
  return getResources().getDimension(dimenId);
 }
 //获取颜色
 @TargetApi(Build.VERSION_CODES.M)
 private int getColor(int colorId) {
  final int version = Build.VERSION.SDK_INT;
  if (version >= 23) {
   return getContext().getColor(colorId);
  } else {
   return ContextCompat.getColor(getContext(), colorId);
  }
 }
 @Override
 protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
  super.onMeasure(widthMeasureSpec, heightMeasureSpec);
  Bitmap bitmap = BitmapFactory.decodeResource(context.getResources(), R.mipmap.setalarm_colock_bg);
  int height = bitmap.getHeight()+main_width*2;
  int width = bitmap.getWidth()+main_width*2;
  int min = Math.min(height, width);
  setMeasuredDimension(min,min);
  initposition();
 }
 private void initposition(){
  //转换为360度
  cur_Angle=(double) cur_progress / max_progress*360.0;
  cur_Angle2=(double)cur_progress2 / max_progress*360.0;
  //计算初始化旋转的角度
  double cos = -Math.cos(Math.toRadians(cur_Angle));
  double cos2 = -Math.cos(Math.toRadians(cur_Angle2));
  //根据旋转的角度来确定位置
  MakeCurPosition(cos);
  MakeCurPosition2(cos2);
  //确定圆环的半径
  ring_Radius=(getMeasuredWidth() - getPaddingLeft() - getPaddingRight() - ring_width) / 2;
 }
 private void MakeCurPosition(double cos){
  //根据旋转的角度来确定圆的位置
  //确定x点的坐标
  mWheelCurX = calcXLocationInWheel(cur_Angle, cos);
  //确定y点的坐标
  mWheelCurY=calcYLocationInWheel(cos);
 }
 private void MakeCurPosition2(double cos2){
  //根据旋转的角度来确定圆的位置
  //确定x点的坐标
  mWheelCurX2 = calcXLocationInWheel(cur_Angle2, cos2);
  //确定y点的坐标
  mWheelCurY2=calcYLocationInWheel(cos2);
 }
 //确定x点的坐标
 private float calcXLocationInWheel(double angle,double cos){
  if (angle < 180) {
   return (float) (getMeasuredWidth() / 2 + Math.sqrt(1 - cos * cos) * ring_Radius); //Math.sqrt正平分根 9-3
  } else {
   return (float) (getMeasuredWidth() / 2 - Math.sqrt(1 - cos * cos) * ring_Radius);
  }
 }
 //确定y点的坐标
 private float calcYLocationInWheel(double cos) {
  return getMeasuredWidth() / 2 + ring_Radius * (float) cos;
 }
 @Override
 protected void onDraw(Canvas canvas) {
  super.onDraw(canvas);
  float left = getPaddingLeft() + ring_width / 2;
  float top = getPaddingTop() + ring_width / 2;
  float right = canvas.getWidth() - getPaddingRight() - ring_width / 2;
  float bottom = canvas.getHeight() - getPaddingBottom() - ring_width / 2;
  float centerX = (left + right) / 2;
  float centerY = (top + bottom) / 2;
  float wheelRadius = (canvas.getWidth() - getPaddingLeft() - getPaddingRight()) / 2 - ring_width / 2;
  canvas.drawCircle(centerX, centerY, wheelRadius, circle_Paint);
  //画选中区域
  // canvas.drawArc(new RectF(left, top, right, bottom), (float) (Math.PI * RADIAN + Math.acos(cur_Angle) * RADIAN), (float) (Math.abs(cur_Angle-cur_Angle2)), false, select_Paint);
  Log.i("TAG","第一个的角度="+cur_Angle);
  Log.i("TAG","第一个的角度2="+cur_Angle2);
  float begin=0; //圆弧的起点位置
  float stop=0;
  if(cur_Angle>180 && cur_Angle>cur_Angle2 ){ //180 -- 360
   begin=(float) (-Math.abs(cur_Angle-360)-90);
   stop=(float) Math.abs(Math.abs(cur_Angle-360)+cur_Angle2);
   Log.i("TAG","begin="+begin);
   Log.i("TAG","stop="+stop);
  }else if(cur_Angle>cur_Angle2){
   begin=(float) cur_Angle-90;
   stop=(float)(360-(cur_Angle-cur_Angle2));
  }else {
    begin=(float) cur_Angle-90;
   stop=(float) Math.abs(cur_Angle-cur_Angle2);
  }
  canvas.drawArc(new RectF(left, top, right, bottom), begin,stop, false, select_Paint);
  //画锚点 画圆
  canvas.drawCircle(mWheelCurX, mWheelCurY, ring_width/2, dot1);
  //画锚点 画圆
  canvas.drawCircle(mWheelCurX2, mWheelCurY2, ring_width/2, dot2);
  Log.i("TAG","锚点1Y"+mWheelCurY+"锚点1X"+mWheelCurX);
  Log.i("TAG","锚点2Y"+mWheelCurY2+"锚点1X"+mWheelCurX2);
 }
 @Override
 public boolean onTouchEvent(MotionEvent event) {
  float x = event.getX();
  float y = event.getY();
  int flag=0;
  //判断是否触控到两个点中的其中某个点
  if(isMovedot2(x,y)){
   flag=2;
  }else if(isMovedot1(x,y)){
   flag=1;
  }
  /* if(isMovedot1(x,y)){
   flag=1;
  }else if(isMovedot2(x,y)){
   flag=2;
  }*/
  if(event.getAction()==MotionEvent.ACTION_MOVE || isMovedot1(x,y) ==true || isMovedot2(x,y)==true ){
   Log.i("TAG","进入X="+x+"进入Y="+y);
   //通过触摸点算出cos角度值
   float cos = calculateCos(x, y);
   // 通过反三角函数获得角度值
   double angle; //获取滑动的角度
   if (x < getWidth() / 2) { // 滑动超过180度
    angle = Math.PI * RADIAN + Math.acos(cos) * RADIAN; //通过计算得到滑动的角度值
   } else { // 没有超过180度
    angle = Math.PI * RADIAN - Math.acos(cos) * RADIAN; //PI 周长比直径 返回弧角度的余弦值
   }
   if(flag==1){
    cur_Angle=angle;
    cur_progress=getSelectedValue(cur_Angle);
    MakeCurPosition(cos);
    if (changeListener != null) {
     changeListener.onChanged(this, cur_progress);
    }
   }else if(flag==2){
    cur_Angle2=angle;
    cur_progress2=getSelectedValue(cur_Angle2);
    MakeCurPosition2(cos);
    if (changeListener2 != null) {
     changeListener2.onChanged(this, cur_progress2);
    }
   }
   invalidate();
   return true;
  }else {
   return super.onTouchEvent(event);
  }
 }
 private boolean isMovedot1(float x,float y){
  float dot1x = Math.abs(mWheelCurX - x);
  float dot1y = Math.abs(mWheelCurY - y);
  if(dot1x<30 && dot1y<30){
   return true;
  }else{
   return false;
  }
 }
 private boolean isMovedot2(float x,float y){
  float dot1x = Math.abs(mWheelCurX2 - x);
  float dot1y = Math.abs(mWheelCurY2 - y);
  if(dot1x<30 && dot1y<30){
   return true;
  }else{
   return false;
  }
 }
 //拿到切斜角的cos值
 private float calculateCos(float x, float y){
  float width = x - getWidth() / 2;
  float height = y - getHeight() / 2;
  float slope = (float) Math.sqrt(width * width + height * height);
  return height / slope;
 }
 private int getSelectedValue(double mCurAngle) { //角度转进度
  return Math.round(max_progress * ((float) mCurAngle / 360)); //四舍五入
 }
 public void setOnSeekBarChangeListener(OnSeekBarChangeListener listener) {
  changeListener = listener;
 }
 public void setOnSeekBarChangeListener2(OnSeekBarChangeListener listener) {
  changeListener2 = listener;
 }
 public void initRadian(int pro1,int pro2){
  this.cur_progress=pro1;
  this.cur_progress2=pro2;
  invalidate();
 }
 public interface OnSeekBarChangeListener {
  void onChanged(Ring_Slide2 seekbar, int curValue);
 }
}

自定义stayle样式,在values下新建sttrs.xml文件

 <declare-styleable name="Cricle_slide">
  //设置最大进度
  <attr name="max_progress" format="integer"></attr>
  //设置当前进度
  <attr name="cur_progress" format="integer"></attr>
  //设置当前进度
  <attr name="cur_progress2" format="integer"></attr>
  //设置底色
  <attr name="bottom_color" format="color"></attr>
  //设置圆的颜色
  <attr name="circle_color" format="color"></attr>
  //设置滑动的颜色
  <attr name="slide_color" format="color"></attr>
  //圆环的宽度 (dimension是代表尺寸值)
  <attr name="Ring_Width" format="dimension"></attr>
 </declare-styleable>

以上所述是小编给大家介绍的Android自定义View仿IOS圆盘时间选择器,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对我们网站的支持!

(0)

相关推荐

  • iOS中使用UIDatePicker制作时间选择器的实例教程

    UIDatePicker的创建 UIDatePicker是一个可以用来选择或者设置日期的控件,不过它是像转轮一样的控件,而且是苹果专门为日历做好的控件,如下图所示: 除了UIDatePicker控件,还有一种更通用的转轮形的控件:UIPickerView,只不过UIDatePicker控件显示的就是日 历,而UIPickerView控件中显示的内容需要我们自己用代码设置.本篇文章简单介绍UIDatePicker控件,后边的文章会介绍 UIPickerView. 1.运行Xcode ,新建一个Si

  • iOS实现自定义起始时间选择器视图

    随着界面的整体效果的各种展现, 起始时间选择器的展现也需求突出! 最近项目中发现时间选择器使用处还挺多, 数了数原型图发现有6处. 便决定自定义时间选择器视图写个 Demo, 封装好在所需控制器里直接调用! 主要功能: 调起时间选择器, 传值(起始时间/截止时间), 两者时间均要合理, 不能超过未来时间, 并且起始时间不能大于截止时间. 点击取消或空白处收起时间选择器. 如果需要可以根据自己的需求来修改界面, 效果如下: 主要步骤: 创建时间选择器Picker 且确认取消按钮实现功能逻辑 创建展

  • Android自定义View仿IOS圆盘时间选择器

    通过自定义view实现仿iOS实现滑动两端的点选择时间的效果 效果图 自定义的view代码 public class Ring_Slide2 extends View { private static final double RADIAN = 180 / Math.PI; private int max_progress; // 设置最大进度 private int cur_progress; //设置锚点1当前进度 private int cur_progress2; //设置锚点2进度 p

  • Android自定义view仿iOS弹出框效果

    本文实例为大家分享了Android自定义view仿iOS弹出框的具体代码,供大家参考,具体内容如下 运行效果图 自定义对话框的使用,仿照ios.从底部弹出,类似pop窗口.包括消息.图片.列表及对话框. 好了,用法都会,直接贴上代码 1.layout布局文件 view_actionsheet.xml <?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="ht

  • Android自定义view仿IOS开关效果

    本文主要讲解如何在 Android 下实现高仿 iOS 的开关按钮,并非是在 Android 自带的 ToggleButton 上修改,而是使用 API 提供的 onDraw.onMeasure.Canvas 方法,纯手工绘制.基本原理就是在 Canvas 上叠着放两张图片,上面的图片根据手指触摸情况,不断移动,实现开关效果. 废话不说,上效果图,看看怎么样 样式如下: 网上也有实现这种效果的,但是大都滑动没中间消失的动画,或者是很复杂,今天用简单的绘图方式实现,重点就在onDraw里绘图. 功

  • Android仿IOS10圆盘时间选择器

    介绍 这是一款仿IOS10(就寝功能)的圆盘时间选择器 项目演示 实现思路 以720度为一个周期,0~360°对应0~12小时,360°~720°对应12~24小时 这里以"开始时间设置按钮"为例来谈谈它的滑动实现: 将"开始时间设置按钮"作为点A,表盘中心作为点O,手指触摸点作为点P.通过反正切公式可以计算出∠AOP的大小,然后随着手指的位置不断变化去更新点A的位置(即点A的角度). // 坐标系的直线表达式 // 直线l1的表达式子:过钟表中心点和开始控件中心点

  • Android 自定义view仿支付宝咻一咻功能

    支付宝上有一个咻一咻的功能,就是点击图片后四周有水波纹的这种效果,今天也写一个类似的功能. 效果如下所示: 思路: 就是几个圆的半径不断在变大,这个可以使用动画缩放实现,还有透明动画 还有就是这是好几个圆,然后执行的动画有个延迟效果,其实这些动画是放在一起执行的,熟悉属性动画的知道已经给我们提供了同步执行动画和顺序执行动画的实现api,也会会有人说这几个view就是在onDraw()方法中画几个圆,可能会说我还要继承容器view去onLayout()方法中这些子view添加在某个特定的区域,当然

  • Android自定义View 仿QQ侧滑菜单的实现代码

    先看看QQ的侧滑效果 分析一下 先上原理图(不知道能否表达的清楚 ==) -首先这里使用了 Android 的HorizontalScrollView 水平滑动布局作为容器,当然我们需要继承它自定义一个侧滑视图 - 这个容器里面有一个父布局(一般用LinerLayout,本demo用的是),这个父布局里面有且只有两个子控件(布局),初始状态菜单页的位置在Y轴上存在偏移这样可以就可以形成主页叠在菜单页的上方的视觉效果:然后在滑动的过程程中 逐渐修正偏移,最后菜单页和主页并排排列.原理搞清了实现起来

  • Android自定义View仿支付宝输入六位密码功能

    跟选择银行卡界面类似,也是用一个PopupWindow,不过输入密码界面是一个自定义view,当输入六位密码完成后用回调在Activity中获取到输入的密码并以Toast显示密码.效果图如下: 自定义view布局效果图及代码如下: <?xml version="1.0" encoding="utf-8"?> <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/

  • Android自定义View仿探探卡片滑动效果

    Android自定义View仿探探卡片滑动这种效果网上有很多人已经讲解了实现思路,大多都用的是RecyclerView来实现的,但是我们今天来换一种实现思路,只用一个自定义的ViewGroup来搞定这个实现. 下面我们先看一下实现的效果: 这个自定义View用法也很简单,首先从github上下载或者fork这个项目,在布局中添加: <com.liyafeng.view.swipecard.SwipeCardLayout android:id="@+id/scl_layout" a

  • Android 自定义view仿微信相机单击拍照长按录视频按钮

    Android仿微信相机的拍照按钮单击拍照,长按录视频.先上效果图. 项目地址:https://github.com/c786909486/PhotoButton2/tree/v1.0 添加依赖 allprojects { repositories { ... maven { url 'https://jitpack.io' } } } dependencies { compile compile 'com.github.c786909486:PhotoButton2:v1.1' } 长按效果分

  • Android自定义view仿QQ的Tab按钮动画效果(示例代码)

    话不多说 先上效果图 实现其实很简单,先用两张图 一张是背景的图,一张是笑脸的图片,笑脸的图片是白色,可能看不出来.实现思路:主要是再触摸view的时候同时移动这两个图片,但是移动的距离不一样,造成的错位感,代码很简单: import android.content.Context import android.graphics.* import android.util.AttributeSet import android.view.MotionEvent import android.vi

随机推荐