UIScrollView視差滑動輪播圖

iOS仿格瓦拉 輪播圖 Parallax Rolling Banner
作者:yuan

前言

在絕大多數(shù)的APP中架忌,產(chǎn)品經(jīng)理都會要求有一個輪播圖來展示重要的圖片與信息世舰。而大多數(shù)的輪播圖都是比較僵硬的Side By Side的滑動動畫俐芯。如何讓這一個枯燥的UI組件變得有趣蔑穴,并且有絲滑般的感覺呢?我想滾動視差會是一個不錯的選擇泥栖。

如果你經(jīng)常使用格瓦拉,我想你也許就會注意到,格瓦拉的首頁就有著這么一個有意思的視差滾動視圖阴绢。讓我們來嘗試實現(xiàn)它吧。

思考與觀察

在滑動scrollView時艰躺,仔細觀察呻袭,你會發(fā)現(xiàn)在這個視差里面包含了兩組動畫

  • 每當你向右滑動時,中間視圖會跟隨你的手指一起向右移動腺兴,但中間視圖里面的圖片則會朝左邊的方向移動左电。 你向左滑動是則相反。
  • 每當你向右滑動時页响,中間視圖會跟隨你的手指一起向右滑動篓足,而左邊的視圖卻朝向相反的方向移動。你向左滑動是則相反闰蚕,右邊的視圖卻朝向相反的方向移動

這時栈拖,我想你就應該想到,它并不是像大多數(shù)的滾動試圖一樣没陡。不是使用N(你需要顯示的圖片數(shù))+2個視圖撲在UIScrollView上來實現(xiàn), 而是使用了4個主要的視圖:

@property (nonatomic, strong)UIView      * midContainter;
@property (nonatomic, strong)UIImageView * midImage;
@property (nonatomic, strong)UIImageView * leftImage;
@property (nonatomic, strong)UIImageView * rightImage;

其中涩哟,midImage是加載在midContainer上的索赏,以產(chǎn)生第一組動畫。而midContainer贴彼、leftImage潜腻、rightImage這三個視圖有著不同的,層次之間的圖層關系锻弓,中間圖層midContainer總是處于另外兩個圖層的上方砾赔,同時三個圖層的在ScrollView中的位置些許的重疊, 這里我們使用一個portion來統(tǒng)一標識重疊的比例

//中間視圖與它的圖片
_midContainter.frame = CGRectMake(self.bounds.size.width, 0, self.bounds.size.width, self.bounds.size.height);
_midContainter.clipsToBounds = YES;//超出bounds rect的視圖講不會顯示
[self addSubview:_midContainter];

_midImage.frame = self.midContainter.bounds;
[self.midContainter addSubview:_midImage];

//左側視圖
_leftImage.frame = CGRectMake(self.bounds.size.width * (1- self.portion), 0, 0, self.bounds.size.width, self.bounds.size.height);
[self insertSubview:self.leftImage belowSubview:self.midContainter];

//右側視圖
_rightImage.frame = CGRectMake(self.bounds.size.width * (1 + self.portion), 0, self.bounds.size.width, self.bounds.size.height);
[self insertSubview:self.rightImage belowSubview:self.midContainter];

格瓦拉的實際視圖結構:


初始化設置

知道我們需要哪些Views青灼,下面就是對一我們的ScrollView和它的視圖進行初始化的設置了:

//初始化設置
- (void)setup{
    self.contentSize = CGSizeMake(self.bounds.size.width*3, 0);
    self.contentOffset = CGPointMake(self.bounds.size.width, 0);
    self.portion = 0.6f;
    self.pagingEnabled = NO;
    self.showsVerticalScrollIndicator = NO;
    self.showsHorizontalScrollIndicator = NO;
    self.bounces = NO;
    self.layer.masksToBounds = YES;
}

