OC_面向对象化色彩渐变工具

线性渐变
扩散渐变

tag: CGColorSpaceRef CGContextDrawLinearGradient CGContextDrawRadialGradient
demo 点这里
常用的框架集锦 PYKit
pod 导入 pod 'PYGradientView'

示例

  1. 添加一个view
self.gradientView = [PYGradientView new];
    CGRect gradientViewFrame = self.view.bounds;
    self.gradientView.frame = gradientViewFrame;
    [self.view addSubview:self.gradientView];
  1. 一共有两种config
    1. 线性渐变PYGradientViewLineConfig
      PYGradientView对象调用方法 drawLineGradient 设置线性config,并立马绘制渐变
[self.gradientView drawLineGradient:^(PYGradientViewLineConfig *lineConfig) {
            lineConfig
            .setUpScaleEndCenter(CGPointMake(0, 0))
            .setUpScaleStartCenter(CGPointMake(1, 1))
            .setUpColorArray(@[
                               UIColor.redColor,
                               UIColor.cyanColor
                               ])
            .setUpLocationArray(@[
                                  @0,@1
                                  ]);
    
        }];
  • 扩散渐变PYGradientViewDrawRadialConfig
    PYGradientView对象调用方法 drawRadialGradient 设置线性config,并立马绘制渐变
[self.gradientView drawRadialGradient:^(PYGradientViewDrawRadialConfig *radialConfig) {
        radialConfig
        .setUpScaleEndCenter(CGPointMake(0.5, 0.5))
        .setUpScaleStartCenter(CGPointMake(0.5, 0.5))
        .setUpColorArray(@[
                           UIColor.redColor,
                           UIColor.blueColor
                           ])
        .setUpStartRadius(0)
        .setUpEndRadius(1000)
        .setUpLocationArray(@[@0.1,@1]);
    }];

具体实现

重写drawRect方法

  1. 线性渐变主要方法
#pragma mark 线性渐变
- (void) drawLineGradientWithContext: (CGContextRef)context {
    //2.创建色彩空间
    CGColorSpaceRef colorSpaceRef = CGColorSpaceCreateDeviceRGB();
    
    //3.创建渐变对象
    NSInteger count = self.drawLineConfig.colorArray.count;
    CGFloat *components = [self parseColorArray:self.drawLineConfig.colorArray];
    CGFloat *locations = [self parseLocationWithArray:self.drawLineConfig.locationArray andLenth:count];
    CGGradientRef gradientRef = CGGradientCreateWithColorComponents(colorSpaceRef,components,locations,count);
    PYGradientViewLineConfig *config = self.drawLineConfig;
    CGPoint startCenter = [self getCenterWithCenter:config.startCenter andScaleCenter:config.startScaleCenter];
    CGPoint endCenter = [self getCenterWithCenter:config.endCenter andScaleCenter:config.endScaleCenter];
    
    /**
     * gradientRef 表示颜色空间
     * startCenter 表示绝对开始位置
     * endCenter 表示绝对结束位置
     */
    CGContextDrawLinearGradient(context,
                                gradientRef,
                                startCenter,
                                endCenter,
                                config.options);
    [self freeArray:components];
    [self freeArray:locations];
    CFRelease(colorSpaceRef);
    CFRelease(gradientRef);
}
  1. 扩散渐变
