ios實(shí)現(xiàn)顏色漸變的幾種方法

demo地址:https://github.com/xiaochaofeiyu/YSCAnimation
有用的話求個(gè)star杖剪,歡迎建議討論笆环。

1. CAGradientLayer實(shí)現(xiàn)漸變

CAGradientLayer是CALayer的一個(gè)特殊子類忍弛,用于生成顏色漸變的圖層癞揉,使用較為方便萧锉,下面介紹下它的相關(guān)屬性:

  1. colors 漸變的顏色
  2. locations 漸變顏色的分割點(diǎn)
  3. startPoint&endPoint 顏色漸變的方向践美,范圍在(0,0)與(1.0,1.0)之間,如(0,0)(1.0,0)代表水平方向漸變,(0,0)(0,1.0)代表豎直方向漸變
    CAGradientLayer *gradientLayer = [CAGradientLayer layer];
    gradientLayer.colors = @[(__bridge id)[UIColor redColor].CGColor, (__bridge id)[UIColor yellowColor].CGColor, (__bridge id)[UIColor blueColor].CGColor];
    gradientLayer.locations = @[@0.3, @0.5, @1.0];
    gradientLayer.startPoint = CGPointMake(0, 0);
    gradientLayer.endPoint = CGPointMake(1.0, 0);
    gradientLayer.frame = CGRectMake(0, 100, 300, 100);
    [self.view.layer addSublayer:gradientLayer];

CAGradientLayer實(shí)現(xiàn)漸變標(biāo)間簡(jiǎn)單直觀寄猩,但存在一定的局限性嫉晶,比如無(wú)法自定義整個(gè)漸變區(qū)域的形狀,如環(huán)形、曲線形的漸變替废。

2. Core Graphics相關(guān)方法實(shí)現(xiàn)漸變

iOS Core Graphics中有兩個(gè)方法用于繪制漸變顏色箍铭,CGContextDrawLinearGradient可以用于生成線性漸變,CGContextDrawRadialGradient用于生成圓半徑方向顏色漸變椎镣。函數(shù)可以自定義path坡疼,無(wú)論是什么形狀都可以,原理都是用來(lái)做Clip衣陶,所以需要在CGContextClip函數(shù)前調(diào)用CGContextAddPath函數(shù)把CGPathRef加入到Context中。
另外一個(gè)需要注意的地方是漸變的方向闸氮,方向是由兩個(gè)點(diǎn)控制的剪况,點(diǎn)的單位就是坐標(biāo)。因此需要正確從CGPathRef中找到正確的點(diǎn)蒲跨,方法當(dāng)然有很多種看具體實(shí)現(xiàn)译断,本例中,我就是簡(jiǎn)單得通過(guò)調(diào)用CGPathGetBoundingBox函數(shù)或悲,返回CGPathRef的矩形區(qū)域孙咪,然后根據(jù)這個(gè)矩形取兩個(gè)點(diǎn),讀者可以根據(jù)自行需求修改具體代碼巡语。

1-> 線性漸變

- (void)drawLinearGradient:(CGContextRef)context
                      path:(CGPathRef)path
                startColor:(CGColorRef)startColor
                  endColor:(CGColorRef)endColor
{
    CGColorSpaceRef colorSpace = CGColorSpaceCreateDeviceRGB();
    CGFloat locations[] = { 0.0, 1.0 };
    
    NSArray *colors = @[(__bridge id) startColor, (__bridge id) endColor];
    
    CGGradientRef gradient = CGGradientCreateWithColors(colorSpace, (__bridge CFArrayRef) colors, locations);
    
    
    CGRect pathRect = CGPathGetBoundingBox(path);
    
    //具體方向可根據(jù)需求修改
    CGPoint startPoint = CGPointMake(CGRectGetMinX(pathRect), CGRectGetMidY(pathRect));
    CGPoint endPoint = CGPointMake(CGRectGetMaxX(pathRect), CGRectGetMidY(pathRect));
    
    CGContextSaveGState(context);
    CGContextAddPath(context, path);
    CGContextClip(context);
    CGContextDrawLinearGradient(context, gradient, startPoint, endPoint, 0);
    CGContextRestoreGState(context);
    
    CGGradientRelease(gradient);
    CGColorSpaceRelease(colorSpace);
}

