返回顶部
首页 > 资讯 > 移动开发 >Android自定义控件实现温度旋转按钮效果
  • 150
分享到

Android自定义控件实现温度旋转按钮效果

按钮Android 2022-06-06 05:06:18 150人浏览 泡泡鱼
摘要

首先看下效果图 温度旋转按钮 实现思路 初始化一些参数 绘制刻度盘 绘制刻度盘下的圆弧 绘制标题与温度标识 绘制旋转按钮 绘制温度 处理滑动事

首先看下效果图


温度旋转按钮

实现思路

初始化一些参数 绘制刻度盘 绘制刻度盘下的圆弧 绘制标题与温度标识 绘制旋转按钮 绘制温度 处理滑动事件 提供一些接口方法

实现方法

初始化一些参数


public class TempControlView extends View {
 // 控件宽
 private int width;
 // 控件高
 private int height;
 // 刻度盘半径
 private int dialRadius;
 // 圆弧半径
 private int arcRadius;
 // 刻度高
 private int scaleHeight = dp2px(10);
 // 刻度盘画笔
 private Paint dialPaint;
 // 圆弧画笔
 private Paint arcPaint;
 // 标题画笔
 private Paint titlePaint;
 // 温度标识画笔
 private Paint tempFlagPaint;
 // 旋转按钮画笔
 private Paint buttonPaint;
 // 温度显示画笔
 private Paint tempPaint;
 // 文本提示
 private String title = "最高温度设置";
 // 温度
 private int temperature;
 // 最低温度
 private int minTemp = 15;
 // 最高温度
 private int maxTemp = 30;
 // 四格(每格4.5度,共18度)代表温度1度
 private int angleRate = 4;
 // 按钮图片
 private Bitmap buttonImage = BitmapFactory.decodeResource(getResources(),
   R.mipmap.btn_rotate);
 // 按钮图片阴影
 private Bitmap buttonImageShadow = BitmapFactory.decodeResource(getResources(),
   R.mipmap.btn_rotate_shadow);
 // 抗锯齿
 private PaintFlagsDrawFilter paintFlagsDrawFilter;
 // 温度改变监听
 private OnTempChangeListener onTempChangeListener;
 // 以下为旋转按钮相关
 // 当前按钮旋转的角度
 private float rotateAngle;
 // 当前的角度
 private float currentAngle;
 ...
 @Override
 protected void onSizeChanged(int w, int h, int oldw, int oldh) {
  super.onSizeChanged(w, h, oldw, oldh);
  // 控件宽、高
  width = height = Math.min(h, w);
  // 刻度盘半径
  dialRadius = width / 2 - dp2px(20);
  // 圆弧半径
  arcRadius = dialRadius - dp2px(20);
 }
 ...
}

绘制刻度盘

以屏幕中心为画布原点,圆弧角度为270°,绘制未选中与选中状态的刻度盘。

旋转方法中多减的2°是后期调整所得,不用在意。



private void drawScale(canvas canvas) {
 canvas.save();
 canvas.translate(getWidth() / 2, getHeight() / 2);
 // 逆时针旋转135-2度
 canvas.rotate(-133);
 dialPaint.setColor(Color.parseColor("#3CB7EA"));
 for (int i = 0; i < 60; i++) {
  canvas.drawLine(0, -dialRadius, 0, -dialRadius + scaleHeight, dialPaint);
  canvas.rotate(4.5f);
 }
 canvas.rotate(90);
 dialPaint.setColor(Color.parseColor("#E37364"));
 for (int i = 0; i < (temperature - minTemp) * angleRate; i++) {
  canvas.drawLine(0, -dialRadius, 0, -dialRadius + scaleHeight, dialPaint);
  canvas.rotate(4.5f);
 }
 canvas.restore();
}


绘制刻度盘下的圆弧



private void drawArc(Canvas canvas) {
 canvas.save();
 canvas.translate(getWidth() / 2, getHeight() / 2);
 canvas.rotate(135 + 2);
 RectF rectF = new RectF(-arcRadius, -arcRadius, arcRadius, arcRadius);
 canvas.drawArc(rectF, 0, 265, false, arcPaint);
 canvas.restore();
}


