iOS开发之高度自定义一个直方图

前言:在日常开发过程中,难免会碰到一些图表之类的需求,网上有很多优秀的图表库,使用起来也挺方便的,但是使用第三方库,难免会碰到UI难以满足需求,还会有些代码兼容性问题,因此本文记录了一个高度自定义直方图的开发之路,实现UI完全自定义,数据随意刷新,不用再受第三方库的约束(不用再跟UI干架啦)。

效果图:

histogam.gif

思路:
1.使用一个横向的UICollectionView,用SectionHeader实现直方图的纵坐标(也就是y轴),纵坐标根据需求选择n等分,取所有数据的最大值然后平均分;
2.自定义一个UICollectionViewCell,每一个cell就是一个单一的直方图,通过纵坐标的最值和每一组数据的比例计算直方图的高度;
3.添加一个点击显示数据的view,通过Masonry添加显示label的约束;
4.每一个cell上添加两个button(单一的直方图就添加一个),通过设置button的frame(横向宽度固定,高度代表数值大小),配合动态数据,实现每个直方图的高度;

PS:由于此直方图原先是放在tableView的cell上,
所以这里依旧是放在一个tableView上进行展示的。

实现步骤:
1.在tableView的cell上添加一个UICollectionView,并按UI需求配置好collectionView:

//初始化UICollectionView,并设置好cell的大小,已经collectionView的sectionHeader
    UICollectionViewFlowLayout * layout = [[UICollectionViewFlowLayout alloc]init];
    [layout setScrollDirection:UICollectionViewScrollDirectionHorizontal];
    layout.minimumLineSpacing = 0;
    layout.minimumInteritemSpacing = 0;
    layout.itemSize = CGSizeMake(55, 250);
    layout.sectionHeadersPinToVisibleBounds = YES;
    
    layout.headerReferenceSize = CGSizeMake(40, 250);
    layout.footerReferenceSize = CGSizeMake(40, 250);
    hxCollectionView = [[UICollectionView alloc]initWithFrame:CGRectMake(0, 0, 100, 100) collectionViewLayout:layout];
    hxCollectionView.backgroundColor = [UIColor whiteColor];
    hxCollectionView.showsHorizontalScrollIndicator = NO;
    hxCollectionView.delegate = self;
    hxCollectionView.dataSource = self;
    [hxCollectionView registerNib:[UINib nibWithNibName:@"FHXCollectionReusableView" bundle:nil] forSupplementaryViewOfKind:UICollectionElementKindSectionHeader withReuseIdentifier:@"FHXCollectionReusableView"];
    [hxCollectionView registerNib:[UINib nibWithNibName:@"HXCollectionReusableView" bundle:nil] forSupplementaryViewOfKind:UICollectionElementKindSectionFooter withReuseIdentifier:@"HXCollectionReusableView"];
    hxCollectionView.bounces = NO;
    //hxCollectionView.contentOffset = CGPointMake(SCREEN_WIDTH/2.0f, 0);
    [self.bgView addSubview:hxCollectionView];
    
    [hxCollectionView mas_makeConstraints:^(MASConstraintMaker *make) {
       
        make.left.equalTo(self.bgView.mas_left).with.offset(0.0);
        make.right.equalTo(self.bgView.mas_right).with.offset(0.0);
        make.top.equalTo(self.bgView.mas_top).with.offset(50.0f);
        make.height.equalTo(@250.0f);
    }];

2.实现collectionView的sectionHeader,这里选择在sectionHeader上放8个label,将纵坐标(y轴)的高度固定,然后根据服务端返回数据的最值,决定(纵坐标)的间隔;


sectionHeader(y轴)初始样式

通过图表数据的最值,动态实现纵坐标赋值:

-(void)setMaxData:(NSInteger)maxData{
    _maxData = maxData;
    currentMax = _maxData;
    //纵坐标间隔
    NSInteger interDiscount = currentMax/7.0f;
    for (int i = 0; i<labelArray.count; i++) {
        UILabel * label = labelArray[I];
        label.text = [NSString stringWithFormat:@"%ld",i * interDiscount];
    }
}

这样实现,每次y周的刻度值都是不一样的,是根据图表数据的最值计算并动态赋值的。

