广告
返回顶部
首页 > 资讯 > 移动开发 >Android自定义View实现折线图效果
  • 341
分享到

Android自定义View实现折线图效果

折线图viewAndroid 2022-06-06 07:06:13 341人浏览 泡泡鱼
摘要

下面就是结果图(每种状态用一个表情图片表示): 一、主页面的布局文件如下: <RelativeLayout xmlns:Android="Http://schema

下面就是结果图(每种状态用一个表情图片表示):

一、主页面的布局文件如下:


<RelativeLayout xmlns:Android="Http://schemas.android.com/apk/res/android" 
 xmlns:tools="http://schemas.android.com/tools" 
 android:layout_width="match_parent" 
 android:layout_height="match_parent" 
 tools:context=".MainActivity" 
 xmlns:app="http://schemas.android.com/apk/res/ting.example.linecharview"> 
 <ting.example.linecharview.LineCharView 
 android:id="@+id/test" 
 android:layout_width="match_parent" 
 android:layout_height="match_parent" 
 app:xytextcolor="@color/bg" 
 app:xytextsize="20sp" 
 app:interval="80dp" 
 /> 
</RelativeLayout> 

其中

linecharview
就是自定义的
View
,而
app:xx
就是这个
View
的各种属性。

二、在

values
的attrs文件中加入如下xml,来定义
linecharview
的各种属性:


<?xml version="1.0" encoding="utf-8"?> 
<resources> 
 <declare-styleable name="LineChar"> 
 <attr name="xylinecolor" fORMat="color"/><!-- xy坐标轴颜色 --> 
 <attr name="xylinewidth" format="dimension"/><!-- xy坐标轴宽度 --> 
 <attr name="xytextcolor" format="color"/><!-- xy坐标轴文字颜色 --> 
 <attr name="xytextsize" format="dimension"/><!-- xy坐标轴文字大小 --> 
 <attr name="linecolor" format="color"/><!-- 折线图中折线的颜色 --> 
 <attr name="interval" format="dimension"/><!-- x轴各个坐标点水平间距 --> 
 <attr name="bGColor" format="color"/><!-- 背景颜色 --> 
 </declare-styleable> 
</resources> 

三、接下来建个类

LineCharView 
继承
View
,并申明如下变量:


<span style="white-space:pre"> </span>private int xori;//圆点x坐标 
 private int yori;//圆点y坐标 
 private int xinit;//第一个点x坐标 
 private int minXinit;//在移动时,第一个点允许最小的x坐标 
 private int maxXinit;//在移动时,第一个点允许允许最大的x坐标 
 private int xylinecolor;//xy坐标轴颜色 
 private int xylinewidth;//xy坐标轴大小 
 private int xytextcolor;//xy坐标轴文字颜色 
 private int xytextsize;//xy坐标轴文字大小 
 private int linecolor;//折线的颜色 
 private int interval;//坐标间的间隔 
 private int bgColor;//背景颜色 
 private List<String> x_coords;//x坐标点的值 
 private List<String> x_coord_values;//每个点状态值 
 private int width;//控件宽度 
 private int heigth;//控件高度 
 private int imageWidth;//表情的宽度 
 private float textwidth;//y轴文字的宽度 
 float startX=0;//滑动时候,上一次手指的x坐标 

在构造函数中读取各个属性值:


public LineCharView(Context context, AttributeSet attrs) { 
 super(context, attrs); 
 TypedArray typedArray= context.obtainStyledAttributes(attrs, R.styleable.LineChar); 
 xylinecolor=typedArray.getColor(R.styleable.LineChar_xylinecolor, Color.GRAY); 
 xylinewidth=typedArray.getInt(R.styleable.LineChar_xylinewidth, 5); 
 xytextcolor=typedArray.getColor(R.styleable.LineChar_xytextcolor, Color.BLACK); 
 xytextsize=typedArray.getLayoutDimension(R.styleable.LineChar_xytextsize, 20); 
 linecolor=typedArray.getColor(R.styleable.LineChar_linecolor, Color.GRAY); 
 interval=typedArray.getLayoutDimension(R.styleable.LineChar_interval, 100); 
 bgColor=typedArray.getColor(R.styleable.LineChar_bgcolor, Color.WHITE); 
 typedArray.recycle(); 
 x_coords=new ArrayList<String>(); 
 x_coord_values=new ArrayList<String>(); 
} 

