美文网首页IOSiOS K线iOS
iOS-Charts看这个就够了

iOS-Charts看这个就够了

作者: f62385835449 | 来源:发表于2018-02-07 23:17 被阅读3844次

            最近做项目需要画K线图和折线图,引入了第三方的图标库Charts。

    Charts

            这个图表库基本上能够满足大家对于图表绘制的需要,但是api接口的解释并不是很详细,该库虽然有强大的功能,但是开发者看了很长时间还是一头雾水。而且网上相关的资源不是很多,所以我写了这篇文章希望对大家有所帮助。

             咱们先看一下官方给出的Demo效果:

    折线图 柱状图 K线图

    看完了效果图,咱们逐一介绍各种图表如何绘制吧!

    第一步

    引入第三方图表绘制库Charts

    我用的是Carthage方法引入,直接在Cartfile中加入"github "danielgindi/Charts" ~> 3.0.4",然后在终端命令行输入:carthage update --platform iOS即可。

    倒入之后,大家的工程项目将会包含这个文件:

    由于这个框架使用swift语言写的,所以OC开发者需要在项目中添加桥接文件,"Charts-Swift"就是桥接文件,大家不用自定义了,想在哪个类中调用Charts,import "Charts-Swift.h"文件就可以了。

    第二步

    定义图表,设置图标的各种属性

    Charts包含的图表类很多,在此我举几个有代表性的例子:BarChartView(柱状图), LineChartView(折线图), CandleStickChartView(烛形图,K线图)等等。

    ChartsView的属性:

    1、scale属性

    顾名思义,就是图标是否支持拖拽,scaleYEnabled = YES即纵坐标支持拖拽,scaleXEnabled = YES即横坐标支持拖拽。

    2、chartDescription属性

    打开这个属性,你的图表上将会有关于你的图标的描述。例如: _chartView.chartDescription.enabled = YES; _chartView.chartDescription.text = @"王尼玛的chartsView";

    下面的这行小字"王尼玛的chartsView"就是我的ChartsView的description。

    3、pinchZoomEnabled

    是否支持X轴、Y轴同时缩放,如果是YES ,则代表支持同时缩放。

    4、maxVisibleCount

    这个属性是控制图标上的数字展示的最大个数的,如果你的图标不想显示上面的数字,就可以把maxVisibleCount属性设置为0。

    5、legend

    这个Bool属性是设置要不要显示图例的,看到上图底部的"-- DataSet 1"了吗,如果你不想要显示这一栏,就把legend.enabled设置为NO。

    6、noDataText

    没有数据的时候ChartsView上需要显示什么文字。

    chartsView中有三个重要的属性需要开发者去设置,那就是leftAxis(左轴),rightAxis(右轴),xAxis(X轴)。

    7、doubleTapToZoomEnabled

    是否允许双击缩放。

    8、delegate

    代理,大家都懂的。对应的protocal有4个,都是可选选项:

    图表中的数值被选中

    - (void)chartValueSelected:(ChartViewBase * _Nonnull)chartView entry:(ChartDataEntry * _Nonnull)entry highlight:(ChartHighlight * _Nonnull)highlight;

    图标中的空白区域被点击

    - (void)chartValueNothingSelected:(ChartViewBase * _Nonnull)chartView;

    图表缩放

    - (void)chartScaled:(ChartViewBase * _Nonnull)chartView scaleX:(CGFloat)scaleX scaleY:(CGFloat)scaleY;

    图标被移动

    - (void)chartTranslated:(ChartViewBase * _Nonnull)chartView dX:(CGFloat)dX dY:(CGFloat)dY;

    Axis属性

    1、enabled

    设置为NO的话,则对应的坐标轴不显示。

    2、labelPosition

    表示左轴坐标的位置,有2个选项:YAxisLabelPositionOutsideChart = 0,YAxisLabelPositionInsideChart = 1,坐标显示在轴内或者是轴外。

    3、labelCount

    即在坐标轴上最多显示多少个坐标点

    4、drawGridBackgroundEnabled

    是否要画网格

    5、gridLineDashLengths

    gridLineDashLengths = @[@2.f, @5.f];即线段宽为2.0f,空格宽为5.0f。

    第三步

    填充数据

    定义了表格的基本属性之后我们就要填充数据了。在这里我仅以折线图表为例。

    这里需要介绍两个概念:set和data,LineChartData就是折线图的data类,它可以由很多组set组成,一组就是一条折线。因此我们可以定义set的属性,从而绘制各种各样的折线。示例如下:

    //是否绘制图标

    set1.drawIconsEnabled = NO;

    //折线颜色

      [set1 setColor:UIColor.blackColor];

    //折线点的颜色

        [set1 setCircleColor:UIColor.blackColor];

    //折线的宽度

         set1.lineWidth = 1.0;

    //折线点的宽度

          set1.circleRadius = 3.0;

    //是否画空心圆

          set1.drawCircleHoleEnabled = NO;

    //折线点的值的大小

          set1.valueFont = [UIFont systemFontOfSize:9.f];

    //图例的线宽

          set1.formLineWidth = 1.0;

    //图例的字体大小

          set1.formSize = 15.0;

    下面具体介绍如何填充数据:

    //定义一个数组承接数据

    NSMutableArray *values = [[NSMutableArray alloc] init];

        for (int i = 0; i < count; i++){

            double val = arc4random_uniform(range) + 3;

    //将横纵坐标以ChartDataEntry的形式保存下来,注意横坐标值一般是i的值,而不是你的数据    //里面具体的值,如何将具体数据展示在X轴上我们下面将会说到。

            [values addObject:[[ChartDataEntry alloc] initWithX:i y:val icon: [UIImage imageNamed:@"icon"]]];

        }

        LineChartDataSet *set1 = nil;

    //请注意这里,如果你的图表以前绘制过,那么这里直接重新给data赋值就行了。

    //如果没有,那么要先定义set的属性。

        if (_chartView.data.dataSetCount > 0)

        {

            set1 = (LineChartDataSet *)_chartView.data.dataSets[0];

            set1.values = values;

    //通知data去更新

            [_chartView.data notifyDataChanged];

    //通知图表去更新

            [_chartView notifyDataSetChanged];

        }

        else

        {

            set1 = [[LineChartDataSet alloc] initWithValues:values label:@"DataSet 1"];

            //自定义set的各种属性

            [self configSet:set1];

            NSMutableArray *dataSets = [[NSMutableArray alloc] init];

            [dataSets addObject:set1];

            //将set保存到data当中

            LineChartData *data = [[LineChartData alloc] initWithDataSets:dataSets];

             _chartView.data = data;

        }

    以上就是填充数据的基本方法。到这里,大家就已经掌握了用charts绘制图表的基本方法了。下面我们来看一下一些比较难的用法。

    重点难点解析

    需求一:两个表格联动,即拖拽或者一个,另一个需要跟着动。

    这个需求可以用协议解决:

    - (void)chartScaled:(ChartViewBase *)chartView scaleX:(CGFloat)scaleX scaleY:(CGFloat)scaleY {

        CGAffineTransform srcMatrix = chartView.viewPortHandler.touchMatrix;

        [self.lineChartView.viewPortHandler refreshWithNewMatrix:srcMatrix

                                                              chart:self.combinedChartView

                                                          invalidate:YES];

        [self.barChartView.viewPortHandler refreshWithNewMatrix:srcMatrix

                                                          chart:self.barChartView

                                                    invalidate:YES];

    }

    - (void)chartTranslated:(ChartViewBase *)chartView dX:(CGFloat)dX dY:(CGFloat)dY {

        CGAffineTransform srcMatrix = chartView.viewPortHandler.touchMatrix;

        [self.lineChartView.viewPortHandler refreshWithNewMatrix:srcMatrix

                                                              chart:self.combinedChartView

                                                          invalidate:YES];

        [self.barChartView.viewPortHandler refreshWithNewMatrix:srcMatrix

                                                          chart:self.barChartView

                                                    invalidate:YES];

    }

    需求二:在一个图表上绘制多种类型的线表,例如K线图+柱状图

    这个需求会用到另一个ChartView类型:CombinedChartView

    CombinedChartData *combinedData = [[CombinedChartData alloc] init];

        combinedData.lineData = [self generateLineData];

        combinedData.candleData = [self generateCandleData];

    话不多说,待会我会贴出Demo

    需求三:希望在X轴上显示出具体的数值

    我刚才说过,绘制表格的时候X值是i的值,即从0到i,那么我们如何显示服务器给我们的X值呢?这里需要引入一个协议:IChartAxisValueFormatter,声明一个NSObject类,如BTCDepthXAxisFormatter遵循IChartAxisValueFormatter协议,重写-(NSString *)stringForValue:(double)value axis:(ChartAxisBase *)axis方法,然后进行赋值,xAxis.valueFormatter = [[BTCDepthXAxisFormatter alloc] init]即可。

    需求四:在K线图上加上最高值和最低值的箭头指示标志,效果如下图:

    重写第三方库中的CandleStickChartRenderer类,这个类是专门负责绘制和渲染K线图数据的。 @objc open func drawDataSet(context: CGContext, dataSet: ICandleChartDataSet)方法会遍历并绘制所有需要展示的数值点,因此,可以在这个方法中取出最大值和最小值的点坐标,代码如下: 

    //写入最小的Y值的位置和数值

                ifminValue > low {

                    minValue = low

                    minPositionX = xPos

                }

                //写入最大的Y值的位置和数值

                ifmaxValue < high {

                    maxValue = high

                    maxPositionX = xPos

                }

     绘制箭头的具体代码: 

    // 可见区域最左界的箭头数据

            guardletlowestVisbleEntry = dataSet.entryForIndex(_xBounds.min)as?CandleChartDataEntryelse{

                return

            }

            varlowestVisblePoint:CGPoint= CGPoint.init(x: lowestVisbleEntry.x, y: lowestVisbleEntry.low)// 此处主要是为了获取X坐标,lowestVisbleEntry.low可为high、open、close

            trans.pointValueToPixel(&lowestVisblePoint)

            // 可见区域最右界的箭头数据

            guardlethighestVisbleEntry = dataSet.entryForIndex( _xBounds.range + _xBounds.min)as?CandleChartDataEntryelse{

                return

            }

            varhighestVisblePoint:CGPoint= CGPoint.init(x: highestVisbleEntry.x, y: highestVisbleEntry.high)

            trans.pointValueToPixel(&highestVisblePoint)

            // 可见区域中的最小值

            letminValueStr = String.init(format:"%.4f", minValue)

            varminPoint:CGPoint= CGPoint.init(x: CGFloat(minPositionX), y: CGFloat(minValue * animator.phaseY))

            // 点转化为像素

            trans.pointValueToPixel(&minPoint)

            calculateTextPosition(minValueStr, originPoint: &minPoint, lowestVisibleX: lowestVisblePoint.x, highestVisibleX: highestVisblePoint.x, isMaxValue:false)

            // 可见区域中的最大值

            letmaxValueStr = String.init(format:"%.4f", maxValue)

            varmaxPoint:CGPoint= CGPoint.init(x: CGFloat(maxPositionX), y: CGFloat(maxValue * animator.phaseY))

            trans.pointValueToPixel(&maxPoint)

            calculateTextPosition(maxValueStr, originPoint: &maxPoint, lowestVisibleX: lowestVisblePoint.x, highestVisibleX: highestVisblePoint.x, isMaxValue:true)

     特别感谢@Leo就是我提供的解决方案,完美的解决了箭头绘制的问题。感兴趣的同学们可以下载CandleStickChartRenderer.swift源码去替换Charts库中的CandleStickChartRenderer文件来实现显示可见区域内最大值和最小值箭头的功能。

    至此,iOS-Charts的基本用法就介绍完了,如果还有不理解的问题,可以加我的微信(justlikeitRobert),或者点击我在github上的Demo进行研究,欢迎Star,作为对我码字的鼓励与奖赏。😄

    相关文章

      网友评论

      • 风灬起时:折线图拐点可以根据数值大小设置不同颜色吗
      • 18f2f27d464d:选中标记文字能设定吗,现在只显示当前y轴数据?希望能够自定义显示格式
      • 95c9800fdf47:这个滑动到x轴最左端的时候,折线超过x轴显示了,怎么解决
      • 凌云幽雨:有尝试过混合图中基于基准线的开发吗
      • 酸三角:哥们 针对需求三 X轴的显示 不能从服务器上直接显示出数据啊
      • 秋S寂S:我想问下,chart里面,数值是0.1的,显示成了.1,这个怎么解决呀
        秋S寂S:@割破了 解决了,通过一个NSNumberformate的东西
        3b73ac1e6515:可以自定义数据格式化
        64debba36f46:解决没
      • Be_ingenious:为什么 drawGridBackgroundEnabled 这个属性谁知为false 还是不管用呢
      • elex的开发之路:请问下 我现在做的饼形图数据会出现重叠的情况怎么处理啊?
      • 叁號選手:请教个问题。怎么设置那个值是整数 没有小数点呢?
        foolish_hungry:我也遇到这个问题, 你弄好了吗?
      • theAnswer_:不知道楼主需求4有没有思路呢现在:smile:
      • 夜若寒灬:你好,看了这篇文章学习了,目前有个需求,就是有20个数据也就是柱状图,不能一下子显示出来,显示10个,另外10条滑动显示出来,请问这个要怎么做,刚学习charts的小白一枚:pray:
        飞翔de小苹果:@夜若寒灬 好的,谢谢
        夜若寒灬:@飞翔de小苹果 [self.barChartView setVisibleXRangeMaximum:10.f]
        后面的数目可以根据自己的需要调整,这句代码的意思就是视图上显示的最大柱状图数目
        飞翔de小苹果:问题解决了?最近也要用到这个呢

      本文标题:iOS-Charts看这个就够了

      本文链接:https://www.haomeiwen.com/subject/jdrazxtx.html