- (UICollectionReusableView *)collectionView:(UICollectionView *)collectionView viewForSupplementaryElementOfKind:(NSString *)kind atIndexPath:(NSIndexPath *)indexPath {
    if ([kind isEqualToString:UICollectionElementKindSectionHeader]) {

        FHXCollectionReusableView *  view = [hxCollectionView dequeueReusableSupplementaryViewOfKind:UICollectionElementKindSectionHeader withReuseIdentifier:@"FHXCollectionReusableView" forIndexPath:indexPath];
        view.backgroundColor = [UIColor whiteColor];
        //给y轴的最大值赋值
        if (resultArray.count > 0 && maxValue - 1 > 0) {
            
            view.maxData = maxValue;
        }
        return view;
    }else if ([kind isEqualToString:UICollectionElementKindSectionFooter]) {
        UICollectionReusableView* view = [hxCollectionView dequeueReusableSupplementaryViewOfKind:UICollectionElementKindSectionFooter withReuseIdentifier:@"HXCollectionReusableView" forIndexPath:indexPath];
        return view;
    }else{
        return nil;
    }
}

这里同时设置一个SectionFooter,是为了更好的调整UI,可以根据实际情况调整。
3.自定义一个UICollectionViewCell,首先添加8条横线(因为这里的纵坐标的8等分)


单个直方图的背景

然后添加显示直方图高度的两个button(如果是单一的直方图,一个button就可以)和一个点击显示数据的自定义view,后面会通过每一组数据的大小来计算messageView(显示文案的view)显示的位置,因为直方图有高有低,所以需要动态计算每个messageView的frame

//老用户
    oldUserBtn = [UIButton buttonWithType:UIButtonTypeCustom];
    [oldUserBtn setBackgroundColor:HXRGB(65, 109, 251)];
    oldUserBtn.userInteractionEnabled = NO;
    [self.contentView addSubview:oldUserBtn];
    [oldUserBtn addTarget:self action:@selector(clickOldBtnAction:) forControlEvents:UIControlEventTouchUpInside];

    //新用户
    newUserBtn = [UIButton buttonWithType:UIButtonTypeCustom];
    newUserBtn.backgroundColor = HXRGB(255, 206, 102);
    newUserBtn.userInteractionEnabled = NO;
    [self.contentView addSubview:newUserBtn];
    [newUserBtn addTarget:self action:@selector(clickNewBtnAction:) forControlEvents:UIControlEventTouchUpInside];

//点击展示数据,默认隐藏
    self.messageView =  [[NSBundle mainBundle]loadNibNamed:@"FHXSmallMessageView" owner:self options:nil][0];
    self.messageView.backgroundColor = [UIColor clearColor];
    self.messageView.frame = CGRectMake(0, 0, 100, 60);
    self.messageView.hidden = YES;
    [self.contentView addSubview:self.messageView];

4.每个直方图是通过UIButton的纵向高度展示出来,根据每一组数据来计算每个button的高度,后面添加数据会展示,到这里,一个直方图的背景图算是基本完成了。

5,准备一组数据,这里是随机生成的一个数据,实际应用中数据将有服务端下发,数据结果如下:

#import <Foundation/Foundation.h>

NS_ASSUME_NONNULL_BEGIN

@interface FHXTrendModel : NSObject
@property(nonatomic,strong)NSString * x;//横坐标数值(x轴)
@property(nonatomic,strong)NSString * y0;//纵坐标数值(y轴)
@property(nonatomic,strong)NSString * y1;//纵坐标数值(y轴)
@property(nonatomic,strong)NSString * y2;//纵坐标数值(y轴)
@end

NS_ASSUME_NONNULL_END

因为本实例中是要展示同一日期的两组数据,所以用到的是y0和y1.

准备数据:用一个数组存放数据,方便后续赋值

#pragma mark -- 创建数据
-(void)creatData{
    
    //模拟20条数据
    for (int i = 0; i < 20; i++) {
        
        FHXTrendModel * model = [[FHXTrendModel alloc]init];
        if (i < 9) {
            model.x = [NSString stringWithFormat:@"2020010%d",i + 1];
        }else{
            model.x = [NSString stringWithFormat:@"202001%d",i + 1];
        }
        
        model.y0 = [NSString stringWithFormat:@"%d",arc4random()%200];
        model.y1 = [NSString stringWithFormat:@"%d",arc4random()%100];
        
        [self.orderArray addObject:model];
    }
    [self.tableView reloadData];
}

