自定义带弹性效果的pageControl

分三部分实现,在drawrect方法里画出灰色背景,根据pageCount创建对应个数的dotView放置在对应位置,并隐藏,创建一个CAShapeView类型的layer,根据scrollView的偏移量 构建贝塞尔曲线,画出红色线条,以及形变的大圆。
大圆的形变以及构建思路:
(图引用自:http://kittenyang.com/deformationandgooey/)
下面就是构建A、B、C、D、c1、c2、c3、c4、c5、c6、c7、c8这些点,给A、B、C、D、四个点,一个关于contentOffset的形变量,以达到园形变的效果。value = (半径 * 比例)*形变因素,其中形变因素于contentOffset相关(在滑动一个页面时会形成这样的关系:0~0.5~0);CGFloat factor = MIN(0.5, MAX(0, (ABS(scrollView.contentOffset.x - last) / scrollView.frame.size.width)));,构建好点后,根据偏移量,更新点的坐标,生成新的贝塞尔曲线,便生成形变的 圆了。
下面是部分代码:
/** 根据scrollView 更新贝塞尔曲线 */
- (void)updateDotLayerScrollView:(UIScrollView *)scrollView{
// 判断滑动方向
BOOL left = (scrollView.contentOffset.x - self.lastOffsetX) >= ?YES:NO; // 向左滑动就显示dotView
if (left) {
[self setDotViewShowWithScrollView:scrollView];
} CGPoint dotCenter = [self calculateDotCenterWithScrollView:scrollView]; // 计算当前偏移量的 归属地(超过一半 就归属后者,反之前者)
int count = (int)(scrollView.contentOffset.x / scrollView.frame.size.width+0.5); // 根据count 计算归属地 的偏移量
CGFloat last = scrollView.frame.size.width * count; CGFloat factor = MIN(0.5, MAX(, (ABS(scrollView.contentOffset.x - last) / scrollView.frame.size.width))); // extra 是 A、B、C、D四点的 位移量,跟factor有关。而factor跟contentOffset相关,其关系为0~0.5~0;
CGFloat extra = self.SeletedDotRadiu * / * factor; // 构成贝塞尔曲线的相关点 的计算
A = CGPointMake(dotCenter.x, dotCenter.y - self.SeletedDotRadiu+extra);
B = CGPointMake(left?(dotCenter.x+self.SeletedDotRadiu):(dotCenter.x+self.SeletedDotRadiu+extra * ), dotCenter.y);
C = CGPointMake(dotCenter.x, dotCenter.y+self.SeletedDotRadiu-extra);
D = CGPointMake(left?(dotCenter.x-self.SeletedDotRadiu-extra*):(dotCenter.x-self.SeletedDotRadiu), dotCenter.y);
// 1.8 的得来应该是能计算出来的,不过不知道咋算,试出来
CGFloat offset = self.SeletedDotRadiu / 1.8; c1 = CGPointMake(A.x + offset, A.y);
c2 = CGPointMake(B.x, B.y - offset);
c3 = CGPointMake(B.x, B.y + offset);
c4 = CGPointMake(C.x + offset, C.y);
c5 = CGPointMake(C.x - offset, C.y);
c6 = CGPointMake(D.x, D.y + offset);
c7 = CGPointMake(D.x, D.y - offset);
c8 = CGPointMake(A.x - offset, A.y); UIBezierPath* ovalPath = [UIBezierPath bezierPath];
CGPoint startPoint = CGPointMake(self.SeletedDotRadiu, CGRectGetHeight(self.frame) / 2.0);
[ovalPath moveToPoint: startPoint];
[ovalPath addLineToPoint:D];
[ovalPath addCurveToPoint:C controlPoint1:c6 controlPoint2:c5];
[ovalPath addCurveToPoint:B controlPoint1:c4 controlPoint2:c3];
[ovalPath addCurveToPoint:A controlPoint1:c2 controlPoint2:c1];
[ovalPath addCurveToPoint:D controlPoint1:c8 controlPoint2:c7]; [ovalPath closePath];
self.dotLayer.path = ovalPath.CGPath; self.lastOffsetX = scrollView.contentOffset.x;
}
drawRect 方法:
/** 画初始 的背景 */
- (void)drawRect:(CGRect)rect { // 控件垂直方向中心
CGFloat verticalCenter = rect.size.height / 2.0;
// 圆点 半径
CGFloat dotR = self.dotRadiu; // 选中圆点 半径
CGFloat selectedDotR = self.SeletedDotRadiu;
// 圆心距离
CGFloat distanceCenter = (rect.size.width - * selectedDotR) / (self.pageCount - ); UIBezierPath *path = [UIBezierPath bezierPathWithRect:CGRectMake(selectedDotR, (rect.size.height - self.lineWidth) / 2.0, rect.size.width - * selectedDotR, self.lineWidth)]; [self.color setFill]; for (int i = ; i < self.pageCount; i++) { UIBezierPath *dotPath = [UIBezierPath bezierPathWithArcCenter:CGPointMake(selectedDotR + distanceCenter * i, verticalCenter) radius:dotR startAngle: endAngle:M_PI * clockwise:YES];
[path appendPath:dotPath];
}
[path fill]; }
完整代码:http://pan.baidu.com/s/1i4y78gt
自定义带弹性效果的pageControl的更多相关文章
- [转]Android UI:看看Google官方自定义带旋转动画的ImageView-----RotateImageView怎么写(附 图片淡入淡出效果)
http://blog.csdn.net/yanzi1225627/article/details/22439119 众所周知,想要让ImageView旋转的话,可以用setRotation()让其围 ...
- android标题栏上面弹出提示框(二) PopupWindow实现,带动画效果
需求:上次用TextView写了一个从标题栏下面弹出的提示框.android标题栏下面弹出提示框(一) TextView实现,带动画效果, 总在找事情做的产品经理又提出了奇葩的需求.之前在通知栏显示 ...
- android标题栏下面弹出提示框(一) TextView实现,带动画效果
产品经理用的是ios手机,于是android就走上了模仿的道路.做这个东西也走了一些弯路,写一篇博客放在这里,以后自己也可用参考,也方便别人学习. 弯路: 1.刚开始本来用PopupWindow去实现 ...
- Android实现自定义带文字和图片的Button
Android实现自定义带文字和图片的Button 在Android开发中经常会需要用到带文字和图片的button,下面来讲解一下常用的实现办法. 一.用系统自带的Button实现 最简单的一种办法就 ...
- 自定义带动画的Toast
一.style样式: 1. // 移动和透明渐变结合的动画 <style name="anim_view"> <item name="@ ...
- js+css实现带缓冲效果右键弹出菜单
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/ ...
- 自定义带图标input样式
<!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <title> ...
- Android带弹性的View
在Android开发中ListView.ScrollView用到的频率相当高,可是一个优秀的应用我们能看到它里面的效果绝对不会那么死板,安卓原生的ListView和ScrollView都不能满足这个要 ...
- Android UI:看看Google官方自定义带旋转动画的ImageView-----RotateImageView怎么写(附 图片淡入淡...)
众所周知,想要让ImageView旋转的话,可以用setRotation()让其围绕中心点旋转,但这个旋转是不带动画的,也就是旋转屏幕时图片噌的一下就转过去了,看不到旋转的过程,此UI体验不大好,为此 ...
随机推荐
- Eclipse MAT: Understand Incoming and Outgoing References
引用:http://xmlandmore.blogspot.hk/2014/01/eclipse-mat-understand-incoming-and.html?utm_source=tuicool ...
- python中raw_input()与input()
raw_input([prompt]) input([prompt]) # prompt:如果参数存在,直接输出到屏幕上,不会再另起一行 raw_input 如其字面意思一样,返回输入字符的字符串形式 ...
- 【3】创建一个简单的Laravel例子
现在我们来创建一个Laravel的例子来帮助理解 1.首先打开app/Http/routes.php文件,在里边写上一条路由: 2.创建一个控制器,有两种方法 ①在app/Http/Controlle ...
- 【0】Laravel 5.1 简介
1.简介 Laravel是一套简洁.优雅的PHP Web开发框架(PHP Web Framework).它可以让你从面条一样杂乱的代码中解脱出来:它可以帮你构建一个完美的网络APP,而且每行代码都可以 ...
- forward 和redirect的区别
1.从地址栏显示来说 forward是服务器请求资源,服务器直接访问目标地址的URL,把那个URL的响应内容读取过来,然后把这些内容再发给浏览器.浏览器根本不知道服务器发送的内容从哪里来的,所以它的地 ...
- 怎样利用putty登陆SSH主机方法
PuTTY 是一套免费的SSH / Telnet 程序,是在Windows 32平台下的telnet.rlogin和ssh客户端,它是一个跨平台的远程登录工具 下载putty成功后,双击打开Putty ...
- 163k地方门户网站系统团购定时结束限量控制
#coding=utf8 #!/usr/bin/env python # 网站自动审核系统 import pymssql import re import sys import datetime im ...
- adnroid 监听收到的短信并根据短信内容进行回复短信
定义一个广播接收器 public class SMSReceiver extends BroadcastReceiver { private SmsManager smsManager; @Overr ...
- Android使用xml中定义的动画效果
Animation animation = AnimationUtils.loadAnimation(getActivity(), R.anim.zqrl_out); animation.setFil ...
- github在eclipse中的配置
http://www.cnblogs.com/yejiurui/archive/2013/07/29/3223153.html http://blog.csdn.net/shehun1/article ...