• 首页
  • 关于
    • zerone-z photo

      zerone-z

      记录下自己学习的过程、点滴,也为了分享自己的收获。

    • Learn More
    • Email
    • Github
  • 博客
    • 博客
    • 标签
  • 随笔
  • 项目

CorePlot_1.5.1 绘制柱状图

07 Dec 2014

Reading time ~4 minutes

前言

在上一篇中讲述了绘制折线图的方法,也较详细的讲述了宿主View、图表CPTXYGraph、绘图空间、坐标系、以及图例的创建过程。这一篇将主要简述柱状图(CPTBarPlot)的创建,以及一些不同点,其它的和上一篇相同。效果图如下: 柱状图

柱状图(CPTBarPlot)的创建

CPTBarPlot用于创建柱状图,继承于CPTPlot。

#pragma mark 创建平面图,柱状图
- (void)createPlots
{
    // 动画
    CABasicAnimation *fadeInAnimation = [CABasicAnimation animationWithKeyPath:@"opacity"];
    fadeInAnimation.duration            = 3.0f;
    fadeInAnimation.removedOnCompletion = NO;
    fadeInAnimation.fillMode            = kCAFillModeForwards;
    fadeInAnimation.toValue             = [NSNumber numberWithFloat:1.0];

    // 第一个柱状图
    {
        // 第一个参数指定渐变色的开始颜色,默认结束颜色为黑色,第二个参数指定是否绘制水平柱子。
        CPTBarPlot *barPlot = [CPTBarPlot tubularBarPlotWithColor:[CPTColor greenColor] horizontalBars:NO];

        // 添加图形到绘图空间
        [_hostView.hostedGraph addPlot:barPlot];

        // 设置数据源 实现CPTBarPlotDataSource委托
        barPlot.dataSource = self;

        // 委托事件
        barPlot.delegate = self;

        // 标识,根据此@ref identifier来区分不同的plot,也是图例显示名称,
        barPlot.identifier = @"BarPlot1" ;

        // 基线值设置
        {
            // NO:@ref baseValue的设置对所有的柱子生效,YES:需要通过数据源设置每一个柱子的@ref baseValue  默认值:NO
            barPlot.barBasesVary = YES;
            // 柱子的基线值 @ref barBasesVary为NO时才会生效,否则需要在数据源中设置枚举为CPTBarPlotFieldBarBase的一个适当的值
            // 柱子都是从此基线值处开始绘制,相当于原点
            barPlot.baseValue = CPTDecimalFromFloat(1);
        }

        // 柱子设置,柱子的实际宽度为@ref barWidth * barWidthScale
        {
            // 宽度计算方式 NO:1主刻度长度=1宽度  YES:1像素=1宽度  默认值:NO
            barPlot.barWidthsAreInViewCoordinates = YES;
            // 宽度
            barPlot.barWidth = CPTDecimalFromCGFloat(20);
            // 柱宽的缩放系数
            barPlot.barWidthScale = CPTFloat(1);
            // 开始绘制的偏移位置,默认为0,表示柱子的中间位置在刻度线上
            barPlot.barOffset = CPTDecimalFromCGFloat(-10) ;
            // 尖端的圆角值 用的是像素单位
            barPlot.barCornerRadius = CPTFloat(0);
            // 底部的圆角值,基线值的圆角 用的是像素单位
            barPlot.barBaseCornerRadius = CPTFloat(0);
            // 外框的线型 默认:黑色 宽度1
            barPlot.lineStyle = nil;
            // 填充色
            CPTGradient *gradient = [CPTGradient gradientWithBeginningColor:[CPTColor greenColor] endingColor:[CPTColor clearColor]];
            CPTFill *fill = [CPTFill fillWithGradient:gradient];
            barPlot.fill = fill;
        }

        // 数据标签设置,如果想用自定义的标签,则需要数据源方法:dataLabelForPlot:recordIndex:
        {
            // 偏移量设置
            barPlot.labelOffset = 15;
            // 数据标签样式
            CPTMutableTextStyle *labelTextStyle = [[CPTMutableTextStyle alloc] init];
            labelTextStyle.color = [CPTColor magentaColor];
            barPlot.labelTextStyle = labelTextStyle;
        }

        // 添加动画
        barPlot.opacity = 0.f;
        [barPlot addAnimation:fadeInAnimation forKey:@"animateOpacity"];

    }

    // 第2个柱状图
    {
        // 第一个参数指定渐变色的开始颜色,默认结束颜色为黑色,第二个参数指定是否绘制水平柱子。
        CPTBarPlot *barPlot = [CPTBarPlot tubularBarPlotWithColor:[CPTColor blueColor] horizontalBars:NO];

        // 添加图形到绘图空间
        [_hostView.hostedGraph addPlot:barPlot];

        // 设置数据源 实现CPTBarPlotDataSource委托
        barPlot.dataSource = self;

        // 标识,根据此@ref identifier来区分不同的plot,也是图例显示名称,
        barPlot.identifier = @"BarPlot2" ;

        // 基线值设置
        {
            // NO:@ref baseValue的设置对所有的柱子生效,YES:需要通过数据源设置每一个柱子的@ref baseValue  默认值:NO
            barPlot.barBasesVary = NO;
            // 柱子的基线值 @ref barBasesVary为NO时才会生效,否则需要在数据源中设置枚举为CPTBarPlotFieldBarBase的一个适当的值
            // 大于这个值以上的点,柱子只从这个点开始画。小于此值的点,则是反向绘制的,即从基线值向下画,一直画到到数据点。
            barPlot.baseValue = CPTDecimalFromFloat(0);
        }

        // 柱子设置
        {
            // 宽度计算方式 NO:1主刻度长度=1宽度  YES:1像素=1宽度  默认值:NO
            barPlot.barWidthsAreInViewCoordinates = NO;
            // 宽度
            barPlot.barWidth = CPTDecimalFromCGFloat(0.4);
            // 柱宽的缩放系数
            barPlot.barWidthScale = CPTFloat(1);
            // 开始绘制的偏移位置
            barPlot.barOffset = CPTDecimalFromCGFloat(0.2) ;
            // 尖端的圆角值 用的是像素单位
            barPlot.barCornerRadius = CPTFloat(0);
            // 底部的圆角值,基线值的圆角 用的是像素单位
            barPlot.barBaseCornerRadius = CPTFloat(0);
            // 外框的线型 默认:黑色 宽度1
            barPlot.lineStyle = nil;
            // 填充色
            CPTGradient *gradient = [CPTGradient gradientWithBeginningColor:[CPTColor blueColor] endingColor:[CPTColor clearColor]];
            CPTFill *fill = [CPTFill fillWithGradient:gradient];
            barPlot.fill = fill;
        }

        // 数据标签设置,如果想用自定义的标签,则需要数据源方法:dataLabelForPlot:recordIndex:
        {
            // 偏移量设置
            barPlot.labelOffset = 15;
            // 数据标签样式
            CPTMutableTextStyle *labelTextStyle = [[CPTMutableTextStyle alloc] init];
            labelTextStyle.color = [CPTColor magentaColor];
            barPlot.labelTextStyle = labelTextStyle;
        }

        // 添加动画
        barPlot.opacity = 0.f;
        [barPlot addAnimation:fadeInAnimation forKey:@"animateOpacity"];
    }
}

