CGContextRef繪圖-iOS球形波浪加載進(jìn)度控件-HcdProcessView詳解

《iOS球形波浪加載進(jìn)度控件-HcdProcessView》這篇文章已經(jīng)展示了我在項(xiàng)目中編寫的一個(gè)球形進(jìn)度加載控件HcdProcessView子房,這篇文章我要簡(jiǎn)單介紹一下我的制作過(guò)程柜思。

思路

首先我放棄了使用通過(guò)改變圖片的位置來(lái)實(shí)現(xiàn)上面的動(dòng)畫效果,雖然這樣也可以實(shí)現(xiàn)如上的效果揩魂,但是從性能和資源消耗上來(lái)說(shuō)都不是最好的選擇。這里我采用了通過(guò)上下文(也就是CGContextRef)來(lái)繪制這樣的效果夷狰,大家對(duì)它應(yīng)該并不陌生袜腥,它既可以繪制直線、曲線胀莹、多邊形圓形以及各種各樣的幾何圖形基跑。

具體步驟

我們可以將上面的復(fù)雜圖形拆分成如下幾步:

  1. 繪制最外面的一圈刻度尺
  2. 繪制表示進(jìn)度的刻度尺
  3. 繪制中間的球形加載界面

繪制刻度尺

如果你先要在控件中繪制自己想要的圖形,你需要重寫UIView的drawRect方法:

- (void)drawRect:(CGRect)rect
{
    CGContextRef context = UIGraphicsGetCurrentContext();
    [self drawScale:context];
}

drawRect方法中描焰,我們先畫出了刻度尺的圖形媳否,刻度尺是由一圈短線在一個(gè)圓內(nèi)圍成的一個(gè)圓。

/**
 *  畫比例尺
 *
 *  @param context 全局context
 */
- (void)drawScale:(CGContextRef)context {
    
    CGContextSetLineWidth(context, _scaleDivisionsWidth);//線的寬度
    
    //先將參照點(diǎn)移到控件中心
    CGContextTranslateCTM(context, fullRect.size.width / 2, fullRect.size.width / 2);
    
    //設(shè)置線的顏色
    CGContextSetStrokeColorWithColor(context, [UIColor colorWithRed:0.655 green:0.710 blue:0.859 alpha:1.00].CGColor);//線框顏色
    //繪制一些圖形
    for (int i = 0; i < _scaleCount; i++) {
        CGContextMoveToPoint(context, scaleRect.size.width/2 - _scaleDivisionsLength, 0);
        CGContextAddLineToPoint(context, scaleRect.size.width/2, 0);
        //    CGContextScaleCTM(ctx, 0.5, 0.5);
        //渲染
        CGContextStrokePath(context);
        CGContextRotateCTM(context, 2 * M_PI / _scaleCount);
    }
    
    //繪制刻度尺外的一個(gè)圈
    CGContextSetStrokeColorWithColor(context, [UIColor colorWithRed:0.694 green:0.745 blue:0.867 alpha:1.00].CGColor);//線框顏色
    CGContextSetLineWidth(context, 0.5);
    CGContextAddArc (context, 0, 0, scaleRect.size.width/2 - _scaleDivisionsLength - 3, 0, M_PI* 2 , 0);
    CGContextStrokePath(context);
    
    //復(fù)原參照點(diǎn)
    CGContextTranslateCTM(context, -fullRect.size.width / 2, -fullRect.size.width / 2);
}

這里需要用到兩個(gè)東西一個(gè)是CGContextAddArc荆秦,一個(gè)是CGContextAddLineToPoint篱竭。創(chuàng)建圓弧的方法有兩種一種是CGContextAddArc,一種是CGContextAddArcToPoint步绸,這里畫的圓比較簡(jiǎn)單所以用的是CGContextAddArc,CGContextAddArcToPoint在后面也會(huì)用到(我會(huì)在用到的地方詳解)掺逼。

CGContextAddArc

 void CGContextAddArc (
    CGContextRef c,    
    CGFloat x,             //圓心的x坐標(biāo)
    CGFloat y,   //圓心的x坐標(biāo)
    CGFloat radius,   //圓的半徑 
    CGFloat startAngle,    //開始弧度
    CGFloat endAngle,   //結(jié)束弧度
    int clockwise          //0表示順時(shí)針,1表示逆時(shí)針
 );