绘制标题与温度标识


 
 private void drawText(Canvas canvas) {
  canvas.save();
  // 绘制标题
  float titleWidth = titlePaint.measureText(title);
  canvas.drawText(title, (width - titleWidth) / 2, dialRadius * 2 + dp2px(15), titlePaint);
  // 绘制最小温度标识
  // 最小温度如果小于10,显示为0x
  String minTempFlag = minTemp < 10 ? "0" + minTemp : minTemp + "";
  float tempFlagWidth = titlePaint.measureText(maxTemp + "");
  canvas.rotate(55, width / 2, height / 2);
  canvas.drawText(minTempFlag, (width - tempFlagWidth) / 2, height + dp2px(5), tempFlagPaint);
  // 绘制最大温度标识
  canvas.rotate(-105, width / 2, height / 2);
  canvas.drawText(maxTemp + "", (width - tempFlagWidth) / 2, height + dp2px(5), tempFlagPaint);
  canvas.restore();
 }

绘制旋转按钮



private void drawButton(Canvas canvas) {
 // 按钮宽高
 int buttonWidth = buttonImage.getWidth();
 int buttonHeight = buttonImage.getHeight();
 // 按钮阴影宽高
 int buttonShadowWidth = buttonImageShadow.getWidth();
 int buttonShadowHeight = buttonImageShadow.getHeight();
 // 绘制按钮阴影
 canvas.drawBitmap(buttonImageShadow, (width - buttonShadowWidth) / 2,
   (height - buttonShadowHeight) / 2, buttonPaint);
 Matrix matrix = new Matrix();
 // 设置按钮位置
 matrix.setTranslate(buttonWidth / 2, buttonHeight / 2);
 // 设置旋转角度
 matrix.preRotate(45 + rotateAngle);
 // 按钮位置还原,此时按钮位置在左上角
 matrix.preTranslate(-buttonWidth / 2, -buttonHeight / 2);
 // 将按钮移到中心位置
 matrix.postTranslate((width - buttonWidth) / 2, (height - buttonHeight) / 2);
 //设置抗锯齿
 canvas.setDrawFilter(paintFlagsDrawFilter);
 canvas.drawBitmap(buttonImage, matrix, buttonPaint);
}


绘制温度



private void drawTemp(Canvas canvas) {
 canvas.save();
 canvas.translate(getWidth() / 2, getHeight() / 2);
 float tempWidth = tempPaint.measureText(temperature + "");
 float tempHeight = (tempPaint.ascent() + tempPaint.descent()) / 2;
 canvas.drawText(temperature + "°", -tempWidth / 2 - dp2px(5), -tempHeight, tempPaint);
 canvas.restore();
}


处理滑动事件


private boolean isDown;
private boolean isMove;
@Override
public boolean onTouchEvent(MotionEvent event) {
 switch (event.getAction()) {
  case MotionEvent.ACTION_DOWN:
   isDown = true;
   float downX = event.getX();
   float downY = event.getY();
   currentAngle = calcAngle(downX, downY);
   break;
  case MotionEvent.ACTION_MOVE:
   isMove = true;
   float targetX;
   float targetY;
   downX = targetX = event.getX();
   downY = targetY = event.getY();
   float angle = calcAngle(targetX, targetY);
   // 滑过的角度增量
   float angleIncreased = angle - currentAngle;
   // 防止越界
   if (angleIncreased < -270) {
    angleIncreased = angleIncreased + 360;
   } else if (angleIncreased > 270) {
    angleIncreased = angleIncreased - 360;
   }
   IncreaseAngle(angleIncreased);
   currentAngle = angle;
   invalidate();
   break;
  case MotionEvent.ACTION_CANCEL:
  case MotionEvent.ACTION_UP: {
   if (isDown && isMove) {
    // 纠正指针位置
    rotateAngle = (float) ((temperature - minTemp) * angleRate * 4.5);
    invalidate();
    // 回调温度改变监听
    onTempChangeListener.change(temperature);
    isDown = false;
    isMove = false;
   }
   break;
  }
 }
 return true;
}