CPTBarPlot的数据源方法CPTBarPlotDataSource

#pragma mark 询问有多少个数据
- (NSUInteger)numberOfRecordsForPlot:(CPTPlot *)plot
{
    NSUInteger count = 0;
    if ([plot.identifier isEqual:@"BarPlot1"]) {
        count = self.dataSource1.count;
    }else {
        count = self.dataSource2.count;
    }

    return count;
}

#pragma mark 询问一个个数据值 fieldEnum:一个轴类型,是一个枚举  idx:坐标轴索引
- (NSNumber *)numberForPlot:(CPTPlot *)plot field:(NSUInteger)fieldEnum recordIndex:(NSUInteger)idx
{
    NSNumber *num = nil;

    if ([plot.identifier isEqual:@"BarPlot1"]) {
        switch (fieldEnum) {
            case CPTBarPlotFieldBarLocation:    // 柱子所处位置 如果是垂直柱子,即为x轴的位置
            {
                num = @(idx + 1);
            }
                break;
            case CPTBarPlotFieldBarTip:         // 柱子尖端位置(柱子的长度) 如果是垂直柱子,即为y轴的位置
            {
                num = self.dataSource1[idx];
            }
                break;
            case CPTBarPlotFieldBarBase:        // 柱子的基线值 只有@ref barBasesVary = YES 时才会用到该枚举
            {
                num = @(0);
            }
                break;
            default:
                break;
        }
    }else if ([plot.identifier isEqual:@"BarPlot2"]){
        switch (fieldEnum) {
            case CPTBarPlotFieldBarLocation:    // 柱子所处位置 如果是垂直柱子,即为x轴的位置
            {
                num = @(idx + 1);
            }
                break;
            case CPTBarPlotFieldBarTip:         // 柱子末端位置(柱子的长度) 如果是垂直柱子,即为y轴的位置
            {
                num = self.dataSource2[idx];
            }
                break;
            case CPTBarPlotFieldBarBase:        // 柱子的基线值 只有@ref barBasesVary = YES 时才会用到该枚举
            {

            }
                break;
            default:
                break;
        }
    }
    return num;
}

