基于Android自定义控件实现雷达效果
发布时间 - 2026-01-11 02:28:22 点击率:次如何制作出类似雷达扫描的效果,具体方法如下

一、效果图
二、实现思路
1、自定义控件RadarView用来画雷达的效果图,可以自定义属性包括
backgroundColor:背景颜色
circleNum:圆的数量
startColor:开始颜色
endColor:结束颜色
lineColor:线的颜色
2、通过Handler循环发送消息到MessageQueue中,将mRotate加3,使Matrix旋转mRotate,重绘雷达扫描的圆。
3、通过梯度渐变扫描渲染器SweepGradient,在绘制圆的过程中,将颜色从startColor变为endColor。
三、实例代码
public class RadarView extends View {
private final String TAG = "RadarView";
private static final int MSG_WHAT = 1;
private static final int DELAY_TIME = 20;
//设置默认宽高,雷达一般都是圆形,所以我们下面取宽高会取Math.min(宽,高)
private final int DEFAULT_WIDTH = 200;
private final int DEFAULT_HEIGHT = 200;
//雷达的半径
private int mRadarRadius;
//雷达画笔
private Paint mRadarPaint;
//雷达底色画笔
private Paint mRadarBg;
//雷达圆圈的个数,默认4个
private int mCircleNum = 4;
//雷达线条的颜色,默认为白色
private int mCircleColor = Color.WHITE;
//雷达圆圈背景色
private int mRadarBgColor = Color.BLACK;
//paintShader
private Shader mRadarShader;
//雷达扫描时候的起始和终止颜色
private int mStartColor = 0x0000ff00;
private int mEndColor = 0xaa00ff00;
private Matrix mMatrix;
//旋转的角度
private int mRotate = 0;
private Handler mHandler = new Handler() {
@Override
public void handleMessage(Message msg) {
super.handleMessage(msg);
mRotate += 3;
postInvalidate();
mMatrix.reset();
mMatrix.preRotate(mRotate, 0, 0);
//延时DELAY_TIME后再发送消息
mHandler.sendEmptyMessageDelayed(MSG_WHAT, DELAY_TIME);
}
};
public RadarView(Context context) {
this(context, null);
}
public RadarView(Context context, AttributeSet attrs) {
this(context, attrs, 0);
}
public RadarView(Context context, AttributeSet attrs, int defStyleAttr) {
super(context, attrs, defStyleAttr);
init(context, attrs);
//设置抗锯齿
mRadarBg = new Paint(Paint.ANTI_ALIAS_FLAG);
//画笔颜色
mRadarBg.setColor(mRadarBgColor);
//画实心圆
mRadarBg.setStyle(Paint.Style.FILL);
//设置抗锯齿
mRadarPaint = new Paint(Paint.ANTI_ALIAS_FLAG);
//画笔颜色
mRadarPaint.setColor(mCircleColor);
//设置空心的画笔,只画圆边
mRadarPaint.setStyle(Paint.Style.STROKE);
//画笔宽度
mRadarPaint.setStrokeWidth(2);
//使用梯度渐变渲染器,
mRadarShader = new SweepGradient(0, 0, mStartColor, mEndColor);
mMatrix = new Matrix();
}
//初始化,拓展可设置参数供布局使用
private void init(Context context, AttributeSet attrs) {
if (attrs != null) {
TypedArray ta = context.obtainStyledAttributes(attrs, R.styleable.RadarView);
mStartColor = ta.getColor(R.styleable.RadarView_startColor, mStartColor);
mEndColor = ta.getColor(R.styleable.RadarView_endColor, mEndColor);
mRadarBgColor = ta.getColor(R.styleable.RadarView_backgroundColor, mRadarBgColor);
mCircleColor = ta.getColor(R.styleable.RadarView_lineColor, mCircleColor);
mCircleNum = ta.getInteger(R.styleable.RadarView_circleNum, mCircleNum);
ta.recycle();
}
}
@Override
protected void onSizeChanged(int w, int h, int oldw, int oldh) {
super.onSizeChanged(w, h, oldw, oldh);
//雷达的半径为宽的一半或高的一半的最小值
mRadarRadius = Math.min(w / 2, h / 2);
}
@Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
super.onMeasure(widthMeasureSpec, heightMeasureSpec);
//获取宽度
int width = measureSize(1, DEFAULT_WIDTH, widthMeasureSpec);
//获取高度
int height = measureSize(0, DEFAULT_HEIGHT, heightMeasureSpec);
//取最大的 宽|高
int measureSize = Math.max(width, height);
setMeasuredDimension(measureSize, measureSize);
}
/**
* 测绘measure
*
* @param specType 1为宽, 其他为高
* @param contentSize 默认值
*/
private int measureSize(int specType, int contentSize, int measureSpec) {
int result;
//获取测量的模式和Size
int specMode = MeasureSpec.getMode(measureSpec);
int specSize = MeasureSpec.getSize(measureSpec);
if (specMode == MeasureSpec.EXACTLY) {
result = Math.max(contentSize, specSize);
} else {
result = contentSize;
if (specType == 1) {
// 根据传入方式计算宽
result += (getPaddingLeft() + getPaddingRight());
} else {
// 根据传入方式计算高
result += (getPaddingTop() + getPaddingBottom());
}
}
return result;
}
@Override
protected void onDraw(Canvas canvas) {
super.onDraw(canvas);
Log.d(TAG, "onDraw " + mRotate);
mRadarBg.setShader(null);
//将画布移动到屏幕的中心点
canvas.translate(mRadarRadius, mRadarRadius);
//绘制底色,让雷达的线看起来更清晰
canvas.drawCircle(0, 0, mRadarRadius, mRadarBg);
//画圆圈
for (int i = 1; i <= mCircleNum; i++) {
canvas.drawCircle(0, 0, (float) (i * 1.0 / mCircleNum * mRadarRadius), mRadarPaint);
}
//绘制雷达基线 x轴
canvas.drawLine(-mRadarRadius, 0, mRadarRadius, 0, mRadarPaint);
//绘制雷达基线 y轴
canvas.drawLine(0, mRadarRadius, 0, -mRadarRadius, mRadarPaint);
//设置颜色渐变从透明到不透明
mRadarBg.setShader(mRadarShader);
//设置矩阵
canvas.concat(mMatrix);
canvas.drawCircle(0, 0, mRadarRadius, mRadarBg);
}
public void startScan() {
mHandler.removeMessages(MSG_WHAT);
mHandler.sendEmptyMessage(MSG_WHAT);
}
public void stopScan() {
mHandler.removeMessages(MSG_WHAT);
}
}
布局文件:
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:app="http://schemas.android.com/apk/res-auto" xmlns:tools="http://schemas.android.com/tools" android:id="@+id/activity_main" android:layout_width="match_parent" android:layout_height="match_parent" > <com.android.demo.ui.shader.RadarView android:id="@+id/radarview" android:layout_width="300dp" android:layout_height="300dp" android:layout_centerInParent="true" app:backgroundColor="#000000" app:circleNum="4" app:endColor="#aaff0000" app:lineColor="#00ff00" app:startColor="#aa0000ff"/> <Button android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_alignParentLeft="true" android:layout_alignParentBottom="true" android:onClick="start" android:text="开始" /> <Button android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_alignParentRight="true" android:layout_alignParentBottom="true" android:onClick="stop" android:text="停止" /> </RelativeLayout>
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。
# Android
# 雷达效果
# android自定义控件ImageView实现圆形图片
# Android自定义控件ImageView实现点击之后出现阴影效果
# Android自定义控件ViewFipper实现竖直跑马灯效果
# Android自定义控件打造绚丽平行空间引导页
# Android自定义控件EditText实现清除和抖动功能
# Android自定义控件EditText使用详解
# Android自定义控件实现下拉刷新效果
# Android编程实现自定义控件的方法示例
# Android自定义控件之日期选择控件使用详解
# Android自定义控件实现九宫格解锁功能
# 实例讲解Android自定义控件
# 自定义
# 都是
# 抗锯齿
# 渲染器
# 中心点
# 他为
# 再发
# 大家多多
# 过程中
# 默认值
# 方法如下
# 背景色
# 更清晰
# 默认为
# 不透明
# 发送消息
# 最小值
# 高会
# mRadarShader
# mStartColor
相关栏目:
【
网站优化151355 】
【
网络推广146373 】
【
网络技术251813 】
【
AI营销90571 】
相关推荐:
Laravel Fortify是什么,和Jetstream有什么关系
公司网站制作需要多少钱,找人做公司网站需要多少钱?
东莞专业网站制作公司有哪些,东莞招聘网站哪个好?
Laravel如何实现用户注册和登录?(Auth脚手架指南)
如何用AI一键生成爆款短视频文案?小红书AI文案写作指令【教程】
Laravel Livewire是什么_使用Laravel Livewire构建动态前端界面
简历没回改:利用AI润色让你的文字更专业
Win10如何卸载预装Edge扩展_Win10卸载Edge扩展教程【方法】
linux写shell需要注意的问题(必看)
Laravel如何部署到服务器_线上部署Laravel项目的完整流程与步骤
Laravel如何实现模型的全局作用域?(Global Scope示例)
北京网站制作的公司有哪些,北京白云观官方网站?
零服务器AI建站解决方案:快速部署与云端平台低成本实践
七夕网站制作视频,七夕大促活动怎么报名?
Laravel事件监听器怎么写_Laravel Event和Listener使用教程
Laravel如何实现多语言支持_Laravel本地化与国际化(i18n)配置教程
Laravel策略(Policy)如何控制权限_Laravel Gates与Policies实现用户授权
香港服务器选型指南:免备案配置与高效建站方案解析
高端云建站费用究竟需要多少预算?
Laravel如何处理文件下载请求?(Response示例)
手机网站制作与建设方案,手机网站如何建设?
佛山企业网站制作公司有哪些,沟通100网上服务官网?
如何获取PHP WAP自助建站系统源码?
公司门户网站制作公司有哪些,怎样使用wordpress制作一个企业网站?
如何用狗爹虚拟主机快速搭建网站?
Laravel如何处理异常和错误?(Handler示例)
详解jQuery停止动画——stop()方法的使用
JavaScript Ajax实现异步通信
消息称 OpenAI 正研发的神秘硬件设备或为智能笔,富士康代工
,交易猫的商品怎么发布到网站上去?
怎么制作网站设计模板图片,有电商商品详情页面的免费模板素材网站推荐吗?
新三国志曹操传主线渭水交兵攻略
Laravel项目如何进行性能优化_Laravel应用性能分析与优化技巧大全
大学网站设计制作软件有哪些,如何将网站制作成自己app?
教学论文网站制作软件有哪些,写论文用什么软件
?
Laravel怎么做数据加密_Laravel内置Crypt门面的加密与解密功能
uc浏览器二维码扫描入口_uc浏览器扫码功能使用地址
深圳网站制作设计招聘,关于服装设计的流行趋势,哪里的资料比较全面?
网站制作怎么样才能赚钱,用自己的电脑做服务器架设网站有什么利弊,能赚钱吗?
Edge浏览器提示“由你的组织管理”怎么解决_去除浏览器托管提示【修复】
Laravel如何与Docker(Sail)协同开发?(环境搭建教程)
如何使用 Go 正则表达式精准提取括号内首个纯字母标识符(忽略数字与嵌套)
Laravel的.env文件有什么用_Laravel环境变量配置与管理详解
jQuery validate插件功能与用法详解
🚀拖拽式CMS建站能否实现高效与个性化并存?
Java垃圾回收器的方法和原理总结
JS经典正则表达式笔试题汇总
韩国服务器如何优化跨境访问实现高效连接?
如何快速搭建安全的FTP站点?
Laravel如何操作JSON类型的数据库字段?(Eloquent示例)