//根據(jù)currentIndex重置srollView為最開始狀態(tài)暴心。
- (void)resetSubViews {
    self.midImage.image = self.sourceArr[self.pageControl.currentPage];
    self.midImage.tag = self.pageControl.currentPage;
    self.midImage.frame = self.midContainter.bounds;
    
    NSInteger leftIndex = self.pageControl.currentPage - 1;
    if (leftIndex < 0) {
        leftIndex = self.sourceArr.count - 1;
    }
    self.leftImage.image = self.sourceArr[leftIndex];
    self.leftImage.tag = leftIndex;
    self.leftImage.frame = CGRectMake(self.bounds.size.width * (1- self.portion), 0, 0, self.bounds.size.width, self.bounds.size.height);
    
    NSInteger rightIndex = self.pageControl.currentPage + 1;
    if (rightIndex >= self.sourceArr.count) {
        rightIndex = 0;
    }
    self.rightImage.image = self.sourceArr[rightIndex];
    self.rightImage.tag = rightIndex;
    self.rightImage.frame = CGRectMake(self.bounds.size.width * (1 +  self.portion), 0, self.bounds.size.width, self.bounds.size.height);
    
    [self bringSubviewToFront:self.midContainter];
    [self sendSubviewToBack:self.leftImage];
    [self sendSubviewToBack:self.rightImage];
    [self setContentOffset:CGPointMake(self.bounds.size.width, 0) animated:NO];    
    self.currentIndex = self.pageControl.currentPage;
}

實現(xiàn)

此外,為了能讓這個視圖循環(huán)滾動杂拨,我們還需要監(jiān)聽滾動時UIScrollViewcontentOffset.x专普。在監(jiān)聽過程中,我們可以根據(jù)self.portion來調(diào)整每個視圖的移動速度弹沽,以此來達到一個滾動視差的效果

- (void)scrollViewDidScroll:(UIScrollView *)scrollView {
    CGFloat moveX = scrollView.contentOffset.x - self.bounds.size.width;
    [self adjsutSubViews:moveX];
    if (fabs(moveX) >= self.bounds.size.width && fabs(self.lastMoveX) < self.bounds.size.width) {
        [self completedHandler];
    }
    self.lastMoveX = moveX;
}

- (void)adjustSubViews:(CGFloat)moveX{
    [self move:self.midImage from:0 byX:moveX * (1 - self.portion)];
    [self move:self.leftImage from:self.bounds.size.width * (1- self.portion) byX:moveX * (1- self.portion)];
    [self move:self.rightImage from:self.bounds.size.width * (1 + self.portion) byX:(moveX) *  (1 - self.portion)];
}

#pragma mark - tools
- (void)move:(UIView *)view from:(CGFloat)start byX:(CGFloat)x {
    CGRect frame = view.frame;
    frame.origin.x = x + start;
    view.frame = frame;
}

這幾行代碼的意義:

  1. 記錄scrollview 相對于初始位置的移動距離moveX
  2. 使leftimagerightImage移動的速率與滑動距離moveX保持一個差值檀夹。
  3. 使midImage與他的父視圖midContainer的移動速率保持不一致。注意我們這里移動的是midContainer里的圖片而不是midContainer
  4. 如果當前的moveX已經(jīng)已經(jīng)是一張圖片的寬度時策橘,調(diào)起completedHandler()
  5. 記錄本次的moveX距離到lastMoveX里炸渡,以方便下一次使用。

由于RunLoop的緣故丽已,ScrollView代理對contentoffset記錄的會非常不精確蚌堵。scrollViewDidScroll()可能會重復調(diào)用completedHandler()。這里記錄lastMoveX是因為我們想確保:當moveX大于一張圖片寬度時沛婴,completedHandler()只被調(diào)起一次吼畏。當lastMoveX已經(jīng)大于一張圖片寬度時,說明completedHandler()已被調(diào)用嘁灯,不需要再重復調(diào)用泻蚊。

completedHandler()里面,我們需要做的是每當一張新圖片被完整顯示在屏幕上時丑婿,不管他是letfImage還是rightImage,我們需要把這張圖片重新賦值到midContainermidImage上面性雄,并根據(jù)這個圖片的index計算出新的leftImagerightImage。同時欺騙用戶,調(diào)用resetSubViews(),把scrollViewoffset重新設置為初始值(顯示中間視圖):

//重新計算letimage, midImage,rightImage的index
- (void)completedHandler{
    CGFloat moveX = self.contentOffset.x - self.bounds.size.width;
    if (fabs(moveX) >= self.bounds.size.width) {
        if (moveX > 0 && self.pageControl.currentPage + 1 < self.sourceArr.count) {
            self.pageControl.currentPage++;
        } else if (moveX >0 && self.pageControl.currentPage +1 == self.sourceArr.count) {
            self.pageControl.currentPage = 0;
        } else if (self.pageControl.currentPage >= 1){
            self.pageControl.currentPage--;
        } else if (self.pageControl.currentPage == 0 && moveX < 0) {
            self.pageControl.currentPage = self.sourceArr.count - 1;
        }
        [self resetSubViews];
    }
}

