【iOS】仿中国联通手机营业厅首页波纹圆环

先上效果图:


003.gif

使用联通号码的朋友对此动画应该很熟悉,自从更新5.0版本后开始出现这个波纹的圆环,分析了一下实现原理,用的东西还不少,使我不得不把高中的数学知识再温习一遍,话说高中的数学知识都还给老师了,深感惭愧啊,波浪线使用的是正余弦函数:


d6ca7bcb0a46f21f0f04e644f0246b600c33aef1.png
正弦型函数解析式:y=Asin(ωx+φ)+b
各常数值对函数图像的影响:
φ:决定波形与X轴位置关系或横向移动距离(左加右减)
ω:决定周期(最小正周期T=2π/∣ω∣)
A:决定峰值(即纵向拉伸压缩的倍数)
b:表示波形在Y轴的位置关系或纵向移动距离(上加下减)

我们想绘制波纹图形,就要用正弦型函数解析式:y=Asin(ωx+φ)+b获取y坐标,将圆环放入坐标系中,并结合我们的手机屏幕的坐标;
圆环的直径相当于一个周期2M_PI,其y坐标为sinf(2 * M_PI / 直径 * x + self.offset * M_PI * 2 / 直径)
代码如下:

设置需显示的数据
- (void)setTitle:(NSString *)title num:(NSString *)num des:(NSString *)des recharge:(NSString *)recharge
{
    self.firstLayer.fillColor = self.firstColor.CGColor;
    self.secondLayer.fillColor = self.secondColor.CGColor;
    [self setContent];
    _titleL.text = title;
    _numL.text = num;
    _desL.text = des;
    [_rechargeB setTitle:recharge forState:UIControlStateNormal];
    _height = self.bounds.size.height * (1 - self.progress);
    [self stopWave];
    [self startWave];
}
- (void)startWave
{
    self.timer = [CADisplayLink displayLinkWithTarget:self selector:@selector(waveAnimation)];
    [self.timer addToRunLoop:[NSRunLoop currentRunLoop] forMode:NSRunLoopCommonModes];
}

看到CADisplayLink你可能会感觉到陌生,CADisplayLink是一个能让我们以和屏幕刷新率相同的频率将内容画到屏幕上的定时器。当创建一个新的CADisplayLink对象并把它添加到一个runloop中,当CADisplayLink以特定的模式注册到runloop之后,每当屏幕需要刷新的时候,runloop就会调用CADisplayLink绑定的target的selector,这时target可以读到CADisplayLink的每次调用的时间戳,用来准备下一帧显示需要的数据。
绘制波纹路径的代码:

- (void)waveAnimation
{
    CGFloat waveHeight = self.waveRange;
    if (self.progress == 0.0f || self.progress == 1.0f) {
        waveHeight = 0.f;
    }
    self.offset += self.rate;
    //第一条波纹路径
    CGMutablePathRef pathRef = CGPathCreateMutable();
    CGFloat startOffY = waveHeight * sinf(self.offset * M_PI * 2 / self.bgview.bounds.size.width);
    CGFloat orignOffY = 0.0;
    CGPathMoveToPoint(pathRef, NULL, 0, startOffY);
    for (CGFloat i = 0.f; i <= self.bgview.bounds.size.width; i++) {
        orignOffY = waveHeight * sinf(2 * M_PI / self.bgview.bounds.size.width * i + self.offset * M_PI * 2 / self.bgview.bounds.size.width) + self.height;
        CGPathAddLineToPoint(pathRef, NULL, i, orignOffY);
    }
    CGPathAddLineToPoint(pathRef, NULL, self.bgview.bounds.size.width, orignOffY);
    CGPathAddLineToPoint(pathRef, NULL, self.bgview.bounds.size.width, self.bgview.bounds.size.height);
    CGPathAddLineToPoint(pathRef, NULL, 0, self.bgview.bounds.size.height);
    CGPathAddLineToPoint(pathRef, NULL, 0, startOffY);
    CGPathCloseSubpath(pathRef);
    self.firstLayer.path = pathRef;
    CGPathRelease(pathRef);
    //第二条波纹路径
    CGMutablePathRef pathRef1 = CGPathCreateMutable();
    CGFloat startOffY1 = waveHeight * sinf(self.offset * M_PI * 2 / self.bgview.bounds.size.width);
    CGFloat orignOffY1 = 0.0;
    CGPathMoveToPoint(pathRef1, NULL, 0, startOffY1);
    for (CGFloat i = 0.f; i <= self.bgview.bounds.size.width; i++) {
        orignOffY1 = waveHeight * cosf(2 * M_PI / self.bgview.bounds.size.width * i + self.offset * M_PI * 2 / self.bgview.bounds.size.width) + self.height;
            CGPathAddLineToPoint(pathRef1, NULL, i, orignOffY1);
    }
    CGPathAddLineToPoint(pathRef1, NULL, self.bgview.bounds.size.width, orignOffY1);
    CGPathAddLineToPoint(pathRef1, NULL, self.bgview.bounds.size.width, self.bgview.bounds.size.height);
    CGPathAddLineToPoint(pathRef1, NULL, 0, self.bgview.bounds.size.height);
    CGPathAddLineToPoint(pathRef1, NULL, 0, startOffY1);
    CGPathCloseSubpath(pathRef1);
    self.secondLayer.path = pathRef1;
    CGPathRelease(pathRef1);
}

对于圆环内的label等控件,使用Masonry进行布局即可,这里就不过多对Masonry进行说明,以上就是根据中国联通手机营业厅首页圆滑进行的仿写,有哪些不足之处请大家提出。

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

推荐阅读更多精彩内容