這里需要?jiǎng)?chuàng)建一個(gè)完整的圓瓤介,那么 開始弧度就是0 結(jié)束弧度是 2PI吕喘, 因?yàn)閳A周長(zhǎng)是 2PIradius赘那。函數(shù)執(zhí)行完后,current point就被重置為(x,y)氯质。CGContextTranslateCTM(context, fullRect.size.width / 2, fullRect.size.width / 2);已經(jīng)將current point移動(dòng)到了(fullRect.size.width / 2, fullRect.size.width / 2)募舟。

CGContextAddLineToPoint

 void CGContextAddLineToPoint (
    CGContextRef c,
    CGFloat x,
    CGFloat y
 );

創(chuàng)建一條直線,從current point到 (x,y)
然后current point會(huì)變成(x,y)闻察。
由于短線不連續(xù)拱礁,所以通過(guò)for循環(huán)來(lái)不斷畫短線,_scaleCount代表的是刻度尺的個(gè)數(shù)辕漂,每次循環(huán)先將current point移動(dòng)到(scaleRect.size.width/2 - _scaleDivisionsLength, 0)點(diǎn)呢灶,_scaleDivisionsLength代表短線的長(zhǎng)度。繪制完短線后將前面繪制完成的圖形旋轉(zhuǎn)一個(gè)刻度尺的角度CGContextRotateCTM(context, 2 * M_PI / _scaleCount);钉嘹,將最終的繪制渲染后就得到了如下的刻度尺:

刻度尺上的進(jìn)度繪制

首先在drawRect中添加drawProcessScale方法鸯乃。

- (void)drawRect:(CGRect)rect
{
    CGContextRef context = UIGraphicsGetCurrentContext();
    [self drawScale:context];
    [self drawProcessScale:context];
}

然后在drawProcessScale方法中實(shí)現(xiàn)左右兩部分的刻度尺進(jìn)度繪制。

/**
 *  比例尺進(jìn)度
 *
 *  @param context 全局context
 */
- (void)drawProcessScale:(CGContextRef)context {
    
    CGContextSetLineWidth(context, _scaleDivisionsWidth);//線的寬度
    CGContextTranslateCTM(context, fullRect.size.width / 2, fullRect.size.width / 2);
    
    CGContextSetStrokeColorWithColor(context, [UIColor colorWithRed:0.969 green:0.937 blue:0.227 alpha:1.00].CGColor);//線框顏色
    
    int count = (_scaleCount / 2 + 1) * currentPercent;
    CGFloat scaleAngle = 2 * M_PI / _scaleCount;
    
    //繪制左邊刻度進(jìn)度
    for (int i = 0; i < count; i++) {
        CGContextMoveToPoint(context, 0, scaleRect.size.width/2 - _scaleDivisionsLength);
        CGContextAddLineToPoint(context, 0, scaleRect.size.width/2);
        //    CGContextScaleCTM(ctx, 0.5, 0.5);
        // 渲染
        CGContextStrokePath(context);
        CGContextRotateCTM(context, scaleAngle);
    }
    //繪制右邊刻度進(jìn)度
    CGContextRotateCTM(context, -count * scaleAngle);
    
    for (int i = 0; i < count; i++) {
        CGContextMoveToPoint(context, 0, scaleRect.size.width/2 - _scaleDivisionsLength);
        CGContextAddLineToPoint(context, 0, scaleRect.size.width/2);
        //    CGContextScaleCTM(ctx, 0.5, 0.5);
        // 渲染
        CGContextStrokePath(context);
        CGContextRotateCTM(context, -scaleAngle);
    }
    
    CGContextTranslateCTM(context, -fullRect.size.width / 2, -fullRect.size.width / 2);
}

繪制完后效果如下:

水的波浪效果繪制

終于到了最主要也是最難的效果繪制了隧期,對(duì)于帶有波浪不斷滾動(dòng)的效果是采用NSTimer來(lái)不斷繪制每一幀圖形實(shí)現(xiàn)的飒责,現(xiàn)在簡(jiǎn)單介紹下每一幀的繪制方法。
首先在drawRect中添加drawWave方法仆潮,

- (void)drawRect:(CGRect)rect
{
    CGContextRef context = UIGraphicsGetCurrentContext();
    [self drawScale:context];
    [self drawProcessScale:context];
    [self drawWave:context];
}

