iOS-Charts看这个就够了

        最近做项目需要画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,作为对我码字的鼓励与奖赏。😄

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,658评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,482评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,213评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,395评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,487评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,523评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,525评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,300评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,753评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,048评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,223评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,905评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,541评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,168评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,417评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,094评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,088评论 2 352

推荐阅读更多精彩内容