我的编程空间,编程开发者的网络收藏夹
学习永远不晚

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

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

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.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(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圆盘时间选择器,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对编程网网站的支持!

您可能感兴趣的文章:Android仿iPhone日期时间选择器详解Android日期和时间选择器实现代码Android Studio时间选择器的创建方法Android时间选择器、日期选择器实现代码Android仿IOS10圆盘时间选择器


免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

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

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

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

通过自定义view实现仿iOS实现滑动两端的点选择时间的效果 效果图自定义的view代码public class Ring_Slide2 extends View {private static final double RADIAN =
2022-06-06

iOS自定义时间滚动选择控件

本文实例为大家分享了iOS自定义时间滚动选择控件的具体代码,供大家参考,具体内容如下 1.先上自定义的控件: public class WheelVi
2022-05-19

Android自定义View实现角度选择器

首先来看一下Google Photos的效果实现最终的效果:实现思路仔细观察这个效果,先分析构成结构,我把它分成三部分: 1、表示刻度的点 2、相应点上方的数字 3、控件中央的当前刻度与三角可以看出,构成元素十分简单,不
2022-06-06

Android自定义dialog可选择展示年月日时间选择栏

自定义dialogpackage com.poptest; import android.app.DatePickerDialog; import android.content.Context; import android.vie
2022-06-06

Android开发中实现IOS风格底部选择器(支持时间 日期 自定义)

本文Github代码链接 https://github.com/AndroidMsky/AndoirdIOSPicker 先上图吧:这是笔者最近一个项目一直再用的一个选择器库,自己也在其中做了修改,并决定持续维护下去。 先看使用方法: 日
2022-06-06

WheelPicker自定义时间选择器控件的方法

小编给大家分享一下WheelPicker自定义时间选择器控件的方法,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!本文实例为大家分享了WheelPicker自定义时
2023-06-15

Android 自定义日期段选择控件功能(开始时间-结束时间)

开发中碰到个需求,需要在一个空间中选择完成开始和结束时间。实现的过程走的是程序员开发的老路子,找到轮子后自己改吧改吧就成了。 当时做的时候有几个需求:1.当天为最大的结束日期,2.最大选择范围1年,3.开始时间和结束时间可以为同一天。如有其
2022-06-06

Vue怎么自定义验证日期时间选择器

这篇文章主要介绍了Vue怎么自定义验证日期时间选择器的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇Vue怎么自定义验证日期时间选择器文章都会有所收获,下面我们一起来看看吧。Vue自定义验证之日期时间选择器自定义
2023-06-29

轻松实现可扩展自定义的Android滚轮时间选择控件

项目需求中有个功能模块需要用到时间选择控件,但是android系统自带的太丑了,只能自己优化下,结合WheelView实现滚轮选择日期,好像网上也挺多这种文章的。但是适用范围还是不同,希望这个能够对需求相同的朋友有一定帮助。控件标题还有年月
2022-06-06

编程热搜

  • Android:VolumeShaper
    VolumeShaper(支持版本改一下,minsdkversion:26,android8.0(api26)进一步学习对声音的编辑,可以让音频的声音有变化的播放 VolumeShaper.Configuration的三个参数 durati
    Android:VolumeShaper
  • Android崩溃异常捕获方法
    开发中最让人头疼的是应用突然爆炸,然后跳回到桌面。而且我们常常不知道这种状况会何时出现,在应用调试阶段还好,还可以通过调试工具的日志查看错误出现在哪里。但平时使用的时候给你闹崩溃,那你就欲哭无泪了。 那么今天主要讲一下如何去捕捉系统出现的U
    Android崩溃异常捕获方法
  • android开发教程之获取power_profile.xml文件的方法(android运行时能耗值)
    系统的设置–>电池–>使用情况中,统计的能耗的使用情况也是以power_profile.xml的value作为基础参数的1、我的手机中power_profile.xml的内容: HTC t328w代码如下:
    android开发教程之获取power_profile.xml文件的方法(android运行时能耗值)
  • Android SQLite数据库基本操作方法
    程序的最主要的功能在于对数据进行操作,通过对数据进行操作来实现某个功能。而数据库就是很重要的一个方面的,Android中内置了小巧轻便,功能却很强的一个数据库–SQLite数据库。那么就来看一下在Android程序中怎么去操作SQLite数
    Android SQLite数据库基本操作方法
  • ubuntu21.04怎么创建桌面快捷图标?ubuntu软件放到桌面的技巧
    工作的时候为了方便直接打开编辑文件,一些常用的软件或者文件我们会放在桌面,但是在ubuntu20.04下直接直接拖拽文件到桌面根本没有效果,在进入桌面后发现软件列表中的软件只能收藏到面板,无法复制到桌面使用,不知道为什么会这样,似乎并不是很
    ubuntu21.04怎么创建桌面快捷图标?ubuntu软件放到桌面的技巧
  • android获取当前手机号示例程序
    代码如下: public String getLocalNumber() { TelephonyManager tManager =
    android获取当前手机号示例程序
  • Android音视频开发(三)TextureView
    简介 TextureView与SurfaceView类似,可用于显示视频或OpenGL场景。 与SurfaceView的区别 SurfaceView不能使用变换和缩放等操作,不能叠加(Overlay)两个SurfaceView。 Textu
    Android音视频开发(三)TextureView
  • android获取屏幕高度和宽度的实现方法
    本文实例讲述了android获取屏幕高度和宽度的实现方法。分享给大家供大家参考。具体分析如下: 我们需要获取Android手机或Pad的屏幕的物理尺寸,以便于界面的设计或是其他功能的实现。下面就介绍讲一讲如何获取屏幕的物理尺寸 下面的代码即
    android获取屏幕高度和宽度的实现方法
  • Android自定义popupwindow实例代码
    先来看看效果图:一、布局
  • Android第一次实验
    一、实验原理 1.1实验目标 编程实现用户名与密码的存储与调用。 1.2实验要求 设计用户登录界面、登录成功界面、用户注册界面,用户注册时,将其用户名、密码保存到SharedPreference中,登录时输入用户名、密码,读取SharedP
    Android第一次实验

目录