- (void)viewDidLoad 
{
    [super viewDidLoad];
    // Do any additional setup after loading the view.
    
    //創(chuàng)建CGContextRef
    UIGraphicsBeginImageContext(self.view.bounds.size);
    CGContextRef gc = UIGraphicsGetCurrentContext();
    
    //創(chuàng)建CGMutablePathRef
    CGMutablePathRef path = CGPathCreateMutable();
    
    //繪制Path
    CGRect rect = CGRectMake(0, 100, 300, 200);
    CGPathMoveToPoint(path, NULL, CGRectGetMinX(rect), CGRectGetMinY(rect));
    CGPathAddLineToPoint(path, NULL, CGRectGetMidX(rect), CGRectGetMaxY(rect));
    CGPathAddLineToPoint(path, NULL, CGRectGetWidth(rect), CGRectGetMaxY(rect));
    CGPathCloseSubpath(path);
    
    //繪制漸變
    [self drawLinearGradient:gc path:path startColor:[UIColor greenColor].CGColor endColor:[UIColor redColor].CGColor];
    
    //注意釋放CGMutablePathRef
    CGPathRelease(path);
    
    //從Context中獲取圖像翎蹈,并顯示在界面上
    UIImage *img = UIGraphicsGetImageFromCurrentImageContext();
    UIGraphicsEndImageContext();
    
    UIImageView *imgView = [[UIImageView alloc] initWithImage:img];
    [self.view addSubview:imgView];
}

2-> 圓半徑方向漸變

- (void)drawRadialGradient:(CGContextRef)context
                      path:(CGPathRef)path
                startColor:(CGColorRef)startColor
                  endColor:(CGColorRef)endColor
{
    CGColorSpaceRef colorSpace = CGColorSpaceCreateDeviceRGB();
    CGFloat locations[] = { 0.0, 1.0 };
    
    NSArray *colors = @[(__bridge id) startColor, (__bridge id) endColor];
    
    CGGradientRef gradient = CGGradientCreateWithColors(colorSpace, (__bridge CFArrayRef) colors, locations);
    
    
    CGRect pathRect = CGPathGetBoundingBox(path);
    CGPoint center = CGPointMake(CGRectGetMidX(pathRect), CGRectGetMidY(pathRect));
    CGFloat radius = MAX(pathRect.size.width / 2.0, pathRect.size.height / 2.0) * sqrt(2);
    
    CGContextSaveGState(context);
    CGContextAddPath(context, path);
    CGContextEOClip(context);
    
    CGContextDrawRadialGradient(context, gradient, center, 0, center, radius, 0);
    
    CGContextRestoreGState(context);
    
    CGGradientRelease(gradient);
    CGColorSpaceRelease(colorSpace);
}

- (void)viewDidLoad 
{
    [super viewDidLoad];
    // Do any additional setup after loading the view.
    
    //創(chuàng)建CGContextRef
    UIGraphicsBeginImageContext(self.view.bounds.size);
    CGContextRef gc = UIGraphicsGetCurrentContext();
    
    //創(chuàng)建CGMutablePathRef
    CGMutablePathRef path = CGPathCreateMutable();
    
    //繪制Path
    CGRect rect = CGRectMake(0, 100, 300, 200);
    CGPathMoveToPoint(path, NULL, CGRectGetMinX(rect), CGRectGetMinY(rect));
    CGPathAddLineToPoint(path, NULL, CGRectGetMidX(rect), CGRectGetMaxY(rect));
    CGPathAddLineToPoint(path, NULL, CGRectGetWidth(rect), CGRectGetMaxY(rect));
    CGPathAddLineToPoint(path, NULL, CGRectGetWidth(rect), CGRectGetMinY(rect));
    CGPathCloseSubpath(path);
    
    //繪制漸變
    [self drawRadialGradient:gc path:path startColor:[UIColor greenColor].CGColor endColor:[UIColor redColor].CGColor];
    
    //注意釋放CGMutablePathRef
    CGPathRelease(path);
    
    //從Context中獲取圖像,并顯示在界面上
    UIImage *img = UIGraphicsGetImageFromCurrentImageContext();
    UIGraphicsEndImageContext();
    
    UIImageView *imgView = [[UIImageView alloc] initWithImage:img];
    [self.view addSubview:imgView];
}

3. 以CAShapeLayer作為layer的mask屬性

CALayer的mask屬性可以作為遮罩讓layer顯示mask遮住(非透明)的部分男公;CAShapeLayer為CALayer的子類荤堪,通過(guò)path屬性可以生成不同的形狀,將CAShapeLayer對(duì)象用作layer的mask屬性的話枢赔,就可以生成不同形狀的圖層澄阳。故生成顏色漸變有以下幾個(gè)步驟:

  1. 生成一個(gè)imageView(也可以為layer),image的屬性為顏色漸變的圖片
  2. 生成一個(gè)CAShapeLayer對(duì)象踏拜,根據(jù)path屬性指定所需的形狀
  3. 將CAShapeLayer對(duì)象賦值給imageView的mask屬性