做完這些羹奉,在設置ScrollView的pagingEnabled屬性為YES

    self.pagingEnabled = YES;

就可以大致完成一個簡單的視差滾動視圖了毅贮。看一下效果:


完整的代碼可以在這里下載尘奏。

但是這是你會發(fā)現(xiàn),當你滑動你的視圖時病蛉,視差滾動視圖并沒有像格瓦拉那樣有如絲般順滑的感覺炫加。格瓦拉到底做了什么呢瑰煎,你可以不妨思考一下?下一篇俗孝,將優(yōu)化滑動動畫酒甸,帶來如絲般的順滑感覺

最后編輯于
?著作權歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市赋铝,隨后出現(xiàn)的幾起案子插勤,更是在濱河造成了極大的恐慌,老刑警劉巖革骨,帶你破解...
    沈念sama閱讀 211,123評論 6 490
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件农尖,死亡現(xiàn)場離奇詭異,居然都是意外死亡良哲,警方通過查閱死者的電腦和手機盛卡,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,031評論 2 384
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來筑凫,“玉大人滑沧,你說我怎么就攤上這事∥∈担” “怎么了滓技?”我有些...
    開封第一講書人閱讀 156,723評論 0 345
  • 文/不壞的土叔 我叫張陵,是天一觀的道長棚潦。 經(jīng)常有香客問我令漂,道長,這世上最難降的妖魔是什么瓦盛? 我笑而不...
    開封第一講書人閱讀 56,357評論 1 283
  • 正文 為了忘掉前任洗显,我火速辦了婚禮,結果婚禮上原环,老公的妹妹穿的比我還像新娘挠唆。我一直安慰自己,他們只是感情好嘱吗,可當我...
    茶點故事閱讀 65,412評論 5 384
  • 文/花漫 我一把揭開白布玄组。 她就那樣靜靜地躺著,像睡著了一般谒麦。 火紅的嫁衣襯著肌膚如雪俄讹。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,760評論 1 289
  • 那天绕德,我揣著相機與錄音患膛,去河邊找鬼。 笑死耻蛇,一個胖子當著我的面吹牛踪蹬,可吹牛的內(nèi)容都是我干的胞此。 我是一名探鬼主播,決...
    沈念sama閱讀 38,904評論 3 405
  • 文/蒼蘭香墨 我猛地睜開眼跃捣,長吁一口氣:“原來是場噩夢啊……” “哼漱牵!你這毒婦竟也來了?” 一聲冷哼從身側響起疚漆,我...
    開封第一講書人閱讀 37,672評論 0 266
  • 序言:老撾萬榮一對情侶失蹤酣胀,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后娶聘,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體闻镶,經(jīng)...
    沈念sama閱讀 44,118評論 1 303
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,456評論 2 325
  • 正文 我和宋清朗相戀三年趴荸,在試婚紗的時候發(fā)現(xiàn)自己被綠了儒溉。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 38,599評論 1 340
  • 序言:一個原本活蹦亂跳的男人離奇死亡发钝,死狀恐怖顿涣,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情酝豪,我是刑警寧澤涛碑,帶...
    沈念sama閱讀 34,264評論 4 328
  • 正文 年R本政府宣布,位于F島的核電站孵淘,受9級特大地震影響蒲障,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜瘫证,卻給世界環(huán)境...
    茶點故事閱讀 39,857評論 3 312
  • 文/蒙蒙 一揉阎、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧背捌,春花似錦毙籽、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,731評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至么抗,卻和暖如春毅否,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背蝇刀。 一陣腳步聲響...
    開封第一講書人閱讀 31,956評論 1 264
  • 我被黑心中介騙來泰國打工螟加, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 46,286評論 2 360
  • 正文 我出身青樓仰迁,卻偏偏與公主長得像甸昏,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子徐许,可洞房花燭夜當晚...
    茶點故事閱讀 43,465評論 2 348

推薦閱讀更多精彩內(nèi)容