#pragma mark 添加数据标签
- (CPTLayer *)dataLabelForPlot:(CPTPlot *)plot recordIndex:(NSUInteger)idx
{
    if ([plot.identifier isEqual:@"BarPlot1"]) {
        // 数据标签样式
        CPTMutableTextStyle *labelTextStyle = [[CPTMutableTextStyle alloc] init];
        labelTextStyle.color = [CPTColor magentaColor];

        // 定义一个 TextLayer
        CPTTextLayer *newLayer = [[CPTTextLayer alloc] initWithText:[NSString stringWithFormat:@"%d",(int)[self.dataSource1[idx] integerValue]] style:labelTextStyle];

        return newLayer;
    }else {
        // 数据标签样式
        CPTMutableTextStyle *labelTextStyle = [[CPTMutableTextStyle alloc] init];
        labelTextStyle.color = [CPTColor magentaColor];

        // 定义一个 TextLayer
        CPTTextLayer *newLayer = [[CPTTextLayer alloc] initWithText:[NSString stringWithFormat:@"%d",(int)[self.dataSource2[idx] integerValue]] style:labelTextStyle];

        return newLayer;
    }
}

#pragma mark 设置图例名称 返回每一个柱子的图例名称 返回nil则不显示该索引下的图例
- (NSString *)legendTitleForBarPlot:(CPTBarPlot *)barPlot recordIndex:(NSUInteger)idx
{
    NSString *legendTitle = nil;
    if ([barPlot.identifier isEqual:@"BarPlot1"]) {
        legendTitle = [NSString stringWithFormat:@"柱状图1-%d",idx];
    }else {
        legendTitle = [NSString stringWithFormat:@"柱状图2-%d",idx];
    }
    return legendTitle;
}

如果想使用数据源方法设置图例方法,必须要用如下方法创建图例,这会使每一个柱子都在图例中。

CPTLegend *legend = [CPTLegend legendWithPlots:_hostView.hostedGraph.allPlots];
CPTLegend *legend = [CPTLegend legendWithGraph:_hostView.hostedGraph]

如果不需要每个柱子都显示图例(不会调用数据legendTitleForBarPlot),只需要设置不同类别的图例,可以使用如下方法。

// 图例样式设置
NSMutableArray *plots = [NSMutableArray array];
for (int i = 0; i < _hostView.hostedGraph.allPlots.count; i++) {
    CPTBarPlot *barPlot = _hostView.hostedGraph.allPlots[i];

    CPTBarPlot *plot = [[CPTBarPlot alloc] init];
    plot.fill = barPlot.fill;
    plot.lineStyle = barPlot.lineStyle;
    plot.identifier = [NSString stringWithFormat:@"柱状图%d", (i + 1)];
    [plots addObject:plot];
}
// 图例初始化 只有把plots 替换为 _hostView.hostedGraph.allPlots 数据源方法的设置图例名称才会生效
CPTLegend *legend = [CPTLegend legendWithPlots:plots];

两种创建图例的方法效果如下:

图例一 图例二

CPTBarPlot的delegate方法

#pragma mark 选中某个柱子的操作 添加注释
- (void)barPlot:(CPTBarPlot *)plot barWasSelectedAtRecordIndex:(NSUInteger)idx withEvent:(UIEvent *)event
{
    // 移除注释
    CPTPlotArea *plotArea = _hostView.hostedGraph.plotAreaFrame.plotArea;
    [plotArea removeAllAnnotations];

    // 创建注释,plotSpace:绘图空间 anchorPlotPoint:坐标点
    CPTPlotSpaceAnnotation *barTextAnnotation = [[CPTPlotSpaceAnnotation alloc] initWithPlotSpace:_hostView.hostedGraph.defaultPlotSpace anchorPlotPoint:@[@(idx + 1),self.dataSource1[idx]]];

    // 文本样式
    CPTMutableTextStyle *annotationTextStyle = [CPTMutableTextStyle textStyle];
    annotationTextStyle.color = [CPTColor redColor];
    annotationTextStyle.fontSize = 17.0f;
    annotationTextStyle.fontName = @"Helvetica-Bold";
    // 显示的字符串
    NSString *randomValue = [NSString stringWithFormat:@"柱状图\n随即值:%@ \n", [self.dataSource1[idx] stringValue]];
    // 注释内容
    CPTTextLayer *textLayer = [[CPTTextLayer alloc] initWithText:randomValue style:annotationTextStyle];
    // 添加注释内容
    barTextAnnotation.contentLayer = textLayer;

    // 注释位置
    barTextAnnotation.displacement = CGPointMake(CPTFloat(0), CPTFloat(20));

    // 把拐点注释添加到绘图区域中
    [plotArea addAnnotation:barTextAnnotation];
}


iOS移动CorePlot图表柱状图 Like Tweet +1