private float calcAngle(float targetX, float targetY) {
 float x = targetX - width / 2;
 float y = targetY - height / 2;
 double radian;
 if (x != 0) {
  float tan = Math.abs(y / x);
  if (x > 0) {
   if (y >= 0) {
    radian = Math.atan(tan);
   } else {
    radian = 2 * Math.PI - Math.atan(tan);
   }
  } else {
   if (y >= 0) {
    radian = Math.PI - Math.atan(tan);
   } else {
    radian = Math.PI + Math.atan(tan);
   }
  }
 } else {
  if (y > 0) {
   radian = Math.PI / 2;
  } else {
   radian = -Math.PI / 2;
  }
 }
 return (float) ((radian * 180) / Math.PI);
}

private void IncreaseAngle(float angle) {
 rotateAngle += angle;
 if (rotateAngle < 0) {
  rotateAngle = 0;
 } else if (rotateAngle > 270) {
  rotateAngle = 270;
 }
 temperature = (int) (rotateAngle / 4.5) / angleRate + minTemp;
}

提供一些接口方法



public void setTemp(int minTemp, int maxTemp, int temp) {
 this.minTemp = minTemp;
 this.maxTemp = maxTemp;
 this.temperature = temp;
 this.angleRate = 60 / (maxTemp - minTemp);
 rotateAngle = (float) ((temp - minTemp) * angleRate * 4.5);
 invalidate();
}

public void setOnTempChangeListener(OnTempChangeListener onTempChangeListener) {
 this.onTempChangeListener = onTempChangeListener;
}

public interface OnTempChangeListener {
 
 void change(int temp);
}

总结

以上就是这篇文章的全部内容了,希望本文的内容对各位Android开发者们能有所帮助,如果有疑问大家可以留言交流。

您可能感兴趣的文章:Android自定义实现开关按钮代码Android自定义View实现拖动选择按钮Android自定义View制作动态炫酷按钮实例解析Android自定义控件之开关按钮学习笔记分享自定义滑动按钮为例图文剖析Android自定义View绘制Android如何自定义按钮效果Android 自定义按钮点击事件和长按事件对比


--结束END--

本文标题: Android自定义控件实现温度旋转按钮效果

本文链接: https://lsjlt.com/news/22979.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