drawWave中實(shí)現(xiàn)如下方法:

/**
 *  畫波浪
 *
 *  @param context 全局context
 */
- (void)drawWave:(CGContextRef)context {
    
    CGMutablePathRef frontPath = CGPathCreateMutable();
    CGMutablePathRef backPath = CGPathCreateMutable();
    
    //畫水
    CGContextSetLineWidth(context, 1);
    CGContextSetFillColorWithColor(context, [_frontWaterColor CGColor]);
    
    CGFloat offset = _scaleMargin + _waveMargin + _scaleDivisionsWidth;
    
    float frontY = currentLinePointY;
    float backY = currentLinePointY;
    
    CGFloat radius = waveRect.size.width / 2;
    
    CGPoint frontStartPoint = CGPointMake(offset, currentLinePointY + offset);
    CGPoint frontEndPoint = CGPointMake(offset, currentLinePointY + offset);
    
    CGPoint backStartPoint = CGPointMake(offset, currentLinePointY + offset);
    CGPoint backEndPoint = CGPointMake(offset, currentLinePointY + offset);
    
    for(float x = 0; x <= waveRect.size.width; x++){
        
        //前浪繪制
        frontY = a * sin( x / 180 * M_PI + 4 * b / M_PI ) * amplitude + currentLinePointY;
        
        CGFloat frontCircleY = frontY;
        if (currentLinePointY < radius) {
            frontCircleY = radius - sqrt(pow(radius, 2) - pow((radius - x), 2));
            if (frontY < frontCircleY) {
                frontY = frontCircleY;
            }
        } else if (currentLinePointY > radius) {
            frontCircleY = radius + sqrt(pow(radius, 2) - pow((radius - x), 2));
            if (frontY > frontCircleY) {
                frontY = frontCircleY;
            }
        }
        
        if (fabs(0 - x) < 0.001) {
            frontStartPoint = CGPointMake(x + offset, frontY + offset);
            CGPathMoveToPoint(frontPath, NULL, frontStartPoint.x, frontStartPoint.y);
        }
        
        frontEndPoint = CGPointMake(x + offset, frontY + offset);
        CGPathAddLineToPoint(frontPath, nil, frontEndPoint.x, frontEndPoint.y);
        
        //后波浪繪制
        backY = a * cos( x / 180 * M_PI + 3 * b / M_PI ) * amplitude + currentLinePointY;
        CGFloat backCircleY = backY;
        if (currentLinePointY < radius) {
            backCircleY = radius - sqrt(pow(radius, 2) - pow((radius - x), 2));
            if (backY < backCircleY) {
                backY = backCircleY;
            }
        } else if (currentLinePointY > radius) {
            backCircleY = radius + sqrt(pow(radius, 2) - pow((radius - x), 2));
            if (backY > backCircleY) {
                backY = backCircleY;
            }
        }
        
        if (fabs(0 - x) < 0.001) {
            backStartPoint = CGPointMake(x + offset, backY + offset);
            CGPathMoveToPoint(backPath, NULL, backStartPoint.x, backStartPoint.y);
        }
        
        backEndPoint = CGPointMake(x + offset, backY + offset);
        CGPathAddLineToPoint(backPath, nil, backEndPoint.x, backEndPoint.y);
    }
    
    CGPoint centerPoint = CGPointMake(fullRect.size.width / 2, fullRect.size.height / 2);
    
    //繪制前浪圓弧
    CGFloat frontStart = [self calculateRotateDegree:centerPoint point:frontStartPoint];
    CGFloat frontEnd = [self calculateRotateDegree:centerPoint point:frontEndPoint];
    
    CGPathAddArc(frontPath, nil, centerPoint.x, centerPoint.y, waveRect.size.width / 2, frontEnd, frontStart, 0);
    CGContextAddPath(context, frontPath);
    CGContextFillPath(context);
    //推入
    CGContextSaveGState(context);
    CGContextDrawPath(context, kCGPathStroke);
    CGPathRelease(frontPath);
    
    
    //繪制后浪圓弧
    CGFloat backStart = [self calculateRotateDegree:centerPoint point:backStartPoint];
    CGFloat backEnd = [self calculateRotateDegree:centerPoint point:backEndPoint];
    
    CGPathAddArc(backPath, nil, centerPoint.x, centerPoint.y, waveRect.size.width / 2, backEnd, backStart, 0);
    
    CGContextSetFillColorWithColor(context, [_backWaterColor CGColor]);
    CGContextAddPath(context, backPath);
    CGContextFillPath(context);
    //推入
    CGContextSaveGState(context);
    CGContextDrawPath(context, kCGPathStroke);
    CGPathRelease(backPath);
    
}

