自定义带弹性效果的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体验不大好,为此 ...
随机推荐
- oracle 数据库 分割字符串返回结果集函数
CREATE OR REPLACE FUNCTION "UFN_SPLIT" ( p_list varchar2, p_sep varchar2 := ',' ...
- Struts2.3.16.3 基本9个jar包
实践证明,Struts2.3.16.3 至少要下面9个Jar包才能正常启动. commons-fileupload-1.3.1.jar commons-logging-1.1.3.jar freema ...
- hdu 1282 回文数猜想
Problem Description 一个正整数,如果从左向右读(称之为正序数)和从右向左读(称之为倒序数)是一样的,这样的数就叫回文数.任取一个正整数,如果不是回文数,将该数与他的倒序数相加,若其 ...
- c#打开指定设备程序以及网址
//打开计算器 ProcessStartInfo startInfo = new ProcessStartInfo(); startInfo.FileName = @"C:\WINDOWS\ ...
- linux上安装apache以及httpd.conf基本配置
1.yum安装apache #yum install httpd -y 2.随系统自启动 #chkconfig httpd on 3.开启apache #service httpd start PS: ...
- Jasper_crosstab_columngroup header position config - (headerPosition="Stretch")
Position of Totals RowThe totalPosition attribute controls the appearance of the row that displays t ...
- 数组对象-new Array
声明空数组 var arr = new Array(); 声明指定长度的数组 var arr = new Array(5) 声明初始值的数组 var a ...
- IOS开发笔记(4)数据离线缓存与读取
IOS开发笔记(4)数据离线缓存与读取 分类: IOS学习2012-12-06 16:30 7082人阅读 评论(0) 收藏 举报 iosiOSIOS 方法一:一般将服务器第一次返回的数据保存在沙盒里 ...
- 浅谈程序员创业(要有一个自己的网站,最好的方式还是自己定位一个产品,用心把这个产品做好。或者满足不同需求的用户,要有特色)good
浅谈程序员创业 ——作者:邓学彬.Jiesoft 1.什么是创业? 关于“创业”二字有必要重新学习一下,找了两个相对权威定义: 创业就是创业者对自己拥有的资源或通过努力能够拥有的资源进行优化整合,从而 ...
- 《Programming WPF》翻译 第5章 8.我们进行到哪里了?
原文:<Programming WPF>翻译 第5章 8.我们进行到哪里了? 样式支持你定义一个策略来设置可视化元素的依赖属性.属性的设置可以被命名以及手动或者编程方式地通过名称应用,或者 ...