#pragma mark 径向渐变
-(void)drawRadialGradientWithContext:(CGContextRef)context{
    //使用rgb颜色空间
    CGColorSpaceRef colorSpace=CGColorSpaceCreateDeviceRGB();
    
    /*指定渐变色
     space:颜色空间
     components:颜色数组,注意由于指定了RGB颜色空间,那么四个数组元素表示一个颜色(red、green、blue、alpha),
     如果有三个颜色则这个数组有4*3个元素
     locations:颜色所在位置(范围0~1),这个数组的个数不小于components中存放颜色的个数
     count:渐变个数,等于locations的个数
     */
    /// 分析color的ARGB
    NSInteger count = self.drawRadialConfig.colorArray.count;
     CGFloat *compoents = [self parseColorArray:self.drawRadialConfig.colorArray];

    CGFloat *locations= [self parseLocationWithArray:self.drawRadialConfig.locationArray andLenth:count];
    CGGradientRef gradient= CGGradientCreateWithColorComponents(colorSpace, compoents, locations, count);
    
    /*绘制径向渐变
     context:图形上下文
     gradient:渐变色
     startCenter:起始点位置
     startRadius:起始半径(通常为0,否则在此半径范围内容无任何填充)
     endCenter:终点位置(通常和起始点相同,否则会有偏移)
     endRadius:终点半径(也就是渐变的扩散长度)
     options:绘制方式,kCGGradientDrawsBeforeStartLocation 开始位置之前就进行绘制,但是到结束位置之后不再绘制,
     kCGGradientDrawsAfterEndLocation开始位置之前不进行绘制,但到结束点之后继续填充
     */
    PYGradientViewDrawRadialConfig *config = self.drawRadialConfig;
    CGPoint startCenter = [self getCenterWithCenter:config.startCenter andScaleCenter:config.startScaleCenter];
    CGPoint endCenter = [self getCenterWithCenter:config.endCenter andScaleCenter:config.endScaleCenter];
    
   
    CGContextDrawRadialGradient(context,
                                gradient,
                                startCenter,
                                config.startRadius,
                                endCenter,
                                config.endRadius,
                                config.options);
    //释放颜色空间
    CGColorSpaceRelease(colorSpace);
    //释放变色对象
    CGGradientRelease(gradient);
    //释放开辟的颜色数组内存空间
    [self freeArray: compoents];
    [self freeArray:locations];
}
  1. NSArrayarray 转化成CGFloat 及其他方法

- (CGPoint) getCenterWithCenter:(CGPoint)center andScaleCenter: (CGPoint)scaleCenter {
    if(!CGPointEqualToPoint(center, PYGradientViewConfigPointDefault)) {
        return center;
    }
    if (!CGPointEqualToPoint(scaleCenter, PYGradientViewConfigPointDefault)) {
        CGFloat x = self.frame.size.width * scaleCenter.x;
        CGFloat y = self.frame.size.height * scaleCenter.y;
        return CGPointMake(x, y);
    }
    NSLog(@". \n 🌶  %@: 获取Center失败",[self class]);
    return CGPointMake(0, 0);
}

- (CGFloat *)parseColorArray: (NSArray <UIColor *>*)colorArray {
    NSInteger count = colorArray.count;
    CGFloat *compoents = [self createArrayWithLenth:4 * count];
    for (int idx = 0; idx < count; idx ++) {
        UIColor *color = colorArray[idx];
        CGFloat r,g,b,a = 0;
        [color getRed:&r green:&g blue:&b alpha:&a];
        NSInteger currentIMinValue = idx * 4;
        CGFloat RGBA[4] = {r,g,b,a};
        for (NSInteger i = currentIMinValue; i < 4 * (idx + 1); i ++) {
            /// 获取ARGB
            compoents[i] = RGBA[i-currentIMinValue];
        }
    }
    return compoents;
}


- (CGFloat *) parseLocationWithArray: (NSArray <NSNumber *>*)array andLenth: (NSInteger)lenth{
    NSInteger count = lenth;
    CGFloat *locations = [self createArrayWithLenth:count];
    
    for (int i = 0; i < count; i++) {
        CGFloat value = 1;
        if (array.count <= i) {
            value = array.lastObject.floatValue;
        }else{
            value = array[i].floatValue;
        };
        locations[i] = value;
        
    }
    return locations;
}


/**
 创建c数组

 @param len 长度
 @return 返回c数组
 */
- (CGFloat *)createArrayWithLenth: (NSInteger)len {
    return malloc(sizeof(CGFloat) * len);
}

- (void) freeArray: (CGFloat *)array {
    free(array);
}

- (CGFloat *)arrayAddLenth: (NSInteger)len andArray: (CGFloat *)array{
    len = sizeof(array)/sizeof(*array) + len;
    CGFloat *array_old = array;
    
    array = (CGFloat *)realloc(array,sizeof(CGFloat)*len);
    
    /**
     *如果地址改变,代表内存在另一个地方划分了一个新的内存空间,
     *要释放旧的内存空间
     */
    if(array_old != array)
        free(array_old);
    return array;
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 158,847评论 4 362
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,208评论 1 292
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 108,587评论 0 243
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 43,942评论 0 205
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,332评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,587评论 1 218
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,853评论 2 312
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,568评论 0 198
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,273评论 1 242
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,542评论 2 246
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,033评论 1 260
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,373评论 2 253
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,031评论 3 236
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,073评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,830评论 0 195
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,628评论 2 274
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,537评论 2 269