上面的代碼較長(zhǎng)宏蛉,可能也比較難以理解。下面我將會(huì)對(duì)上述代碼簡(jiǎn)單解讀一下性置,已前浪為例(前浪和后浪的實(shí)現(xiàn)方式基本一樣拾并,只是兩個(gè)浪正余弦函數(shù)不一樣而已)。兩個(gè)浪都是由一條曲線和和一個(gè)圓弧構(gòu)成的封閉區(qū)間鹏浅,曲線的x區(qū)間為[0, waveRect.size.width]嗅义,y值坐標(biāo)為frontY = a * sin( x / 180 * M_PI + 4 * b / M_PI ) * amplitude + currentLinePointY;(currentLinePointY為偏移量),通過(guò)for循環(huán)自增x隐砸,計(jì)算出y的位置來(lái)不斷CGPathAddLineToPoint繪制出一條曲線之碗,這就構(gòu)成了波浪的曲線。然后我們需要根據(jù)波浪曲線的起始點(diǎn)和結(jié)束點(diǎn)以及圓心點(diǎn)(fullRect.size.width / 2, fullRect.size.height / 2)季希,來(lái)繪制一段封閉的圓弧褪那。
這里就需要用到CGPathAddArc方法;CGPathAddArc方法和CGContextAddArc類似式塌。需要先計(jì)算出點(diǎn)波浪的起始點(diǎn)和結(jié)束點(diǎn)分別與圓心之間的夾角博敬。知道兩點(diǎn)計(jì)算夾角的方式如下:

/**
 *  根據(jù)圓心點(diǎn)和圓上一個(gè)點(diǎn)計(jì)算角度
 *
 *  @param centerPoint 圓心點(diǎn)
 *  @param point       圓上的一個(gè)點(diǎn)
 *
 *  @return 角度
 */
- (CGFloat)calculateRotateDegree:(CGPoint)centerPoint point:(CGPoint)point {
    
    CGFloat rotateDegree = asin(fabs(point.y - centerPoint.y) / (sqrt(pow(point.x - centerPoint.x, 2) + pow(point.y - centerPoint.y, 2))));
    
    //如果point縱坐標(biāo)大于原點(diǎn)centerPoint縱坐標(biāo)(在第一和第二象限)
    if (point.y > centerPoint.y) {
        //第一象限
        if (point.x >= centerPoint.x) {
            rotateDegree = rotateDegree;
        }
        //第二象限
        else {
            rotateDegree = M_PI - rotateDegree;
        }
    } else //第三和第四象限
    {
        if (point.x <= centerPoint.x) //第三象限,不做任何處理
        {
            rotateDegree = M_PI + rotateDegree;
        }
        else //第四象限
        {
            rotateDegree = 2 * M_PI - rotateDegree;
        }
    }
    return rotateDegree;
}

波浪繪制的相關(guān)判斷

由于曲線x區(qū)間是[0, waveRect.size.width]峰尝,y值是根據(jù)公式frontY = a * sin( x / 180 * M_PI + 4 * b / M_PI ) * amplitude + currentLinePointY;計(jì)算出來(lái)的偏窝,但是最終構(gòu)成的波浪是一個(gè)球形的,所以對(duì)于計(jì)算出來(lái)的y值坐標(biāo),我們需要判斷它是否在圓上祭往,如果不在圓上伦意,我們應(yīng)該將它移到圓上。

判斷分為兩種情況:

currentLinePointY<fullRect.size.height / 2

當(dāng)currentLinePointY<fullRect.size.height / 2時(shí)链沼,已知點(diǎn)的坐標(biāo)x默赂,根據(jù)公式y1 = a * sin( x / 180 * M_PI + 4 * b / M_PI ) * amplitude + currentLinePointY;算出來(lái)的點(diǎn)位置為(x, y1),而在圓上點(diǎn)坐標(biāo)為x的點(diǎn)的位置在(x,y2)括勺,如果y1<y2 則最終應(yīng)該放到波浪上的點(diǎn)為 (x,y2)

currentLinePointY>fullRect.size.height / 2

