iOS绘制收益柱状图
项目需求,参考了其他绘图demo,自己绘制出来了,不过代码改得有点乱,添加了很多变量,时间关系没用太合适的命名,逻辑处理也没进行优化。
看看效果图(虚线区域都是画的,其他区域添加的都是控件),附上源码




#import <UIKit/UIKit.h>
typedef enum : NSUInteger {
CSYieldTypeWeek = 0, //周收益
CSYieldTypeMonth = 1, //月收益
CSYieldTypeYear = 2, //年收益
} CSYieldType; //收益类型
typedef enum : NSUInteger {
CSTimePointJanuary = 0, //一月份
CSTimePointFebruary,
CSTimePointMarch,
CSTimePointApril,
CSTimePointMay, //.
CSTimePointJun, //.
CSTimePointJuly, //.
CSTimePointAugust,
CSTimePointSeptember,
CSTimePointOctober,
CSTimePointNovember,
CSTimePointDecember, //十二月份
CSTimePointThisWeek, //本周
CSTimePointLastWeek, //上周
} CSTimePoint;
@interface CSYieldChartView : UIView
@property (nonatomic,assign)CSYieldType yieldType; //收益类型
@property (nonatomic,assign)CSTimePoint timePoint; //时间点
@property (nonatomic,strong) NSArray *yields; //收益数组
@property (nonatomic,strong) NSArray *dayPoints; //时间点数组
//刷新图表
- (void)refreshChartWithYields:(NSArray *)yields dayPoints:(NSArray *)dayPoints yieldType:(CSYieldType)yieldType timePoint:(CSTimePoint)timePoint;
@end
#define yieldLineSpace 4
#define lineLeftMargin 40
#define lineRightMargin 10
#define lineTopMargin 10
#define lineBottomMargin 30
#define lineWidth 0.5f
#define itemLeftMargin 16
#define itemRightMargin 16
#define itemSpace 6
#define itemWidthRatio (5/12.f)
#define positiveItemHexColor 0xe83846 //正收益条颜色值
#define negativeItemHexColor 0x17c7ba //负收益条颜色值
#import "CSYieldChartView.h"
#import "UIColor+Addition.h"
@implementation CSYieldChartView
-(void)refreshChartWithYields:(NSArray *)yields dayPoints:(NSArray *)dayPoints yieldType:(CSYieldType)yieldType timePoint:(CSTimePoint)timePoint
{
self.yields = yields;
self.dayPoints = dayPoints;
self.yieldType = yieldType;
self.timePoint = timePoint;
[self setNeedsDisplay];
}
- (void)drawRect:(CGRect)rect
{
//声明最大的收益
float maxYield = 0;
//获取最大的收益绝对值
for (NSNumber *number in _yields) {
if (maxYield < fabs([number floatValue])) {
maxYield = fabs([number floatValue]);
}
}
//若最大收益少于10,则将最大值设为10
if (maxYield < 10.0) {
maxYield = 10.0;
}
//分几段
NSInteger sectionNum = 2;
//平均每段的收益
CGFloat sectionYield = maxYield / 2;
UIColor *currentColor = [UIColor lightGrayColor];
CGContextRef contextRef = UIGraphicsGetCurrentContext();
CGContextBeginPath(contextRef);
CGContextSetLineWidth(contextRef, lineWidth); //设置线粗
CGContextSetStrokeColorWithColor(contextRef, currentColor.CGColor); //设置画笔颜色
//中线的y坐标
CGFloat centerLineY = (self.frame.size.height - lineTopMargin - lineBottomMargin)/2.f + lineTopMargin;
//设置段落风格
NSMutableParagraphStyle *paragraph = [[NSMutableParagraphStyle alloc] init];
paragraph.alignment = NSTextAlignmentRight;
//设置字体颜色
UIColor *textColor = [UIColor lightGrayColor];
//设置字体大小
UIFont *textFont = [UIFont systemFontOfSize:10];
//0收益中线
CGContextMoveToPoint(contextRef, lineLeftMargin, centerLineY);
CGContextAddLineToPoint(contextRef, self.frame.size.width - lineRightMargin, centerLineY);
CGContextStrokePath(contextRef);
//0收益率
[@"0%" drawInRect:CGRectMake(0, centerLineY - 7, lineLeftMargin - yieldLineSpace, lineLeftMargin) withAttributes:@{NSParagraphStyleAttributeName:paragraph,NSForegroundColorAttributeName:textColor,NSFontAttributeName:textFont}];
//线与线的垂直距离
CGFloat lineSpace = (self.frame.size.height - lineTopMargin - lineBottomMargin) / 4.f;
for (int i = 0; i < sectionNum; i++) {
CGFloat topLineY = centerLineY - lineSpace * (i + 1);
CGFloat bottomLineY = centerLineY + lineSpace * (i + 1);
//虚线 线宽与空格宽
CGFloat dashes[] = {2,1};
CGContextSetLineDash(contextRef, 0.0, dashes, 2);
CGContextSetStrokeColorWithColor(contextRef, currentColor.CGColor); //设置画笔颜色
//中线以上
CGContextMoveToPoint(contextRef, lineLeftMargin, topLineY);
CGContextAddLineToPoint(contextRef, self.frame.size.width - lineRightMargin, topLineY);
CGContextStrokePath(contextRef);
//收益率
[[NSString stringWithFormat:@"%.f%%",sectionYield * (i + 1)] drawInRect:CGRectMake(0, topLineY - 7, lineLeftMargin - yieldLineSpace, lineLeftMargin) withAttributes:@{NSParagraphStyleAttributeName:paragraph,NSForegroundColorAttributeName:textColor,NSFontAttributeName:textFont}];
CGContextSetStrokeColorWithColor(contextRef, currentColor.CGColor); //设置画笔颜色
//中线以下
CGContextMoveToPoint(contextRef, lineLeftMargin, bottomLineY);
CGContextAddLineToPoint(contextRef, self.frame.size.width - lineRightMargin, bottomLineY);
CGContextStrokePath(contextRef);
//负收益率
[[NSString stringWithFormat:@"-%.f%%",sectionYield * (i + 1)] drawInRect:CGRectMake(0, bottomLineY - 7, lineLeftMargin - yieldLineSpace, lineLeftMargin) withAttributes:@{NSParagraphStyleAttributeName:paragraph,NSForegroundColorAttributeName:textColor,NSFontAttributeName:textFont}];
}
if (0 == _yields.count) {
paragraph.alignment = NSTextAlignmentCenter;
NSDictionary *attributes = @{NSParagraphStyleAttributeName:paragraph,NSForegroundColorAttributeName:textColor,NSFontAttributeName:[UIFont systemFontOfSize:40]};
CGSize textSize = [@"暂无数据" sizeWithAttributes:attributes];
[@"暂无数据" drawInRect:CGRectMake(lineLeftMargin + itemLeftMargin,centerLineY - textSize.height / 2.f,self.width - lineLeftMargin - itemLeftMargin - lineRightMargin - itemRightMargin,textSize.height) withAttributes:attributes];
return;
}
NSString *lastDay = [_dayPoints lastObject];
//最多有几个柱(如果是月收益,且收益数小于22,按22算,否则按收益数多少算)
NSInteger partNum = (_yields.count < 22 && _yieldType == CSYieldTypeMonth) ? 22 : _yields.count;
paragraph.alignment = NSTextAlignmentLeft;
//绘制柱状图
CGFloat averageSpace = (self.frame.size.width - lineLeftMargin - itemLeftMargin - lineRightMargin - itemRightMargin) / (partNum - 1);
//柱宽
CGFloat itemWidth = averageSpace * itemWidthRatio;
for (int i=0; i<partNum; i++) {
if (i > _yields.count - 1) //后面暂时还没收益就不画柱状条
{
break;
}
if (i % 5 == 0) {
//绘制日期
NSString *dayStr = nil;
if (self.yieldType == CSYieldTypeMonth && 0 == i) {
dayStr = [NSString stringWithFormat:@"%@(日)",_dayPoints[0]];
}
else
{
dayStr = [NSString stringWithFormat:@"%@",_dayPoints[i]];
}
[dayStr drawInRect:CGRectMake(lineLeftMargin + itemLeftMargin - 5 + averageSpace * i, self.frame.size.height - (lineBottomMargin - 6), 50, 15)withAttributes:@{NSParagraphStyleAttributeName:paragraph,NSForegroundColorAttributeName:textColor,NSFontAttributeName:textFont}];
}
else if (i == _yields.count - 1) //最后一个柱状图日期
{
[[NSString stringWithFormat:@"%@",lastDay] drawInRect:CGRectMake(lineLeftMargin + itemLeftMargin - 5 + averageSpace * i, self.frame.size.height - (lineBottomMargin - 6), 50, 15)withAttributes:@{NSParagraphStyleAttributeName:paragraph,NSForegroundColorAttributeName:textColor,NSFontAttributeName:textFont}];
}
//绘制柱状图
float yield = [_yields[i] floatValue];
float radius = itemWidth/2.f; //柱状末尾的半圆半径
//虚线 线宽与空格宽 (之前设了虚线,现在把虚线的间隙变为0)
CGFloat dashes[] = {1,0};
CGContextSetLineDash(contextRef, 0.0, dashes, 2);
CGFloat itemX = lineLeftMargin + itemLeftMargin + averageSpace * i - itemWidth/2.f;
CGFloat itemY = centerLineY - (lineSpace * sectionNum) * yield / maxYield;
//是否要画半圆(如果收益高度低于宽度,就不画半圆)
BOOL shouldDrawCircle = (fabs(itemY - centerLineY) > itemWidth/2.f) ? YES : NO;
if (yield >= 0) {
if (shouldDrawCircle) {
itemY += radius;
}
//正收益颜色
CGContextSetStrokeColorWithColor(contextRef, [UIColor colorWithHex:positiveItemHexColor].CGColor);
CGContextSetFillColorWithColor(contextRef, [UIColor colorWithHex:positiveItemHexColor].CGColor);
}
else
{
if (shouldDrawCircle) {
itemY -= radius;
}
//负收益颜色
CGContextSetStrokeColorWithColor(contextRef, [UIColor colorWithHex:negativeItemHexColor].CGColor);
CGContextSetFillColorWithColor(contextRef, [UIColor colorWithHex:negativeItemHexColor].CGColor);
}
// CGContextSetShadow(contextRef,CGSizeMake(2, 0) , 2); //阴影效果
CGMutablePathRef pathRef = CGPathCreateMutable();
CGPathMoveToPoint(pathRef, NULL, itemX, centerLineY);
CGPathAddLineToPoint(pathRef, NULL, itemX, itemY);
CGPathAddLineToPoint(pathRef, NULL, itemX + itemWidth, itemY);
CGPathAddLineToPoint(pathRef, NULL, itemX + itemWidth, centerLineY);
CGPathCloseSubpath(pathRef);
CGContextAddPath(contextRef, pathRef);
CGContextFillPath(contextRef);
CGContextAddPath(contextRef, pathRef);
CGContextStrokePath(contextRef);
if (shouldDrawCircle) {
//画线末圆角
CGPathMoveToPoint(pathRef, NULL, itemX + itemWidth/2.f, itemY);
if (yield >= 0) {
CGPathAddArc(pathRef, NULL, itemX + itemWidth/2.f, itemY, itemWidth/2.f, M_PI, 0, NO);
}
else
{
CGPathAddArc(pathRef, NULL, itemX + itemWidth/2.f, itemY, itemWidth/2.f, 0, M_PI, NO);
}
CGPathCloseSubpath(pathRef);
CGContextAddPath(contextRef, pathRef);
CGContextFillPath(contextRef);
CGContextAddPath(contextRef, pathRef);
CGContextStrokePath(contextRef);
}
}
}
@end
iOS绘制收益柱状图的更多相关文章
- C# 绘制统计图(柱状图, 折线图, 扇形图)【转载】
统计图形种类繁多, 有柱状图, 折线图, 扇形图等等, 而统计图形的绘制方法也有很多, 有Flash制作的统计图形, 有水晶报表生成统计图形, 有专门制图软件制作, 也有编程语言自己制作的:这里我们用 ...
- C# 绘制统计图(柱状图, 折线图, 扇形图)
统计图形种类繁多, 有柱状图, 折线图, 扇形图等等, 而统计图形的绘制方法也有很多, 有Flash制作的统计图形, 有水晶报表生成统计图形, 有专门制图软件制作, 也有编程语言自己制作的:这里我们用 ...
- ASP.NET Core +Highchart+ajax绘制动态柱状图
一.项目介绍利用前端Highchart,以及ajax向后台获取数据,绘制动态柱状图.hightchart其他实例可查看官网文档.[Highchart](https://www.highcharts.c ...
- echarts —— 绘制横向柱状图(圆角、无坐标轴)
UI给了设计图,看了一眼觉得简单,不就是无序列表布局嘛(ul,li),后来才知道那是echarts图,好吧,样式如下: 代码如下:(渐变色没做) <!DOCTYPE html> <h ...
- 【转】使用DevExpress的WebChartControl控件绘制图表(柱状图、折线图、饼图)
第一次写博,没什么经验,主要是把最近自己对Dev的一些研究贴出来大家共同探讨,有不足之处望大家帮忙斧正. WebChartControl是DevExpress控件群下的一个Web图表控件,它使用非常的 ...
- ios 绘制wav波形图
最近研究了如何在iOS上绘制wav波形图.查了很多资料,都没能找到一个很完整的介绍,我这里总结一下一些经验. 首先需要了解wav的这3个重要指标:采样率.采样位数.声道数.下面以16KHz, 16Bi ...
- iOS绘制坐标图,折线图-Swift
坐标图,经常会在各种各样的App中使用,最常用的一种坐标图就是折线图,根据给定的点绘制出对应的坐标图是最基本的需求.由于本人的项目需要使用折线图,第一反应就是搜索已经存在的解决方案,因为这种需求应该很 ...
- 转:iOS绘制一个UIView
绘制一个UIView 绘制一个UIVIew最灵活的方式就是由它自己完成绘制.实际上你不是绘制一个UIView,你只是子类化了UIView并赋予子类绘制自己的能力.当一个UIVIew需要执行绘图操作的时 ...
- iOS 绘制1像素的线
一.Point Vs Pixel iOS中当我们使用Quartz,UIKit,CoreAnimation等框架时,所有的坐标系统采用Point来衡量.系统在实际渲染到设置时会帮助我们处理Point到P ...
随机推荐
- 阿里云服务器上配置并使用: PHP + Redis + Mysql 从配置到使用
(原创出处为本博客,http://www.cnblogs.com/linguanh/) 目录: 一,下载 二,解压 三,配置与启动 四,测试 Redis 五,配置 phpRedis 扩展 六,综合测试 ...
- Kooboo CMS技术文档之一:Kooboo CMS技术背景
语言平台 依赖注入方案 存储模型 1. 语言平台 Kooboo CMS基于.NET Framework 4.x,.NET Framework 4.x的一些技术特性成为站点开发人员使用Kooboo CM ...
- [C#] C# 知识回顾 - 特性 Attribute
C# 知识回顾 - 特性 Attribute [博主]反骨仔 [原文地址]http://www.cnblogs.com/liqingwen/p/5911289.html 目录 特性简介 使用特性 特性 ...
- BPM SharePoint解决方案分享
一.需求分析 SharePoint作为微软推出的协同类平台产品,为客户提供了门户.内容.文档.流程.社区.搜索.BI等一系列的解决方案,然而其流程功能由于设计理念差异,不能完全满足客户的需求,主要原因 ...
- Android—Volley:接收服务端发送的json数据乱码问题解决
new JsonObjectRequest中重写方法parseNetworkResponse,内容如下: /** * 重写此方法不会导致乱码 */ @Override protected Respon ...
- Android Weekly Notes Issue #234
Android Weekly Issue #234 December 4th, 2016 Android Weekly Issue #234 本期内容包括: ConstraintLayout的使用; ...
- postgresql 基本语法
postgresql数据库创建/修改/删除等写入类代码语法总结: 1,创建库 2,创建/删除表 2.1 创建表 create table myTableName 2.2 如果表不存在则创建表 crea ...
- java 泛型
1.Student stu =tool.getObj();右边得到的是Object类型,需要向下转型,强转换. 2. 3. 4.泛型方法不能被静态修饰这样写 5.如果想定义定义静态泛型方法,只能这样写 ...
- 《MSSQL2008技术内幕:T-SQL语言基础》读书笔记(上)
索引: 一.SQL Server的体系结构 二.查询 三.表表达式 四.集合运算 五.透视.逆透视及分组 六.数据修改 七.事务和并发 八.可编程对象 一.SQL Server体系结构 1.1 数据库 ...
- 物联网实验4 alljoyn物联网实验之手机局域网控制设备
AllJoyn开源物联网协议框架,官方描述是一个能够使连接设备之间进行互操作的通用软件框架和系统服务核心集,也是一个跨制造商来创建动态近端网络的软件应用.高通已经将该项目捐赠给了一个名为“AllSee ...