猜你喜欢
  • Android自定义控件实现温度旋转按钮效果
    首先看下效果图 温度旋转按钮 实现思路 初始化一些参数 绘制刻度盘 绘制刻度盘下的圆弧 绘制标题与温度标识 绘制旋转按钮 绘制温度 处理滑动事...
    99+
    2022-06-06
    按钮 Android
  • Android如何自定义按钮效果
    安卓原生的按钮是多么丑,效果是多么单调,大家也是有目共睹的。 要做一个APP少不了使用按钮,一个好看的按钮少不了好看的效果和外表,这次主要跟大家讲讲如何用drawable的x...
    99+
    2022-06-06
    自定义 按钮 Android
  • Android自定义悬浮按钮效果
    本文实例为大家分享了Android自定义悬浮按钮效果的具体代码,供大家参考,具体内容如下 以下:内容没有参考,写的也是一个比较简单的例子,主要就是应用切换前后台时会显示/隐藏悬浮窗。...
    99+
    2024-04-02
  • android自定义按钮示例(重写imagebutton控件实现图片按钮)
    由于项目这种类型的图片按钮比较多,所以重写了ImageButton类。 代码如下:package me.henji.widget; import android.conten...
    99+
    2022-06-06
    示例 图片 按钮 Android
  • Android 自定义Button控件实现按钮点击变色
    效果图如下所示: 一、shape 样式:(在drawable新建--》new--》Drawable resource file 在父级标签selector添加Item ) ...
    99+
    2022-06-06
    button 按钮 Android
  • Android自定义控件实现时钟效果
    在学习安卓群英传自定义控件章节的时候,有一个例子是绘制时钟,在实现了书上的例子后就想看这个时钟能不能动起来。 这里选择延迟一秒发送消息重绘view来实现的动画,对外提供了开启时...
    99+
    2022-06-06
    Android
  • Android SeekBar 自定义thumb旋转动画效果
    目录简介示例dimens.xmldrawableshape_thumb_round_1.xmllayers_thumb_ring_sweep_1.xmlrotate_thumb_1....
    99+
    2024-04-02
  • Android自定义View实现叶子飘动旋转效果(四)
    上一篇实现了叶子飘动功能,《Android自定义叶子飘动》 现在实现旋转效果 要实现这个效果,要在之前的功能上添加2个功能 1、通过matrix.postTranslate(...
    99+
    2022-06-06
    view Android
  • Android自定义控件实现方向盘效果
    在很多开发中,为了界面更加的友好,在自定义View的基础上,开发者会开发出各种各样的自定义控件来满足实际开发需要,其中有一种”方向盘”的控件在实际开发中非常常见,便于用户进行一...
    99+
    2022-06-06
    Android
  • Android自定义View控件实现刷新效果
    三种得到LinearInflater的方法 a. LayoutInflater inflater = getLayoutInflater(); b. LayoutInflate...
    99+
    2022-06-06
    view Android
  • Android自定义控件实现雷达图效果
    本文实例为大家分享了Android自定义控件实现雷达图的具体代码,供大家参考,具体内容如下 学习了大神的源代码(奈何不知大神的博客地址),觉得必须记录一下,方便以后再次学习。 效果如...
    99+
    2024-04-02
  • Android如何自定义Switch开关按钮控件
    这篇“Android如何自定义Switch开关按钮控件”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“Android如何自定义...
    99+
    2023-07-02
  • Android自定义View实现开关按钮
     前言:Android自定义View对于刚入门乃至工作几年的程序员来说都是非常恐惧的,但也是Android进阶学习的必经之路,平时项目中经常会有一些苛刻的需求,我们可...
    99+
    2022-06-06
    view 开关 按钮 Android
  • Android ImageButton自定义按钮的按下效果的代码实现方法分享
    使用Button时为了让用户有“按下”的效果,有两种实现方式:1.在代码里面。 代码如下:imageButton.setOnTouchListener(new OnTouch...
    99+
    2022-06-06
    方法 按钮 Android
  • Android自定义控件实现滑动开关效果
    自定义开关控件   Android自定义控件一般有三种方式 1、继承Android固有的控件,在Android原生控件的基础上,进行添加功能和逻辑。 2、继承V...
    99+
    2022-06-06
    开关 Android
  • 基于Android自定义控件实现雷达效果
    如何制作出类似雷达扫描的效果,具体方法如下一、效果图二、实现思路 自定义控件RadarView用来画雷达的效果图,可以自定义属性包括 backgroundColor:背景颜色 circleNum:圆的数量 startColor:开始颜色 e...
    99+
    2023-05-30
    android 雷达 roi
  • Android SeekBar如何自定义thumb旋转动画效果
    这篇文章给大家分享的是有关Android SeekBar如何自定义thumb旋转动画效果的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。示例dimens.xml为方便管理,可以添加一些尺寸设置<dimen&n...
    99+
    2023-06-25
  • Android自定义TimeButton实现倒计时按钮
    项目需要要实现一个带有倒计时功能的按钮,其效果类似发送验证码之后在按钮上显示倒计时并且将按钮设置为不可用的功能。 为了项目中其他地方能够调用到,便重写了一个继承于Button的...
    99+
    2022-06-06
    倒计时 按钮 Android
  • Android自定义实现开关按钮代码
    我们在应用中经常看到一些选择开关状态的配置文件,做项目的时候用的是android的Switch控件,但是感觉好丑的样子子 个人认为还是自定义的比较好,先上个效果图: 实...
    99+
    2022-06-06
    开关 按钮 Android
  • Android怎么实现自定义开关按钮
    这篇文章主要讲解了“Android怎么实现自定义开关按钮”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“Android怎么实现自定义开关按钮”吧!一、原理我们在界面的某一个区域里放置一个背景图...
    99+
    2023-06-30
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作