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

首頁 > 系統 > iOS > 正文

IOS實現碎片化動畫詳解

2020-07-26 03:16:30
字體:
來源:轉載
供稿:網友

碎片化效果圖

遮罩視圖

在UIView中有一個maskView屬性,這個屬性是我們今天實現動畫的最重要的變量。這個屬性在iOS8之后開始使用,用來表示視圖的遮罩。什么是遮罩呢?我想了很久都沒有找到合適的比喻來介紹這個。簡單來說,一個UIView的對象,可以通過設置alpha來改變這個視圖的透明度,遮罩的實現效果也是一樣的。唯一的差別在于前者是通過修改0~1之間的值來改變透明效果,作為遮罩的視圖對象的backgroundColor、alpha、transform等等屬性都會影響到被遮蓋的視圖的透明效果。

例如下面這段代碼:

UIView * viewContainer = [[UIView alloc] initWithFrame: CGRectMake(0, 0, 200, 200)];viewContainer.backgroundColor = [UIColor blueColor];UIView * contentView = [[UIView alloc] initWithFrame: CGRectMake(20, 20, 160, 160)];contentView.backgroundColor = [UIColor redColor];[viewContainer addSubview: contentView];UIView * maskView = [[UIView alloc] initWithFrame: CGRectMake(100, 100, 35, 80)];maskView.backgroundColor = [UIColor yellowColor];contentView.maskView = maskView;

遮罩視圖決定了視圖的顯示內容

上面的代碼小小的改動一下,我們分別修改一下maskViewcontentView的透明度,看看在遮罩透明度改變之后紅色的視圖會發生什么變化:


修改透明度.png

通過實驗我們可以看到修改視圖自身的透明度或者修改maskView的透明度達成的效果是一樣的。換句話說,遮蓋視圖對于視圖自身的影響直接決定在透明度和顯示尺寸這兩個可視的屬性。

那么,遮蓋視圖除了alpha屬性外,還有什么屬性影響了視圖本身的顯示效果呢?

顏色

上面的透明度效果得出了一個結論。視圖本身的顯示效果取決于maskView的透明程度。在顏色不含透明空間的時候,視圖是不存在透明效果的。但是假設我們設置遮罩視圖的顏色透明度時:

maskView.backgroundColor = [UIColor colorWithWhite: 1 alpha: 0.5]; //任意顏色

顯示的效果跟直接設置alpha = 0.5的效果是一樣的。在繪制像素到屏幕上中可以獲知顏色渲染和alpha屬性存在的關聯

maskView的子視圖

maskView.backgroundColor = [UIColor clearColor];UIView * sub1 = [[UIView alloc] initWithFrame: CGRectMake(0, 0, 20, 34)];sub1.backgroundColor = [UIColor blackColor];UIView * sub2 = [[UIView alloc] initWithFrame: CGRectMake(15, 18, 33, 40)];sub2.backgroundColor = [UIColor blackColor];[maskView addSubview: sub1];[maskView addSubview: sub2];

要了解maskView的子視圖對遮罩效果的影響,我們需要排除遮罩視圖自身的干擾,因此maskView的背景顏色要設置成透明色


子視圖對于遮罩的影響

可以看到,在遮罩自身透明的情況下,子視圖也可以實現部分遮罩視圖的效果。因此如果我們改變這些子視圖的透明度的時候,遮罩效果也同樣會發生改變

動畫實現

回到上面展示的動畫效果,我們可以看到圖片被分割成多個長方形的小塊逐漸消失。其中,垂直方向分為上下兩份,橫向大概有15份左右。因此我們需要現在maskView上面添加2*15個子視圖,均勻分布。為了保證在動畫的時候我們能依次實現子視圖的隱藏,我們需要給子視圖加上標識:

UIView * maskView = [[UIView alloc] initWithFrame: contentView.bounds];const NSInteger horizontalCount = 15;const NSInteger verticalCount = 2;const CGFloat fadeWidth = CGRectGetWidth(maskView.frame) / horizontalCount;const CGFloat fadeHeight = CGRectGetHeight(maskView.frame) / verticalCount;for (NSInteger line = 0; line < horizontalCount; line ++) {  for (NSInteger row = 0; row < verticalCount; row++) {    CGRect frame = CGRectMake(line*fadeWidth, row*fadeHeight, fadeWidth, fadeHeight);    UIView * fadeView = [[UIView alloc] initWithFrame: frame];    fadeView.tag = [self viewTag: line*verticalCount+row];    fadeView.backgroundColor = [UIColor whiteColor];    [maskView addSubview: fadeView];  }}contentView.maskView = maskView;

那么在動畫開始的時候,我們需要依次遍歷maskView上面的所有子視圖,并且讓他們依次執行動畫:

for (NSInteger line = 0; line < horizontalCount; line ++) {  for (NSInteger row = 0; row < verticalCount; row++) {    NSInteger idx = line*verticalCount+row;    UIView * fadeView = [contentView.maskView viewWithTag: [self viewWithTag: idx];    [UIView animateWithDuration: fadeDuration delay: interval*idx options: UIViewAnimationOptionCurveLinear animations: ^{      fadeView.alpha = 0;    } completion: nil];  }}

我們在實現動畫的同時,都應該考慮如何把動畫封裝出來方便以后復用。上面的碎片化動畫完全可以作為UIViewcategory進行封裝,以此來降低入侵性,實現低耦合的要求:

#define LXDMAXDURATION 1.2#define LXDMINDURATION .2#define LXDMULTIPLED .25@interface UIView (LXDFadeAnimation)/*! * @brief 視圖是否隱藏 */@property (nonatomic, assign, readonly) BOOL isFade;/*! * @brief 是否處在動畫中 */@property (nonatomic, assign, readonly) BOOL isFading;/*! * @brief 垂直方塊個數。默認為3 */@property (nonatomic, assign) NSInteger verticalCount;/*! * @brief 水平方塊個數。默認為18 */@property (nonatomic, assign) NSInteger horizontalCount;/*! * @brief 方塊動畫之間的間隔0.2~1.2。默認0.7 */@property (nonatomic, assign) NSTimeInterval intervalDuration;/*! * @brief 每個方塊隱藏的動畫時間0.05~0.3,最多為動畫時長的25%。默認為0.175 */@property (nonatomic, assign) NSTimeInterval fadeAnimationDuration;- (void)configurateWithVerticalCount: (NSInteger)verticalCount horizontalCount: (NSInteger)horizontalCount interval: (NSTimeInterval)interval duration: (NSTimeInterval)duration;- (void)reverseWithComplete: (void(^)(void))complete;- (void)animateFadeWithComplete: (void(^)(void))complete;- (void)reverseWithoutAnimate;@end

在iOS中,在category中聲明的所有屬性編譯器都不會自動綁定gettersetter方法,這意味著我們需要重寫這兩種方法,而且還不能使用下劃線+變量名的方式直接訪問變量。因此我們需要導入objc/runtime.h文件使用動態時提供的objc_associateObject機制來為視圖動態增加屬性:

- (BOOL)isFade{  return [objc_getAssociatedObject(self, kIsFadeKey) boolValue];}  // other getAssociatedObject method- (void)setIsFade: (BOOL)isFade{  objc_setAssociatedObject(self, kIsFadeKey, @(isFade), OBJC_ASSOCIATION_RETAIN_NONATOMIC);}// other setAssociatedObject method

有了碎片化隱藏視圖的動畫,同樣需要一個還原的動畫效果:

NSInteger fadeCount = self.verticalCount * self.horizontalCount;for (NSInteger idx = fadeCount - 1; idx >= 0; idx--) {  UIView * subview = [self.maskView viewWithTag: [self subViewTag: idx]];  [UIView animateWithDuration: self.fadeAnimationDuration delay: self.intervalDuration * (fadeCount - 1 - idx) options: UIViewAnimationOptionCurveLinear animations: ^{    subview.alpha = 1;  } completion: nil];}

現在我們還要考慮一個問題:假設用戶點擊某張圖片的時候就根據視圖是否隱藏狀態來開始隱藏/顯示的動畫,當用戶多次點擊的時候,我們應該判斷是否已經處在動畫狀態,如果是,那么不繼續執行動畫代碼。另外,在動畫開始之前,我們需要把標識動畫狀態的isFading設為YES,但是由于每個方塊隱藏都存在一個動畫,動畫的結束時間應該怎么判斷呢?已知fadeView的個數是count,那么當最后一個方塊隱藏即是第count個動畫完成的時候,整個碎片化動畫就結束了。所以我們需要借助一個臨時變量來記錄:

__block NSInteger timeCount = 0;//......[UIView animateWithDuration: self.fadeAnimationDuration delay: self.intervalDuration * (fadeCount - 1 - idx) options: UIViewAnimationOptionCurveLinear animations: ^{  subview.alpha = 1;} completion: ^(BOOL finished) {  if (++timeCount == fadeCount) {    self.isFade = NO;    self.isFading = NO;    if (complete) { complete(); }  }}];//......

得到動畫結束的時間后,我們就可以增加一個block提供給調用者在動畫結束時進行其他的處理。

輪播碎片動畫

在知道了碎片動畫的實現之后,我要做一個酷炫的廣告輪播頁。同樣采用category的方式來實現?,F在放上效果圖:


廣告輪播頁

那么實現一個廣告頁輪播需要哪些步驟呢?

    1、在當前動畫的圖片下面插入一個UIImageView來展示下一張圖片。如果可以,盡量復用這個imageView

    2、添加UIPageControl來標識圖片的下標

因此我提供了一個接口傳入圖片數組執行動畫:

// 獲取動態綁定臨時展示的UIImageView- (UIImageView *)associateTempBannerWithImage: (UIImage *)image{  UIImageView * tempBanner = objc_getAssociatedObject(self, kTempImageKey);  if (!tempBanner) {    tempBanner = [[UIImageView alloc] initWithFrame: self.frame];    objc_setAssociatedObject(self, kTempImageKey, tempBanner, OBJC_ASSOCIATION_RETAIN_NONATOMIC);    [self.superview insertSubview: tempBanner belowSubview: self];  }  tempBanner.image = image;  return tempBanner;}

此外,pageControl一開始我加在執行動畫的imageView上面,但是在動畫執行到一半的時候,pageControl也會隨著局部隱藏動畫隱藏起來。因此根據imageView當前的坐標重新計算出合適的尺寸范圍:

- (void)associatePageControlWithCurrentIdx: (NSInteger)idx{  UIPageControl * pageControl = objc_getAssociatedObject(self, kPageControlKey);  if (!pageControl) {    pageControl = [[UIPageControl alloc] initWithFrame: CGRectMake(self.frame.origin.x, CGRectGetHeight(self.frame) - 37 + self.frame.origin.y, CGRectGetWidth(self.frame), 37)];    [self.superview addSubview: pageControl];    pageControl.numberOfPages = self.bannerImages.count;    objc_setAssociatedObject(self, kPageControlKey, pageControl, OBJC_ASSOCIATION_RETAIN_NONATOMIC);  }  pageControl.currentPage = idx;}

由于每次圖片碎片化動畫執行完成之后,都需要再次執行相同的碎片動畫代碼。而動畫結束是通過block執行,即我們需要在block中嵌套使用同一個block,因此首先我們需要把這段執行代碼聲明成一個block變量。另外,需要一個聲明一個idx在每次碎片動畫完成的時候更新圖片,用__block修飾來讓我們在回調中修改這個值:

- (void)fadeBanner  NSParameterAssert(self.superview);  UIImageView * tempBanner = [self associateTempBannerWithImage: [UIImage imageNamed: self.bannerImages[1]]];  self.stop = NO;  __block NSInteger idx = 0;  __weak typeof(self) weakSelf = self;  [self associatePageControlWithCurrentIdx: idx];  void (^complete)() = ^{    NSInteger updateIndex = [weakSelf updateImageWithCurrentIndex: ++idx tempBanner: tempBanner];    idx = updateIndex;    [weakSelf associatePageControlWithCurrentIdx: idx];  };  // 保存block并執行動畫  objc_setAssociatedObject(self, kCompleteBlockKey, complete, OBJC_ASSOCIATION_COPY_NONATOMIC);  [self animateFadeWithComplete: ^{    if (!self.stop) {      complete();    }  }];}// 更新展示的圖片,并且返回下一次要展示的圖片下標- (NSInteger)updateImageWithCurrentIndex: (NSInteger)idx tempBanner: (UIImageView *)tempBanner{  if (idx >= self.bannerImages.count) { idx = 0; }  self.image = [UIImage imageNamed: self.bannerImages[idx]];  [self reverseWithoutAnimate];  NSInteger nextIdx = idx + 1;  if (nextIdx >= self.bannerImages.count) { nextIdx = 0; }  tempBanner.image = [UIImage imageNamed: self.bannerImages[nextIdx]];  [self animateFadeWithComplete: ^{    if (!self.stop) {      void (^complete)() = objc_getAssociatedObject(self, kCompleteBlockKey);      complete();    }  }];  return idx;}

代碼中需要注意的是,我在上面使用objc_Associate的機制保存了這個完成回調的block,這個是必要的。假設你不喜歡把更新圖片的代碼封裝出來,直接把這一步驟放到上面的complete聲明中,依舊還是要動態保存起來,否則這個block執行到第三次圖片碎片的時候就會被釋放從而導致崩潰

別忘了在每次圖片切換完成之后,將所有的子視圖遮罩還原,并且更新圖片顯示

- (void)reverseWithoutAnimate{  if (self.isFading) {    NSLog(@"It's animating!");    return;  }  for (UIView * subview in self.maskView.subviews) {    subview.alpha = 1;  }}

總結

以上就是關于IOS實現碎片化動畫的全部內容,希望本文的內容對大家開發IOS動畫的時候能有所幫助。

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
欧美日韩激情美女| 国产97在线|日韩| 一区二区中文字幕| 色老头一区二区三区| 精品性高朝久久久久久久| 欧美中在线观看| 亚洲缚视频在线观看| 97精品伊人久久久大香线蕉| 日韩欧美国产高清91| 成人欧美一区二区三区黑人| 久久视频国产精品免费视频在线| 亚洲成人国产精品| 一区二区中文字幕| 91在线观看免费观看| 亚洲午夜久久久久久久| 国产精品十八以下禁看| 午夜精品久久久99热福利| 日韩免费黄色av| 久久精品这里热有精品| 久久91亚洲精品中文字幕| 国产成人免费91av在线| 中文字幕亚洲欧美日韩2019| 国产一区二区三区在线观看网站| 日韩电影在线观看永久视频免费网站| 色悠悠国产精品| 日韩在线资源网| 亚洲激情国产精品| 国产成人精品免费视频| 亚洲在线观看视频| 欧美精品在线免费| 色综合久久久888| 日韩av片免费在线观看| 色一情一乱一区二区| 欧美日韩xxx| 欧美精品在线免费播放| 亚洲精品国产精品自产a区红杏吧| 亚洲国产精品电影在线观看| 人人澡人人澡人人看欧美| 欧美性猛交xxxx乱大交极品| 成人性生交大片免费观看嘿嘿视频| 久久久久久久久综合| 国产精品久久久久久中文字| 精品久久久91| 国产91免费观看| 国产一级揄自揄精品视频| 欧美国产中文字幕| 亚洲国产成人精品久久| 亚洲国产精品成人va在线观看| 成人激情免费在线| 久久九九全国免费精品观看| 亚洲第一福利网| 中文字幕精品久久| 亚洲国产小视频| 成人激情在线观看| 日韩av片永久免费网站| 国产日韩精品综合网站| 性日韩欧美在线视频| 亚洲国产免费av| 日韩av色在线| 91精品视频免费看| www日韩中文字幕在线看| 久久大大胆人体| 亚洲国产日韩欧美在线图片| 久久频这里精品99香蕉| 国产婷婷成人久久av免费高清| 福利二区91精品bt7086| 久久久久久香蕉网| 在线观看久久av| 欧美日韩精品二区| 欧美日韩免费看| 国内精品久久久久影院 日本资源| 久久福利视频导航| 日韩av在线最新| 在线观看欧美成人| 日韩精品中文字幕在线播放| 亚洲欧美国产制服动漫| 亚洲成av人乱码色午夜| 欧美激情在线一区| 亚洲欧洲在线免费| 欧美又大粗又爽又黄大片视频| 国产小视频国产精品| 亚洲性视频网站| 日韩在线观看免费网站| 国产精品自拍网| 亚洲精品自拍偷拍| 日韩在线中文视频| 久久久人成影片一区二区三区| 国产成人久久久精品一区| 亚洲精品美女在线观看| 欧美成人高清视频| 欧美一性一乱一交一视频| 夜夜嗨av色综合久久久综合网| 在线观看欧美日韩国产| 欧美一区亚洲一区| 社区色欧美激情 | 国产成人精品在线播放| 亚洲天堂成人在线| 欧美日韩另类字幕中文| 欧美日韩国产精品| 国产一区视频在线播放| 伦理中文字幕亚洲| 热门国产精品亚洲第一区在线| 国产精品91在线| 日韩日本欧美亚洲| 久久精品国产91精品亚洲| 成人97在线观看视频| 欧美视频精品一区| 成人h视频在线| 国产精品美女免费看| 久久97精品久久久久久久不卡| 国语自产偷拍精品视频偷| 91精品在线一区| 777777777亚洲妇女| 国自产精品手机在线观看视频| 啪一啪鲁一鲁2019在线视频| xvideos国产精品| 欧美国产精品人人做人人爱| 国产综合在线看| 日韩精品免费一线在线观看| 精品动漫一区二区| 亚洲国产99精品国自产| 久久影视电视剧凤归四时歌| 欧美多人爱爱视频网站| 亚洲精品福利视频| 97久久久免费福利网址| 97国产精品视频人人做人人爱| 久久久久久香蕉网| 亚洲福利在线看| 欧美一区视频在线| 国产精品视频白浆免费视频| 4438全国亚洲精品在线观看视频| 亚洲日本aⅴ片在线观看香蕉| 久久久久久九九九| 久久青草精品视频免费观看| 亚洲最大成人免费视频| 亚洲欧美精品一区二区| 日韩电影大片中文字幕| 久久亚洲电影天堂| 韩剧1988免费观看全集| 亚洲国产毛片完整版| 久久精品99久久香蕉国产色戒| 精品久久久久久久久国产字幕| 91牛牛免费视频| 国产成人综合一区二区三区| 国产成+人+综合+亚洲欧美丁香花| 北条麻妃一区二区三区中文字幕| 国产精自产拍久久久久久| 亚洲美女喷白浆| 欧美激情中文字幕乱码免费| 久久久久久久久久久久av| 欧美诱惑福利视频| 久久亚洲精品视频| 亚洲一区二区三区乱码aⅴ蜜桃女| 亚洲成人久久网| 国产成人涩涩涩视频在线观看| 亚洲视频在线观看免费| 久久久女女女女999久久| 国产欧美精品一区二区三区介绍| 国产免费一区视频观看免费| 色综合色综合久久综合频道88| 欧美精品福利在线| 国产欧美日韩中文字幕在线| 不卡在线观看电视剧完整版| 国产精品免费视频久久久|