四、接下来可以重写

onLayout
方法,来计算控件宽高和坐标轴的原点坐标,坐标轴原点的x坐标可以通过y轴文字的宽度,y轴宽度,和距离y的水平距离进行计算,这里y轴文字只有4种状态(A、B、C、D),可以使用下面方法来计算出原点的x坐标:


Paint paint=new Paint(); 
paint.setTextSize(xytextsize); 
textwidth= paint.measureText("A"); 
xori=(int) (textwidth+6+2*xylinewidth);//6 为与y轴的间隔 

原点的y坐标也可以用类似的方法计算出来:


yori=heigth-xytextsize-2*xylinewidth-3; //3为x轴的间隔,heigth为控件高度。 

当需要展示的数据量多时候,无法全部展示时候,需要通过滑动折线图进行展示,我们只需要控制第一点x坐标,就可以通过interval这个属性计算出后面每个点的坐标,但是为了防止将所有的数据滑动出界面外,需要在滑动时进行控制,其实就是控制第一个点x坐标的范围,第一个点的x坐标的最小值可以通过控件的宽度减去原点x坐标再减去所有折线图的水平距离,代码如下:


minXinit=width-xori-x_coords.size()*interval; 

控件在默认第一个展示时,第一个点与y轴的水平距离等于

interval
的一半,在滑动时候如果第一个点出现在这个位置了,就不允许再继续向右滑动,所以第一个点x坐标的最大值就等这个起始x坐标。


xinit=interval/2+xori; 
maxXinit=xinit; 

重写

onLayout
方法的代码如下:


@Override 
 protected void onLayout(boolean changed, int left, int top, int right, 
 int bottom) { 
 if(changed){ 
 width=getWidth(); 
 heigth=getHeight(); 
 Paint paint=new Paint(); 
 paint.setTextSize(xytextsize); 
 textwidth= paint.measureText("A"); 
 xori=(int) (textwidth+6+2*xylinewidth);//6 为与y轴的间隔 
 yori=heigth-xytextsize-2*xylinewidth-3;//3为x轴的间隔 
 xinit=interval/2+xori; 
 imageWidth= BitmapFactory.decodeResource(getResources(), R.drawable.facea).getWidth(); 
 minXinit=width-xori-x_coords.size()*interval; 
 maxXinit=xinit; 
 setBackgroundColor(bgColor); 
 } 
 super.onLayout(changed, left, top, right, bottom); 
 } 

五、接下来就可以画折线、x坐标轴上的小圆点和折线上表情

代码如下:


//画X轴坐标点,折线,表情 
 @SuppressLint("ResourceAsColor") 
 private void drawX (canvas canvas) { 
 Paint x_coordPaint =new Paint(); 
 x_coordPaint.setTextSize(xytextsize); 
 x_coordPaint.setStyle(Paint.Style.FILL); 
 Path path=new Path(); 
 //画坐标轴上小原点,坐标轴文字 
 for(int i=0;i<x_coords.size();i++){ 
 int x=i*interval+xinit; 
 if(i==0){ 
 path.moveTo(x, getYValue(x_coord_values.get(i))); 
 }else{ 
 path.lineTo(x, getYValue(x_coord_values.get(i))); 
 } 
 x_coordPaint.setColor(xylinecolor); 
 canvas.drawCircle(x, yori, xylinewidth*2, x_coordPaint); 
 String text=x_coords.get(i); 
 x_coordPaint.setColor(xytextcolor); 
 canvas.drawText(text, x-x_coordPaint.measureText(text)/2, yori+xytextsize+xylinewidth*2, x_coordPaint); 
 } 
 x_coordPaint.setStyle(Paint.Style.STROKE); 
 x_coordPaint.setStrokeWidth(xylinewidth); 
 x_coordPaint.setColor(linecolor); 
 //画折线 
 canvas.drawPath(path, x_coordPaint); 
 //画表情 
 for(int i=0;i<x_coords.size();i++){ 
 int x=i*interval+xinit; 
 canvas.drawBitmap(getYBitmap(x_coord_values.get(i)), x-imageWidth/2, getYValue(x_coord_values.get(i))-imageWidth/2, x_coordPaint); 
 } 
 //将折线超出x轴坐标的部分截取掉 
 x_coordPaint.setStyle(Paint.Style.FILL); 
 x_coordPaint.setColor(bgColor); 
 x_coordPaint.setXfermode(new PorterDuffXfermode( PorterDuff.Mode.SRC_OVER)); 
 RectF rectF=new RectF(0, 0, xori, heigth); 
 canvas.drawRect(rectF, x_coordPaint); 
 } 