- (void)viewDidLoad
{
    [super viewDidLoad];
        
    [self.view addSubview:self.firstCircle];
    _firstCircle.frame = CGRectMake(0, 0, 200, 200);
    _firstCircle.center = CGPointMake(CGRectGetWidth(self.view.bounds) / 2.0, CGRectGetHeight(self.view.bounds) / 2.0);
    CGFloat firsCircleWidth = 5;
    self.firstCircleShapeLayer = [self generateShapeLayerWithLineWidth:firsCircleWidth];
    _firstCircleShapeLayer.path = [self generateBezierPathWithCenter:CGPointMake(100, 100) radius:100].CGPath;
    _firstCircle.layer.mask = _firstCircleShapeLayer;
} 

- (CAShapeLayer *)generateShapeLayerWithLineWidth:(CGFloat)lineWidth
{
    CAShapeLayer *waveline = [CAShapeLayer layer];
    waveline.lineCap = kCALineCapButt;
    waveline.lineJoin = kCALineJoinRound;
    waveline.strokeColor = [UIColor redColor].CGColor;
    waveline.fillColor = [[UIColor clearColor] CGColor];
    waveline.lineWidth = lineWidth;
    waveline.backgroundColor = [UIColor clearColor].CGColor;
    
    return waveline;
}

- (UIBezierPath *)generateBezierPathWithCenter:(CGPoint)center radius:(CGFloat)radius
{
    UIBezierPath *circlePath = [UIBezierPath bezierPathWithArcCenter:center radius:radius startAngle:0 endAngle:2*M_PI clockwise:NO];
    
    return circlePath;
}

- (UIImageView *)firstCircle
{
    if (!_firstCircle) {
        self.firstCircle = [[UIImageView alloc] initWithImage:[UIImage imageNamed:@"circleBackground"]];
        _firstCircle.layer.masksToBounds = YES;
        _firstCircle.alpha = 1.0;
    }
    
    return _firstCircle;
}

demo地址:https://github.com/xiaochaofeiyu/YSCAnimation
在YSCNewVoiceWaveView與YSCVoiceLoadingCircleView中

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末碎赢,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子速梗,更是在濱河造成了極大的恐慌肮塞,老刑警劉巖,帶你破解...
    沈念sama閱讀 219,427評(píng)論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件镀琉,死亡現(xiàn)場(chǎng)離奇詭異峦嗤,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)屋摔,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,551評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門烁设,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人,你說(shuō)我怎么就攤上這事装黑「逼伲” “怎么了?”我有些...
    開(kāi)封第一講書人閱讀 165,747評(píng)論 0 356
  • 文/不壞的土叔 我叫張陵恋谭,是天一觀的道長(zhǎng)糠睡。 經(jīng)常有香客問(wèn)我,道長(zhǎng)疚颊,這世上最難降的妖魔是什么狈孔? 我笑而不...
    開(kāi)封第一講書人閱讀 58,939評(píng)論 1 295
  • 正文 為了忘掉前任,我火速辦了婚禮材义,結(jié)果婚禮上均抽,老公的妹妹穿的比我還像新娘。我一直安慰自己其掂,他們只是感情好油挥,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,955評(píng)論 6 392
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著款熬,像睡著了一般深寥。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上贤牛,一...
    開(kāi)封第一講書人閱讀 51,737評(píng)論 1 305
  • 那天惋鹅,我揣著相機(jī)與錄音,去河邊找鬼殉簸。 笑死负饲,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的喂链。 我是一名探鬼主播返十,決...
    沈念sama閱讀 40,448評(píng)論 3 420
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼椭微!你這毒婦竟也來(lái)了洞坑?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書人閱讀 39,352評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤蝇率,失蹤者是張志新(化名)和其女友劉穎迟杂,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體本慕,經(jīng)...
    沈念sama閱讀 45,834評(píng)論 1 317
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡排拷,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,992評(píng)論 3 338
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了锅尘。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片监氢。...
    茶點(diǎn)故事閱讀 40,133評(píng)論 1 351
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出浪腐,到底是詐尸還是另有隱情纵揍,我是刑警寧澤,帶...
    沈念sama閱讀 35,815評(píng)論 5 346
  • 正文 年R本政府宣布议街,位于F島的核電站泽谨,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏特漩。R本人自食惡果不足惜吧雹,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,477評(píng)論 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望涂身。 院中可真熱鬧吮炕,春花似錦、人聲如沸访得。這莊子的主人今日做“春日...
    開(kāi)封第一講書人閱讀 32,022評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)悍抑。三九已至,卻和暖如春杜耙,著一層夾襖步出監(jiān)牢的瞬間搜骡,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書人閱讀 33,147評(píng)論 1 272
  • 我被黑心中介騙來(lái)泰國(guó)打工佑女, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留记靡,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 48,398評(píng)論 3 373
  • 正文 我出身青樓团驱,卻偏偏與公主長(zhǎng)得像摸吠,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子嚎花,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,077評(píng)論 2 355

推薦閱讀更多精彩內(nèi)容