自定义带弹性效果的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体验不大好,为此 ...
随机推荐
- LAMP架构搭建+Discuz论坛搭建【weber出品必属精品】
一. 本机简介: 本机系统: CentOS-6.4-x86_64 主机名:oracle.ywb IP地址:192.168.146.129 二. 在Linux环境下安装Apache步骤 ...
- MySql安装与卸载
win2003下MySql的配置 准备相关组件 1.MySql安装包 mysql-installer-commercial- 5.6.14.0.msi 2.Microsoft .NETFramewor ...
- arry()数组的理解及api的使用(一)
我们想要了解数组,首先就要先要了解到什么是数据结构,所谓的数据结构就是把数据与数据见的关系按照特定的结构来保存.设计合理的数据结构是解决问题的前提.了解了数据结构后我们下面来数组的定义:数组(arra ...
- JSP EL
一.JSP EL语言定义 E L(Expression Language) 目的:为了使JSP写起来更加简单. 表达式语言的灵感来自于 ECMAScript 和 XPath 表达式语言,它提供了在 ...
- Onthink学习随笔
-------------------------------------(写代码不孤独_小小代) 用Onthink写了一个网站暂时还没上线功能略显不全没,完全没有发挥出应有的强大拓展之处,各种地方略 ...
- C语言实现的OOP
我倒不是为了OOP而OOP,实在是OOP的一些特征,例如封装,多态其实是软件工程思想,这些思想不分语言,遵循了这些思想可以使得程序更有弹性,更易修改和维护,避免僵化,脆弱 shape.h 该文件定义的 ...
- oracle安装报错2
[oracle@centos1 database]$ ./runInstaller Starting Oracle Universal Installer... Checking installer ...
- JAVA实现实用的ZIP压缩与解压
http://blog.csdn.net/z69183787/article/details/38555913
- POJ 2778 DNA Sequence (AC自动机,矩阵乘法)
题意:给定n个不能出现的模式串,给定一个长度m,要求长度为m的合法串有多少种. 思路:用AC自动机,利用AC自动机上的节点做矩阵乘法. #include<iostream> #includ ...
- LeetCode_Word Ladder
Given two words (start and end), and a dictionary, find the length of shortest transformation sequen ...