以上代码首先通过遍历

x_coords
x_coord_values
这两个List集合,来画坐标点,折线,表情,由于在向左滑动的时候有可能会将坐标点,折线绘制到y轴的左边,所以需要对其进行截取。其中
getYValue
和getYBitmap方法,可以通过
x_coord_values
的值计算y坐标和相应的表情。两方法如:


//得到y坐标 
 private float getYValue(String value) 
 { 
 if(value.equalsIgnoreCase("A")){ 
 return yori-interval/2; 
 } 
 else if(value.equalsIgnoreCase("B")){ 
 return yori-interval; 
 } 
 else if(value.equalsIgnoreCase("C")){ 
 return (float) (yori-interval*1.5); 
 } 
 else if(value.equalsIgnoreCase("D")){ 
 return yori-interval*2; 
 }else{ 
 return yori; 
 } 
 } 
 //得到表情图 
 private Bitmap getYBitmap(String value){ 
 Bitmap bitmap=null; 
 if(value.equalsIgnoreCase("A")){ 
 bitmap=BitmapFactory.decodeResource(getResources(), R.drawable.facea); 
 } 
 else if(value.equalsIgnoreCase("B")){ 
 bitmap=BitmapFactory.decodeResource(getResources(), R.drawable.faceb); 
 } 
 else if(value.equalsIgnoreCase("C")){ 
 bitmap=BitmapFactory.decodeResource(getResources(), R.drawable.facec); 
 } 
 else if(value.equalsIgnoreCase("D")){ 
 bitmap=BitmapFactory.decodeResource(getResources(), R.drawable.faced); 
 } 
 return bitmap; 
 } 

六、画好了坐标点,折线,表情,接下来就简单,就可以画x y轴了,x y轴只要确定的原点坐标,就非常简单了,代码如下:


//画坐标轴 
private void drawXY(Canvas canvas){ 
 Paint paint=new Paint(); 
 paint.setColor(xylinecolor); 
 paint.setStrokeWidth(xylinewidth); 
 canvas.drawLine(xori, 0, xori, yori, paint); 
 canvas.drawLine(xori, yori, width, yori, paint); 
} 

七、最后就可以画y轴上的坐标小原点和y轴的文字了:


//画Y轴坐标点 
 private void drawY(Canvas canvas){ 
 Paint paint=new Paint(); 
 paint.setColor(xylinecolor); 
 paint.setStyle(Paint.Style.FILL); 
 for(int i=1;i<5 ;i++){ 
 canvas.drawCircle(xori, yori-(i*interval/2), xylinewidth*2, paint); 
 } 
 paint.setTextSize(xytextsize); 
 paint.setColor(xytextcolor); 
 canvas.drawText("D",xori-textwidth-6-xylinewidth , yori-(2*interval)+xytextsize/2, paint); 
 canvas.drawText("C",xori-textwidth-6-xylinewidth , (float) (yori-(1.5*interval)+xytextsize/2), paint); 
 canvas.drawText("B",xori-textwidth-6-xylinewidth , yori-interval+xytextsize/2, paint); 
 canvas.drawText("A",xori-textwidth-6-xylinewidth , (float) (yori-(0.5*interval)+xytextsize/2), paint); 
 } 

八、写完了以上三个方法:只需要重写

onDraw
方法,就可以进行绘制了。


@Override 
 protected void onDraw(Canvas canvas) { 
 drawX(canvas); 
 drawXY(canvas); 
 drawY(canvas); 
 } 

