亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 系統 > iOS > 正文

iOS如何自定義控制器轉場動畫push詳解

2020-07-26 02:34:27
字體:
來源:轉載
供稿:網友

前言

最近有些空閑時間,整理了下最近做的項目,本文主要介紹了關于iOS自定義控制器轉場動畫push的相關內容,分享出來供大家參考學習,下面話不多說了,來一起看看詳細的介紹吧。

效果圖:


iOS7 開始蘋果推出了自定義轉場的 API 。從此,任何可以用 CoreAnimation 實現的動畫,都可以出現在兩個 ViewController 的切換之間。并且實現方式高度解耦,這也意味著在保證代碼干凈的同時想要替換其他動畫方案時只需簡單改一個類名就可以了,真正體會了一把高顏值代碼帶來的愉悅感。

其實網上關于自定義轉場動畫的教程很多,這里我是希望同學們能易懂,易上手。

轉場分兩種Push和Modal,所以自定義轉場動畫也就肯定分兩種,今天我們講的是Push

自定義轉場動畫Push

首先搭建界面,添加4個按鈕:

- (void)addButton{   self.buttonArr = [NSMutableArray array];   CGFloat margin=50; CGFloat width=(self.view.frame.size.width-margin*3)/2; CGFloat height = width; CGFloat x = 0; CGFloat y = 0; //列 NSInteger col = 2; for (NSInteger i = 0; i < 4; i ++) {     x = margin + (i%col)*(margin+width);  y = margin + (i/col)*(margin+height) + 150;     UIButton *button = [UIButton buttonWithType:UIButtonTypeCustom];  button.frame = CGRectMake(x, y, width, height);  button.layer.cornerRadius = width * 0.5;  [button addTarget:self action:@selector(btnclick:) forControlEvents:UIControlEventTouchUpInside];  button.backgroundColor = [UIColor colorWithRed:arc4random()%255/255.0 green:arc4random()%255/255.0 blue:arc4random()%255/255.0 alpha:1.0];  button.tag = i+1;  [self.view addSubview:button];  [self.buttonArr addObject:button]; }}

添加動畫:

- (void)setupButtonAnimation{   [self.buttonArr enumerateObjectsUsingBlock:^(UIButton * _Nonnull button, NSUInteger idx, BOOL * _Nonnull stop) {     // positionAnimation  CAKeyframeAnimation *positionAnimation = [CAKeyframeAnimation animationWithKeyPath:@"position"];  positionAnimation.calculationMode = kCAAnimationPaced;  positionAnimation.fillMode = kCAFillModeForwards;  positionAnimation.repeatCount = MAXFLOAT;  positionAnimation.autoreverses = YES;  positionAnimation.timingFunction = [CAMediaTimingFunction functionWithName:kCAMediaTimingFunctionEaseInEaseOut];  positionAnimation.duration = (idx == self.buttonArr.count - 1) ? 4 : 5+idx;  UIBezierPath *positionPath = [UIBezierPath bezierPathWithOvalInRect:CGRectInset(button.frame, button.frame.size.width/2-5, button.frame.size.height/2-5)];  positionAnimation.path = positionPath.CGPath;  [button.layer addAnimation:positionAnimation forKey:nil];     // scaleXAniamtion  CAKeyframeAnimation *scaleXAniamtion = [CAKeyframeAnimation animationWithKeyPath:@"transform.scale.x"];  scaleXAniamtion.values = @[@1.0,@1.1,@1.0];  scaleXAniamtion.keyTimes = @[@0.0,@0.5,@1.0];  scaleXAniamtion.repeatCount = MAXFLOAT;  scaleXAniamtion.autoreverses = YES;  scaleXAniamtion.duration = 4+idx;  [button.layer addAnimation:scaleXAniamtion forKey:nil];     // scaleYAniamtion  CAKeyframeAnimation *scaleYAnimation = [CAKeyframeAnimation animationWithKeyPath:@"transform.scale.y"];  scaleYAnimation.values = @[@1,@1.1,@1.0];  scaleYAnimation.keyTimes = @[@0.0,@0.5,@1.0];  scaleYAnimation.autoreverses = YES;  scaleYAnimation.repeatCount = YES;  scaleYAnimation.duration = 4+idx;  [button.layer addAnimation:scaleYAnimation forKey:nil];    }];}

界面搭建好了:

然后想在Push的時候實現自定義轉場動畫首先要遵守一個協議UINavigationControllerDelegate

蘋果在 UINavigationControllerDelegate 中給出了幾個協議方法,通過返回類型就可以很清楚地知道各自的具體作用。

//用來自定義轉場動畫- (nullable id)navigationController:(UINavigationController *)navigationController         animationControllerForOperation:(UINavigationControllerOperation)operation            fromViewController:(UIViewController *)fromVC             toViewController:(UIViewController *)toVC NS_AVAILABLE_IOS(7_0);
//為這個動畫添加用戶交互- (nullable id)navigationController:(UINavigationController *)navigationController       interactionControllerForAnimationController:(id) animationController NS_AVAILABLE_IOS(7_0);

在第一個方法里只要返回一個準守UIViewControllerInteractiveTransitioning協議的對象,并在里面實現動畫即可

  • 創建繼承自 NSObject 并且聲明 UIViewControllerAnimatedTransitioning 的的動畫類。
  • 重載 UIViewControllerAnimatedTransitioning 中的協議方法。
//返回動畫時間- (NSTimeInterval)transitionDuration:(nullable id)transitionContext;//將動畫的代碼寫到里面即可- (void)animateTransition:(id)transitionContext;

首先我自定義一個名為LRTransitionPushController的類繼承于NSObject準守了UIViewControllerAnimatedTransitioning協議

 - (void)animateTransition:(id)transitionContext{   self.transitionContext = transitionContext;   //獲取源控制器 注意不要寫成 UITransitionContextFromViewKey LRTransitionPushController *fromVc = [transitionContext viewControllerForKey:UITransitionContextFromViewControllerKey]; //獲取目標控制器 注意不要寫成 UITransitionContextToViewKey LRTransitionPopController *toVc = [transitionContext viewControllerForKey:UITransitionContextToViewControllerKey];   //獲得容器視圖 UIView *containView = [transitionContext containerView]; // 都添加到container中。注意順序 目標控制器的view需要后面添加 [containView addSubview:fromVc.view]; [containView addSubview:toVc.view];   UIButton *button = fromVc.button; //繪制圓形 UIBezierPath *startPath = [UIBezierPath bezierPathWithOvalInRect:button.frame];   //創建兩個圓形的 UIBezierPath 實例;一個是 button 的 size ,另外一個則擁有足夠覆蓋屏幕的半徑。最終的動畫則是在這兩個貝塞爾路徑之間進行的 //按鈕中心離屏幕最遠的那個角的點 CGPoint finalPoint; //判斷觸發點在那個象限 if(button.frame.origin.x > (toVc.view.bounds.size.width / 2)){  if (button.frame.origin.y < (toVc.view.bounds.size.height / 2)) {   //第一象限   finalPoint = CGPointMake(0, CGRectGetMaxY(toVc.view.frame));  }else{   //第四象限   finalPoint = CGPointMake(0, 0);  } }else{  if (button.frame.origin.y < (toVc.view.bounds.size.height / 2)) {   //第二象限   finalPoint = CGPointMake(CGRectGetMaxX(toVc.view.frame), CGRectGetMaxY(toVc.view.frame));  }else{   //第三象限   finalPoint = CGPointMake(CGRectGetMaxX(toVc.view.frame), 0);  } }  CGPoint startPoint = CGPointMake(button.center.x, button.center.y); //計算向外擴散的半徑 = 按鈕中心離屏幕最遠的那個角距離 - 按鈕半徑 CGFloat radius = sqrt((finalPoint.x-startPoint.x) * (finalPoint.x-startPoint.x) + (finalPoint.y-startPoint.y) * (finalPoint.y-startPoint.y)) - sqrt(button.frame.size.width/2 * button.frame.size.width/2 + button.frame.size.height/2 * button.frame.size.height/2); UIBezierPath *endPath = [UIBezierPath bezierPathWithOvalInRect:CGRectInset(button.frame, -radius, -radius)];   //賦值給toVc視圖layer的mask CAShapeLayer *maskLayer = [CAShapeLayer layer]; maskLayer.path = endPath.CGPath; toVc.view.layer.mask = maskLayer;   CABasicAnimation *maskAnimation =[CABasicAnimation animationWithKeyPath:@"path"]; maskAnimation.fromValue = (__bridge id)startPath.CGPath; maskAnimation.toValue = (__bridge id)endPath.CGPath; maskAnimation.duration = [self transitionDuration:transitionContext]; maskAnimation.timingFunction = [CAMediaTimingFunction functionWithName:kCAMediaTimingFunctionEaseInEaseOut]; maskAnimation.delegate = self; [maskLayer addAnimation:maskAnimation forKey:@"path"]; }

在控制器里面用來自定義轉場動畫的方法里返回剛才自定義的動畫類

- (id)navigationController:(UINavigationController *)navigationController animationControllerForOperation:(UINavigationControllerOperation)operation fromViewController:(UIViewController *)fromVC toViewController:(UIViewController *)toVC{   if (operation == UINavigationControllerOperationPush) {  return [LRTranstionAnimationPush new]; }else{  return nil; }}

到此為止自定義轉場動畫就完成了

pop的動畫只是把push動畫反過來做一遍這里就不細講了,有疑問的可以去看代碼

添加滑動返回手勢

上面說到這個方法是為這個動畫添加用戶交互的所以我們要在pop時實現滑動返回

最簡單的方式應該就是利用UIKit提供的UIPercentDrivenInteractiveTransition類了,這個類已經實現了UIViewControllerInteractiveTransitioning協議,同學men可以通過這個類的對象指定轉場動畫的完成百分比。

//為這個動畫添加用戶交互- (nullable id)navigationController:(UINavigationController *)navigationController       interactionControllerForAnimationController:(id) animationController NS_AVAILABLE_IOS(7_0);

第一步 添加手勢

 UIPanGestureRecognizer *gestureRecognizer = [[UIPanGestureRecognizer alloc] initWithTarget:self action:@selector(handlePan:)]; [self.view addGestureRecognizer:gestureRecognizer];

第二步 通過用戶滑動的變化確定動畫執行的比例

- (void)handlePan:(UIPanGestureRecognizer *)gestureRecognizer {   /*調用UIPercentDrivenInteractiveTransition的updateInteractiveTransition:方法可以控制轉場動畫進行到哪了,  當用戶的下拉手勢完成時,調用finishInteractiveTransition或者cancelInteractiveTransition,UIKit會自動執行剩下的一半動畫,  或者讓動畫回到最開始的狀態。*/    if ([gestureRecognizer translationInView:self.view].x>=0) {  //手勢滑動的比例  CGFloat per = [gestureRecognizer translationInView:self.view].x / (self.view.bounds.size.width);  per = MIN(1.0,(MAX(0.0, per)));     if (gestureRecognizer.state == UIGestureRecognizerStateBegan) {       self.interactiveTransition = [UIPercentDrivenInteractiveTransition new];   [self.navigationController popViewControllerAnimated:YES];      } else if (gestureRecognizer.state == UIGestureRecognizerStateChanged){       if([gestureRecognizer translationInView:self.view].x ==0){         [self.interactiveTransition updateInteractiveTransition:0.01];        }else{         [self.interactiveTransition updateInteractiveTransition:per];   }      } else if (gestureRecognizer.state == UIGestureRecognizerStateEnded || gestureRecognizer.state == UIGestureRecognizerStateCancelled){       if([gestureRecognizer translationInView:self.view].x == 0){         [self.interactiveTransition cancelInteractiveTransition];    self.interactiveTransition = nil;        }else if (per > 0.5) {         [ self.interactiveTransition finishInteractiveTransition];   }else{         [ self.interactiveTransition cancelInteractiveTransition];   }   self.interactiveTransition = nil;  }      } else if (gestureRecognizer.state == UIGestureRecognizerStateChanged){  [self.interactiveTransition updateInteractiveTransition:0.01];  [self.interactiveTransition cancelInteractiveTransition];  } else if ((gestureRecognizer.state == UIGestureRecognizerStateEnded || gestureRecognizer.state == UIGestureRecognizerStateCancelled)){     self.interactiveTransition = nil; }  }

第三步 在為動畫添加用戶交互的代理方法里返回UIPercentDrivenInteractiveTransition的實例

- (id)navigationController:(UINavigationController *)navigationController       interactionControllerForAnimationController:(id) animationController { return self.interactiveTransition;}

如果感覺這篇文章對您有所幫助,順手點個喜歡,謝謝啦

代碼放在了GitHub上大家可以下載,當然也可以通過本地下載

總結

以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作具有一定的參考學習價值,如果有疑問大家可以留言交流,謝謝大家對武林網的支持。

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
一区二区三区日韩在线| 久久精品亚洲国产| 欧美大片欧美激情性色a∨久久| 福利二区91精品bt7086| 国a精品视频大全| 欧美国产第二页| 亚洲国产精品成人一区二区| 中文字幕国内精品| 久久久伊人日本| 国产精品青草久久久久福利99| 久久成人av网站| 68精品久久久久久欧美| 欧美麻豆久久久久久中文| 亚洲精品国产美女| 国产精品久久久久久久av大片| 91精品国产91| 日韩**中文字幕毛片| 亚洲无av在线中文字幕| 国产女人精品视频| 国产精品久久久久久影视| 亚洲女人被黑人巨大进入al| 欧美成人精品一区二区三区| 精品国产一区二区三区久久狼5月| 亚洲精品第一国产综合精品| 欧美大人香蕉在线| 欧美日韩加勒比精品一区| 懂色av影视一区二区三区| 国产欧美精品一区二区三区介绍| 中文字幕在线看视频国产欧美| 欧洲美女免费图片一区| 岛国视频午夜一区免费在线观看| 欧美在线免费观看| 欧美日韩第一视频| 国内精品一区二区三区四区| 国内精品视频一区| 久久久久久久久久久免费精品| 日韩经典第一页| 国产精品一区二区女厕厕| 国产91精品久久久久| 97精品视频在线播放| 欧美一性一乱一交一视频| 亚洲一区久久久| 国产精品旅馆在线| 亚洲美女又黄又爽在线观看| 疯狂做受xxxx高潮欧美日本| 国产视频一区在线| 国产精品电影观看| 亚洲乱码国产乱码精品精| 亚洲护士老师的毛茸茸最新章节| 国产不卡视频在线| 最近2019免费中文字幕视频三| 91精品久久久久久久久久久久久久| 国产剧情久久久久久| 亚洲理论电影网| 日韩欧美亚洲范冰冰与中字| 日韩久久午夜影院| 国产精品极品尤物在线观看| 美日韩精品视频免费看| 91精品国产自产在线| 国产欧美在线看| 亚洲女人天堂网| 国产在线一区二区三区| 欧美性猛交xxxx偷拍洗澡| 超碰91人人草人人干| 亚洲欧美国产日韩中文字幕| 麻豆国产va免费精品高清在线| 欧洲日本亚洲国产区| 亚洲精品99999| 欧洲精品毛片网站| 欧美日韩国产丝袜另类| 亚洲xxxx在线| 亚洲美女免费精品视频在线观看| 奇米影视亚洲狠狠色| 亚洲国产精品成人一区二区| 97久久精品视频| 91亚洲精品视频| 91丨九色丨国产在线| 亚洲成人性视频| 在线不卡国产精品| 成人激情春色网| 日韩中文字幕免费看| 国产综合视频在线观看| 91系列在线播放| 欧美高清视频在线观看| 亚洲成色777777女色窝| 精品视频偷偷看在线观看| 国产精品久久久久久久久久| 国产精品综合不卡av| 精品视频在线观看日韩| 日韩一区二区三区国产| 欧美大片大片在线播放| 国产91露脸中文字幕在线| 日韩精品中文字幕在线播放| 国产精品国产福利国产秒拍| 91干在线观看| 日韩欧美成人免费视频| 日韩视频在线观看免费| 精品小视频在线| 欧美日韩免费网站| 伦伦影院午夜日韩欧美限制| 日韩在线视频线视频免费网站| 九九热r在线视频精品| 国产精品视频精品| 美日韩精品免费视频| 尤物九九久久国产精品的分类| 久久这里只有精品视频首页| 中文字幕国产日韩| 丰满岳妇乱一区二区三区| 欧美理论电影在线观看| 欧美成人精品一区| 国产欧美日韩精品在线观看| 日韩av电影手机在线观看| 成人网页在线免费观看| 亚洲网站视频福利| 少妇激情综合网| 欧美黑人极品猛少妇色xxxxx| 亚洲欧洲激情在线| 国产一区二区三区毛片| 影音先锋欧美在线资源| 亚洲电影免费观看高清| 91精品国产91久久| 久久av红桃一区二区小说| 美日韩精品免费观看视频| 成人黄色片在线| 欧美在线视频观看免费网站| 亚洲综合在线中文字幕| 国产精品稀缺呦系列在线| 国产精品扒开腿做爽爽爽男男| 国产精品久久久91| 91啪国产在线| 亚洲综合自拍一区| 亚洲天堂av在线免费| 国产99视频在线观看| 亚洲www在线观看| 亚洲欧美综合另类中字| 一区二区日韩精品| 疯狂欧美牲乱大交777| 国产成人精品在线| 日本中文字幕久久看| 久久精品中文字幕一区| 欧美大人香蕉在线| 综合国产在线视频| 欧美精品激情视频| 亚洲综合在线中文字幕| 91精品在线观看视频| 在线成人一区二区| 伊人男人综合视频网| 日韩av免费在线看| 国产精品白嫩美女在线观看| 亚洲自拍欧美色图| 国产精品在线看| 亚洲成人黄色网址| 日本精品久久久久久久| 97视频在线播放| 欧美日在线观看| 黑人巨大精品欧美一区免费视频| 亚洲一区国产精品| 亚洲精品自在久久| 亚洲一区www| 国产精品美女免费| 久久精品国产欧美亚洲人人爽| 日韩免费精品视频| 国产精品自产拍在线观看中文| 91久久国产精品|