同理當(dāng)currentLinePointY>fullRect.size.height / 2時(shí)曲掰,已知點(diǎn)的坐標(biāo)x疾捍,根據(jù)公式y1 = a * sin( x / 180 * M_PI + 4 * b / M_PI ) * amplitude + currentLinePointY;算出來(lái)的點(diǎn)位置為(x, y1),而在圓上點(diǎn)坐標(biāo)為x的點(diǎn)的位置在(x,y2)栏妖,如果y1>y2 則最終應(yīng)該放到波浪上的點(diǎn)為 (x,y2)乱豆。

其中判斷的代碼如下:

frontY = a * sin( x / 180 * M_PI + 4 * b / M_PI ) * amplitude + currentLinePointY;
        
CGFloat frontCircleY = frontY;
if (currentLinePointY < radius) {
    frontCircleY = radius - sqrt(pow(radius, 2) - pow((radius - x), 2));
    if (frontY < frontCircleY) {
        frontY = frontCircleY;
    }
} else if (currentLinePointY > radius) {
    frontCircleY = radius + sqrt(pow(radius, 2) - pow((radius - x), 2));
    if (frontY > frontCircleY) {
        frontY = frontCircleY;
    }
}

其中當(dāng)currentLinePointY < radius時(shí),y2=radius - sqrt(pow(radius, 2) - pow((radius - x), 2));
當(dāng)currentLinePointY > radius時(shí)吊趾,y2=radius + sqrt(pow(radius, 2) - pow((radius - x), 2))宛裕;

這樣就構(gòu)成了一個(gè)如下的效果:


然后通過(guò)Timer不斷的改變ab的值就得到了我想要的動(dòng)畫效果论泛。

Github地址:https://github.com/Jvaeyhcd/HcdProcessView

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末揩尸,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子屁奏,更是在濱河造成了極大的恐慌岩榆,老刑警劉巖,帶你破解...
    沈念sama閱讀 217,907評(píng)論 6 506
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件坟瓢,死亡現(xiàn)場(chǎng)離奇詭異勇边,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)折联,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,987評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門粒褒,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人诚镰,你說(shuō)我怎么就攤上這事奕坟。” “怎么了怕享?”我有些...
    開封第一講書人閱讀 164,298評(píng)論 0 354
  • 文/不壞的土叔 我叫張陵执赡,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我函筋,道長(zhǎng)沙合,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,586評(píng)論 1 293
  • 正文 為了忘掉前任跌帐,我火速辦了婚禮首懈,結(jié)果婚禮上绊率,老公的妹妹穿的比我還像新娘。我一直安慰自己究履,他們只是感情好滤否,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,633評(píng)論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著最仑,像睡著了一般藐俺。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上泥彤,一...
    開封第一講書人閱讀 51,488評(píng)論 1 302
  • 那天欲芹,我揣著相機(jī)與錄音,去河邊找鬼吟吝。 笑死菱父,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的剑逃。 我是一名探鬼主播浙宜,決...
    沈念sama閱讀 40,275評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼蛹磺!你這毒婦竟也來(lái)了粟瞬?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,176評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤称开,失蹤者是張志新(化名)和其女友劉穎亩钟,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體鳖轰,經(jīng)...
    沈念sama閱讀 45,619評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡清酥,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,819評(píng)論 3 336
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了蕴侣。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片焰轻。...
    茶點(diǎn)故事閱讀 39,932評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖昆雀,靈堂內(nèi)的尸體忽然破棺而出辱志,到底是詐尸還是另有隱情,我是刑警寧澤狞膘,帶...
    沈念sama閱讀 35,655評(píng)論 5 346
  • 正文 年R本政府宣布揩懒,位于F島的核電站,受9級(jí)特大地震影響挽封,放射性物質(zhì)發(fā)生泄漏已球。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,265評(píng)論 3 329
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望智亮。 院中可真熱鬧忆某,春花似錦、人聲如沸阔蛉。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,871評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)状原。三九已至聋呢,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間颠区,已是汗流浹背坝冕。 一陣腳步聲響...
    開封第一講書人閱讀 32,994評(píng)論 1 269
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留瓦呼,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 48,095評(píng)論 3 370
  • 正文 我出身青樓测暗,卻偏偏與公主長(zhǎng)得像央串,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子碗啄,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,884評(píng)論 2 354

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