九、为了可以进行水平滑动,需要重写控件的onTouchEvent方法,在滑动时候,实时计算手指滑动的距离来改变第一个点的x坐标,然后调用

invalidate();
就可以刷新控件,重新绘制达到滑动效果。


@Override 
 public boolean onTouchEvent(MotionEvent event) { 
 //如果不用滑动就可以展示所有数据,就不让滑动 
 if(interval*x_coord_values.size()<=width-xori){ 
 return false; 
 } 
 switch (event.getAction()) { 
 case MotionEvent.ACTION_DOWN: 
 startX=event.getX(); 
 break; 
 case MotionEvent.ACTION_MOVE: 
 float dis=event.getX()-startX; 
 startX=event.getX(); 
 if(xinit+dis>maxXinit){ 
 xinit=maxXinit; 
 }else if(xinit+dis<minXinit){ 
 xinit=minXinit; 
 }else{ 
 xinit=(int) (xinit+dis); 
 } 
 invalidate(); 
 break; 
 } 
 return true; 
 } 

十、最后添加一个设置数据源的方法,设置

x_coords
x_coord_values
这个两个
List
集合,在设置完成之后调用
invalidate() 
,进行控件刷新:


 
public void setValue( List<String> x_coords ,List<String> x_coord_values) { 
 if(x_coord_values.size()!=x_coords.size()){ 
 throw new IllegalArgumentException("坐标轴点和坐标轴点的值的个数必须一样!"); 
 } 
 this.x_coord_values=x_coord_values; 
 this.x_coords=x_coords; 
 invalidate(); 
} 

总结

以上就是Android自定义View实现折线图效果的全部内容,希望对大家开发Android能有所帮助。

您可能感兴趣的文章:Android自定义可左右滑动和点击的折线图Android绘制动态折线图Android自定义控件实现折线图Android HelloChart开源库图表之折线图的实例代码Android MPAndroidChart开源库图表之折线图的实例代码Android开发实现绘制淘宝收益图折线效果示例Android自定义View简易折线图控件(二)详解Android图表 MPAndroidChart折线图Android开发之天气趋势折线图MPAndroidChart开源图表库的使用介绍之饼状图、折线图和柱状图Android实现折线走势图


--结束END--

本文标题: Android自定义View实现折线图效果

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

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

本篇文章演示代码以及资料文档资料下载

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

