iOS9 TableView和ScrollView滾動不響應問題


今天開始適配iOS9,發如今有UITableView或是ScrollView的界面滾動不靈敏有些卡頓,研究了下是由於tableview的cell裏有能夠相應點擊事件的Button,當我點擊到Button上滑動的時候,tableview不滾動!!!!什麼狀況這是,後來看看scrollview的官方文檔API裏有這兩個屬性less

@property(nonatomic) BOOL delaysContentTouches;       // default is YES. if NO, we immediately call -touchesShouldBegin:withEvent:inContentView:
@property(nonatomic) BOOL canCancelContentTouches;    // default is YES. if NO, then once we start tracking, we don't try to drag if the touch moves

UIScrollView有一個BOOL類型的tracking屬性,用來返回用戶是否已經觸及內容並打算開始滾動,咱們從這個屬性開始探究UIScrollView的工做原理:

當手指觸摸到UIScrollView內容的一瞬間,會產生下面的動做:post

  • 攔截觸摸事件
  • tracking屬性變爲YESthis

  • 一個內置的計時器開始生效,用來監控在極短的事件間隔內是否發生了手指移動

    case1:當檢測到時間間隔內手指發生了移動,UIScrollView本身觸發滾動,tracking屬性變爲NO,手指觸摸下即便有(能夠響應觸摸事件的)內部控件也不會再響應觸摸事件。
    atom

    case2:當檢測到時間間隔內手指沒有移動,tracking屬性保持YES,手指觸摸下若是有(能夠響應觸摸事件的)內部控件,則將觸摸事件傳遞給控件進行處理。spa

     

有不少新聞類的App頂部都有一個滑動菜單欄,主要模型多是由一個UIScrollView包含多個UIButton控件組成;當你操做的時候,手指若是是很迅速的在上面劃過,會發現即便手指觸摸的地方有UIButton,可是並無觸發該UIButton的任何觸摸事件,這就是上面提到的case1;當你手指是緩慢劃過或根本就沒動,纔會觸發UIButton的觸摸事件,這是case2的狀況。翻譯

上面的工做原理其實有一個屬性開關來控制:delaysContentTouches。默認值爲YES;若是設置爲NO,則不管手指移動的多麼快,始終都會將觸摸事件傳遞給內部控件;設置爲NO可能會影響到UIScrollView的滾動功能。code

再看另外一個BOOL類型的屬性canCencelContentTouches,從字面上理解是「能夠取消內容觸摸「,默認值爲YES。文檔裏的解釋是這樣的:blog

A Boolean value that controls whether touches in the content view always lead to tracking.事件

If the value of this property is YES and a view in the content has begun tracking a finger touching it, and if the user drags the finger enough to initiate a scroll, the view receives a touchesCancelled:withEvent: message and the scroll view handles the touch as a scroll. If the value of this property is NO, the scroll view does not scroll regardless of finger movement once the content view starts tracking.文檔

翻譯爲中文大體以下:

這個BOOL類型的值控制content view裏的觸摸是否總能引起跟蹤(tracking)

若是屬性值爲YES而且跟蹤到手指正觸摸到一個內容控件,這時若是用戶拖動手指的距離足夠產生滾動,那麼內容控件將收到一個touchesCancelled:withEvent:消息,而scroll view將此次觸摸做爲滾動來處理。若是值爲NO,一旦content view開始跟蹤(tracking==YES),則不管手指是否移動,scrollView都不會滾動。

簡單通俗點說,若是爲YES,就會等待用戶下一步動做,若是用戶移動手指到必定距離,就會把這個操做做爲滾動來處理並開始滾動,同時發送一個touchesCancelled:withEvent:消息給內容控件,由控件自行處理。若是爲NO,就不會等待用戶下一步動做,並始終不會觸發scrollView的滾動了。

那我直接在代碼裏都設置爲YES不就OK了,可是設置後仍是不行,後來在發現UIScrollView中有個UIPanGestureRecognizer也有相似的屬性

delaysTouchesBegan,而後也設置成YES

_mTableView.panGestureRecognizer.delaysTouchesBegan = _mTableView.delaysContentTouches;
終於解決這個問題了,這個應該是iOS9的Bug吧,但願Apple後面的版本能修復這個問題
相關文章
相關標籤/搜索