6.将数据添加到图表,由于这里是用的是一个tableView来展示数据,所以这个直方图(UICollectionView)放在了tableView的cell上

#pragma mark -- UITableViewDelegate,UITableViewDataSource
-(UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath{
    
        FHXOrderChartCell * cell = [tableView dequeueReusableCellWithIdentifier:@"FHXOrderChartCell"];
        if (cell == nil)
        {
            cell = [[[NSBundle mainBundle]loadNibNamed:@"FHXOrderChartCell" owner:self options:nil] lastObject];
        }
        cell.delegate = self;
        cell.columnarDataArray = self.orderArray;
        cell.unitLabel.text = @"单";
        [cell.titleButton setTitle:@"订单数" forState:UIControlStateNormal];
        cell.selectionStyle = UITableViewCellSelectionStyleNone;
        return cell;
}

7.提取数据,并做一些换算,需要将数据的x轴数据,y轴数据,y0+y1的最值全部找出来

    NSMutableArray * arrayX;//横坐标
    NSMutableArray * arrayY0;//纵坐标
    NSMutableArray * arrayY1;//纵坐标
    NSMutableArray * resultArray;//y0+y1
    NSInteger maxValue;//y0+y1最大值
    NSIndexPath * selIndex;//记录当前选中cell
-(void)setColumnarDataArray:(NSMutableArray *)columnarDataArray{
    
    //暂无数据处理
    if (columnarDataArray.count == 0) {
        self.noDataView.hidden = NO;
        return;
    }else{
        self.noDataView.hidden = YES;
    }
    
    //分离数据
    _columnarDataArray = columnarDataArray;
    for (FHXTrendModel * model in _columnarDataArray) {
        
        [arrayX addObject:model.x];
        [arrayY0 addObject:model.y0];
        [arrayY1 addObject:model.y1];
        CGFloat result = ([model.y0 floatValue] + [model.y1 floatValue]);
        [resultArray addObject:[NSString stringWithFormat:@"%.2f",result]];
    }
    //取出y0+y1的最大值
    CGFloat maxMun = [[resultArray valueForKeyPath:@"@max.floatValue"] floatValue];
    if (maxMun == 0) {
        self.noDataView.hidden = NO;
        return;
    }else{
        self.noDataView.hidden = YES;
    }
    maxValue = (NSInteger)(maxMun) + 1;
    //对7取余数
    int remainder = maxValue%7;
    //确保maxValue能被7整除
    maxValue = maxValue + (7 - remainder);
    [hxCollectionView reloadData];
}

8.将处理好的数据赋给collectionView,然后在collectionView的计算直方图的高度,以及点击显示view的frame

-(UICollectionViewCell *)collectionView:(UICollectionView *)collectionView cellForItemAtIndexPath:(NSIndexPath *)indexPath{
    
    UINib *nib = [UINib nibWithNibName:@"FHXOrderCollectionCell" bundle: [NSBundle mainBundle]];
    [collectionView registerNib:nib forCellWithReuseIdentifier:@"FHXOrderCollectionCell"];
    FHXOrderCollectionCell *cell = [collectionView dequeueReusableCellWithReuseIdentifier:@"FHXOrderCollectionCell" forIndexPath:indexPath];
    if (arrayX.count > 0 && maxValue - 1 > 0) {
      
        cell.dateLabel.text = arrayX[indexPath.row];
    }
    if (_columnarDataArray.count > 0 && maxValue - 1 > 0) {
        
        cell.maxValue = maxValue;
        cell.trendModel = _columnarDataArray[indexPath.row];
    }
    if (selIndex == indexPath) {
        cell.messageView.hidden = NO;
    }else{
        cell.messageView.hidden = YES;
    }
    cell.backgroundColor = [UIColor whiteColor];
        return cell;
}

计算直方图的位置和高度(其实是通过UIButton实现的):

-(void)setMaxValue:(NSInteger)maxValue{
    
    _maxValue = maxValue;
}

-(void)setTrendModel:(FHXTrendModel *)trendModel{
    
    _trendModel = trendModel;
    //计算老用户占比
    CGFloat originY0 = [_trendModel.y0 floatValue];
    CGFloat maxMun = (CGFloat)(_maxValue);
    CGFloat positionY0 = kheight*(1 - originY0/maxMun);
    oldUserBtn.frame = CGRectMake((55-kwidth)/2.0, positionY0 + 6, kwidth, kheight*originY0/maxMun);
    
    //计算新用户占比
    CGFloat originY1 = [_trendModel.y1 floatValue];
    CGFloat positionY1 = positionY0 - (originY1/maxMun)*kheight;
    newUserBtn.frame = CGRectMake((55-kwidth)/2.0, positionY1 + 6, kwidth, kheight*originY1/maxMun);
    
    //button上半部分圆角
    UIBezierPath * maskPath = [UIBezierPath bezierPathWithRoundedRect:newUserBtn.bounds byRoundingCorners:UIRectCornerTopRight | UIRectCornerTopLeft cornerRadii:CGSizeMake(5, 5)];
    CAShapeLayer *maskLayer = [[CAShapeLayer alloc] init];
    maskLayer.frame = newUserBtn.bounds;
    maskLayer.path = maskPath.CGPath;
    newUserBtn.layer.mask = maskLayer;
    
    //显示view的位置
    self.messageView.firstLabel.text = [Helper notRounding:_trendModel.y1 afterPoint:2];
    self.messageView.secondLabel.text = [Helper notRounding:_trendModel.y0 afterPoint:2];
    
    //根据y0+y1的值判断显示view位置
    CGFloat currentY = _trendModel.y0.floatValue + _trendModel.y1.floatValue;
    if (currentY > _maxValue*(5/7.0)) {
        self.messageView.type = 0;
        [self.messageView mas_remakeConstraints:^(MASConstraintMaker *make) {
            make.bottom.equalTo(newUserBtn.mas_top).offset(65);
            make.centerX.equalTo(newUserBtn);
            make.width.equalTo(@60.0f);
            make.height.equalTo(@60.0f);
        }];
    }else{
        self.messageView.type = 1;
        [self.messageView mas_remakeConstraints:^(MASConstraintMaker *make) {
            make.bottom.equalTo(newUserBtn.mas_top).offset(-5);
            make.centerX.equalTo(newUserBtn);
            make.width.equalTo(@60.0f);
            make.height.equalTo(@60.0f);
        }];
    }
    
    self.firstLineView.backgroundColor = HXRGB(226, 235, 242);
    self.secondLineView.backgroundColor = HXRGB(226, 235, 242);
    self.thirdLineView.backgroundColor = HXRGB(226, 235, 242);
    self.fourthLineView.backgroundColor = HXRGB(226, 235, 242);
    self.fifthLineView.backgroundColor = HXRGB(226, 235, 242);
    self.sixLineView.backgroundColor = HXRGB(226, 235, 242);
    self.sevenLineView.backgroundColor = HXRGB(226, 235, 242);
    self.eightLineView.backgroundColor = HXRGB(226, 235, 242);
}

9.到此为止,完成这个直方图的主要工作基本完成了,具体实现请参考Demo,大部分这种图包括折线图,曲线图都是为了看数据变化趋势,和做数据对比,所以,做出来的图表能达到100%的UI还原,还是比较舒心的。

PS: 之前在项目开发过程还涉及到数据类型的筛选,这里省略掉了,实现筛选其实就是数据重载,这里因为用的的UICollectionView实现,所以数据可以随意刷新,不存在卡顿,或者是线程阻塞等问题。

END:具体实现详见面Demo

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

推荐阅读更多精彩内容

  • 直方图主要用在数据图表,作为对比数据,用柱体高度的高低,形象直观地表达出来,往往与折线图配合使用,而折线图便于从众...
    理想是试阅读 945评论 0 0
  • 一个潜水新手初次下海的真实心路历程。 文/小小蒋 2017年10月23日,在垦丁后壁湖,下潜10米。这是我第一次潜...
    夜昕阅读 574评论 0 0
  • 昨日中午爆出白百何疑似出轨的猛料,马上就横扫网络,刷爆朋友圈!? 不知大家的朋友圈有没有被刷爆呢?反正我的朋友圈有...
    聆枫物语阅读 209评论 5 9
  • 文/子亿 在家里呆了八天了。我是门都没出过一步。这几天里,老公负责烧饭,女儿负责做作业,我负责督促和辅导女儿的作业...
    快乐的子亿阅读 969评论 8 55