下载Word文档
猜你喜欢
  • Android自定义View实现折线图效果
    下面就是结果图(每种状态用一个表情图片表示): 一、主页面的布局文件如下: <RelativeLayout xmlns:android="http://schema...
    99+
    2022-06-06
    折线图 view Android
  • Android自定义View实现天气预报折线图
    本文实例为大家分享了Android自定义View画天气预报折线图的具体代码,供大家参考,具体内容如下 效果图如下: 刚开始尝试用第三方画曲线的框架来画效果图,后来发现曲线间的阴影当...
    99+
    2022-11-13
  • Android 自定义View实现芝麻分曲线图效果
    1.简介 其实这个效果几天之前就写了,但是一直没有更新博客,本来想着把芝麻分雷达图也做好再发博客的,然后今天看到鸿洋的微信公众号有朋友发了芝麻分的雷达图,所以就算了,算是一个互...
    99+
    2022-06-06
    view 自定义view Android
  • Android自定义View之渐变色折线图的实现
    目录前言如何实现总结前言 在之前的项目中,有做过一个需求,需要实现一个颜色渐变的折线图。当时项目中使用的图表库是MPAndroidChart,但是该库没有提供合适的方法来实现想要的效...
    99+
    2022-11-13
  • Android怎么自定义View实现渐变色折线图
    这篇文章主要介绍“Android怎么自定义View实现渐变色折线图”,在日常操作中,相信很多人在Android怎么自定义View实现渐变色折线图问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”Android怎么...
    99+
    2023-06-30
  • Android自定义View简易折线图控件(二)
    继续练习自定义View,这次带来的是简易折线图,支持坐标点点击监听,效果如下: 画坐标轴、画刻度、画点、连线。。x、y轴的数据范围是写死的 1 <= x <= 7...
    99+
    2022-06-06
    折线图 view Android
  • Android 实现自定义折线图控件
    目录前言概述原点计算Y轴宽度计算X轴高度X轴绘制轴线X轴刻度间隔网格线、文本Y轴计算Y轴分布刻度间隔、网格线、文本折线代码前言 日前,有一个“折现图”的需求,...
    99+
    2022-11-13
  • android自定义view实现钟表效果
    本文实例为大家分享了android view实现钟表的具体代码,供大家参考,具体内容如下 先看效果图: 自定义view大家肯定已经不陌生了,所以直接今天直接步入正题:如何利用...
    99+
    2022-06-06
    view Android
  • Android自定义View实现扫描效果
    本文实例为大家分享了Android自定义View实现扫描效果的具体代码,供大家参考,具体内容如下 演示效果如下: 实现内容: 1、控制动画是竖向或者横向 2、控制动画初始是从底部/...
    99+
    2022-11-12
  • Android自定义View实现时钟效果
    本文实例为大家分享了Android自定义View实现时钟效果的具体代码,供大家参考,具体内容如下 自定义时钟 初学自定义View,先画一个不太成熟的时钟(甚至只有秒针) 时钟效果 ...
    99+
    2022-11-12
  • Android中自定义view实现侧滑效果
    效果图: 看网上的都是两个view拼接,默认右侧的不显示,水平移动的时候把右侧的view显示出来。但是看最新版QQ上的效果不是这样的,但给人的感觉却很好,所以献丑来一发比较高...
    99+
    2022-06-06
    view 自定义view Android
  • Android自定义Animation实现View摇摆效果
    使用自定义Animation,实现View的左右摇摆效果,如图所示: 代码很简单,直接上源码 activity_maini.xml布局文件: <?xml v...
    99+
    2022-06-06
    view animation Android
  • Android自定义View实现打字机效果
    一、先来看看效果演示 二、实现原理: 这个其实不难实现,通过一个定时器不断调用TextView的setText就行了,在setText的时候播放打字的音效。 具体代码如下:...
    99+
    2022-06-06
    view 打字机 Android
  • Android自定义view实现输入框效果
    本文实例为大家分享了Android自定义view实现输入框的具体代码,供大家参考,具体内容如下 自定义输入框的View package com.fenghongzhang.day...
    99+
    2022-11-11
  • Android自定义view实现半圆环效果
    本文实例为大家分享了Android自定义view实现半圆环的具体代码,供大家参考,具体内容如下 1.自定义属性 <declare-styleable name="Semicir...
    99+
    2022-11-13
  • Android自定义View实现标签流效果
    本文实例为大家分享了Android自定义View实现标签流效果的具体代码,供大家参考,具体内容如下 一、概述 Android自定义View实现标签流效果,一行放不下时会自动换行,用户...
    99+
    2022-11-13
  • Android自定义View实现水波纹效果
    介绍:水波纹散开效果的控件在 App 里面还是比较常见的,例如 网易云音乐歌曲识别,附近搜索场景。看下实现的效果:实现思路: 先将最大圆半径与最小圆半径间距分成几等份,从内到外,Paint 透明度依次递减,绘制出同心圆,然后不断的改变这些同...
    99+
    2023-05-30
    android view 水波纹
  • Android怎么实现自定义折线图控件
    这篇“Android怎么实现自定义折线图控件”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“Android怎么实现自定义折线图...
    99+
    2023-07-02
  • 自定义视图view的折线图使用讲解
    目录一、如何绘制X和Y轴。1、我们来分析下,我们想知道三个坐标,那么这三个坐标是多少呢,我们该怎么计算呢? 答:这里,我是在onSizeChanged()方法中获取到了父类控件的宽度...
    99+
    2023-05-14
    自定义视图 自定义view 自定义view折线图
  • 自定义视图view的折线图怎么使用
    这篇文章主要讲解了“自定义视图view的折线图怎么使用”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“自定义视图view的折线图怎么使用”吧!绘制折线图预览图绘制这个折线图需要都需要哪些步骤?...
    99+
    2023-07-05
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作