深入解析Scroller滾動(dòng)原理

博文出處:深入解析Scroller滾動(dòng)原理宁舰,歡迎大家關(guān)注我的博客蛮艰,謝謝!

最近在看《Android開發(fā)藝術(shù)探索》這本書壤蚜,不得不贊一句主席寫得真好袜刷,受益匪淺。在書中的相關(guān)章節(jié)有介紹用Scroller來實(shí)現(xiàn)平滑滾動(dòng)的效果著蟹。而我們今天就來探究一下為什么Scroller能夠?qū)崿F(xiàn)平滑滾動(dòng)萧豆。

首先我們先來看一下Scroller的用法,基本可概括為“三部曲”:

  1. 創(chuàng)建一個(gè)Scroller對(duì)象阵面,一般在View的構(gòu)造器中創(chuàng)建:
public ScrollViewGroup(Context context) {
    this(context, null);
}

public ScrollViewGroup(Context context, AttributeSet attrs) {
    this(context, attrs, 0);
}

public ScrollViewGroup(Context context, AttributeSet attrs, int defStyleAttr) {
    super(context, attrs, defStyleAttr);
    mScroller = new Scroller(context);
}
  1. 重寫View的computeScroll()方法洪鸭,下面的代碼基本是不會(huì)變化的:
@Override
public void computeScroll() {
    super.computeScroll();
    if (mScroller.computeScrollOffset()) {
        scrollTo(mScroller.getCurrX(), mScroller.getCurrY());
        postInvalidate();
    }
}
  1. 調(diào)用startScroll()方法览爵,startX和startY為開始滾動(dòng)的坐標(biāo)點(diǎn),dx和dy為對(duì)應(yīng)的偏移量:
mScroller.startScroll (int startX, int startY, int dx, int dy);
invalidate();

上面的三步就是Scroller的基本用法了拾枣。那接下來的任務(wù)就是解析Scroller的滾動(dòng)原理了梅肤。

而在這之前,我們還有一件事要辦俊啼,那就是搞清楚scrollTo()和scrollBy()的原理左医。scrollTo()和scrollBy()的區(qū)別我這里就不重復(fù)敘述了,不懂的可以自行g(shù)oogle或百度跛十。下面貼出scrollTo()的源碼:

public void scrollTo(int x, int y) {
    if (mScrollX != x || mScrollY != y) {
        int oldX = mScrollX;
        int oldY = mScrollY;
        mScrollX = x;
        mScrollY = y;
        invalidateParentCaches();
        onScrollChanged(mScrollX, mScrollY, oldX, oldY);
        if (!awakenScrollBars()) {
            postInvalidateOnAnimation();
        }
    }
}

設(shè)置好mScrollX和mScrollY之后秕硝,調(diào)用了onScrollChanged(mScrollX, mScrollY, oldX, oldY);,View就會(huì)被重新繪制奈偏。這樣就達(dá)到了滑動(dòng)的效果躯护。

下面我們?cè)賮砜纯磗crollBy():

public void scrollBy(int x, int y) {
    scrollTo(mScrollX + x, mScrollY + y);
}

這樣簡(jiǎn)短的代碼相信大家都懂了棺滞,原來scrollBy()內(nèi)部是調(diào)用了scrollTo()的。但是scrollTo()/scrollBy()的滾動(dòng)都是瞬間完成的继准,怎么樣才能實(shí)現(xiàn)平滑滾動(dòng)呢。

不知道大家有沒有這樣一種想法:如果我們把要滾動(dòng)的偏移量分成若干份小的偏移量锰瘸,當(dāng)然這份量要大避凝。然后用scrollTo()/scrollBy()每次都滾動(dòng)小份的偏移量。在一定的時(shí)間內(nèi)管削,不就成了平滑滾動(dòng)了嗎含思?沒錯(cuò)甘晤,Scroller正是借助這一原理來實(shí)現(xiàn)平滑滾動(dòng)的饲做。下面我們就來看看源碼吧盆均!

根據(jù)“三部曲”中第一部,先來看看Scroller的構(gòu)造器:

public Scroller(Context context, Interpolator interpolator, boolean flywheel) {
    mFinished = true;
    if (interpolator == null) {
        mInterpolator = new ViscousFluidInterpolator();
    } else {
        mInterpolator = interpolator;
    }
    mPpi = context.getResources().getDisplayMetrics().density * 160.0f;
    mDeceleration = computeDeceleration(ViewConfiguration.getScrollFriction());
    mFlywheel = flywheel;

    mPhysicalCoeff = computeDeceleration(0.84f); // look and feel tuning
}

在構(gòu)造器中做的主要就是指定了插補(bǔ)器游沿,如果沒有指定插補(bǔ)器肮砾,那么就用默認(rèn)的ViscousFluidInterpolator。

我們?cè)賮砜纯碨croller的startScroll():

public void startScroll(int startX, int startY, int dx, int dy, int duration) {
    mMode = SCROLL_MODE;
    mFinished = false;
    mDuration = duration;
    mStartTime = AnimationUtils.currentAnimationTimeMillis();
    mStartX = startX;
    mStartY = startY;
    mFinalX = startX + dx;
    mFinalY = startY + dy;
    mDeltaX = dx;
    mDeltaY = dy;
    mDurationReciprocal = 1.0f / (float) mDuration;
}

我們發(fā)現(xiàn)眯勾,在startScroll()里面并沒有開始滾動(dòng)咒精,而是設(shè)置了一堆變量的初始值旷档,那么到底是什么讓View開始滾動(dòng)的?我們應(yīng)該把目標(biāo)集中在startScroll()的下一句invalidate();身上范咨。我們可以這樣理解:首先在startScroll()設(shè)置好了一堆初始值厂庇,之后調(diào)用了invalidate();讓View重新繪制,這里又有一個(gè)很重要的點(diǎn)替蛉,在draw()中會(huì)調(diào)用computeScroll()這個(gè)方法拄氯!

源碼太長(zhǎng)了,在這里就不貼出來了镣煮。想看的童鞋在View類里面搜boolean draw(Canvas canvas, ViewGroup parent, long drawingTime)這個(gè)方法就能看到了鄙麦。通過ViewGroup.drawChild()方法就會(huì)調(diào)用子View的draw()方法。而在View類里面的computeScroll()是一個(gè)空的方法介衔,需要我們?nèi)?shí)現(xiàn):

/**
 * Called by a parent to request that a child update its values for mScrollX
 * and mScrollY if necessary. This will typically be done if the child is
 * animating a scroll using a {@link android.widget.Scroller Scroller}
 * object.
 */
public void computeScroll() {
}

而在上面“三部曲”的第二部中夜牡,我們就已經(jīng)實(shí)現(xiàn)了computeScroll()侣签。首先判斷了computeScrollOffset(),我們來看看相關(guān)源碼:

/**
 * Call this when you want to know the new location.  If it returns true,
 * the animation is not yet finished.
 */ 
public boolean computeScrollOffset() {
    if (mFinished) {
        return false;
    }

    int timePassed = (int)(AnimationUtils.currentAnimationTimeMillis() - mStartTime);

    if (timePassed < mDuration) {
        switch (mMode) {
        case SCROLL_MODE:
            final float x = mInterpolator.getInterpolation(timePassed * mDurationReciprocal);
            mCurrX = mStartX + Math.round(x * mDeltaX);
            mCurrY = mStartY + Math.round(x * mDeltaY);
            break;
        case FLING_MODE:
            final float t = (float) timePassed / mDuration;
            final int index = (int) (NB_SAMPLES * t);
            float distanceCoef = 1.f;
            float velocityCoef = 0.f;
            if (index < NB_SAMPLES) {
                final float t_inf = (float) index / NB_SAMPLES;
                final float t_sup = (float) (index + 1) / NB_SAMPLES;
                final float d_inf = SPLINE_POSITION[index];
                final float d_sup = SPLINE_POSITION[index + 1];
                velocityCoef = (d_sup - d_inf) / (t_sup - t_inf);
                distanceCoef = d_inf + (t - t_inf) * velocityCoef;
            }

            mCurrVelocity = velocityCoef * mDistance / mDuration * 1000.0f;
            
            mCurrX = mStartX + Math.round(distanceCoef * (mFinalX - mStartX));
            // Pin to mMinX <= mCurrX <= mMaxX
            mCurrX = Math.min(mCurrX, mMaxX);
            mCurrX = Math.max(mCurrX, mMinX);
            
            mCurrY = mStartY + Math.round(distanceCoef * (mFinalY - mStartY));
            // Pin to mMinY <= mCurrY <= mMaxY
            mCurrY = Math.min(mCurrY, mMaxY);
            mCurrY = Math.max(mCurrY, mMinY);

            if (mCurrX == mFinalX && mCurrY == mFinalY) {
                mFinished = true;
            }

            break;
        }
    }
    else {
        mCurrX = mFinalX;
        mCurrY = mFinalY;
        mFinished = true;
    }
    return true;
}

這個(gè)方法的返回值有講究,若返回true則說明Scroller的滑動(dòng)沒有結(jié)束阴幌;若返回false說明Scroller的滑動(dòng)結(jié)束了卷中。再來看看內(nèi)部的代碼:先是計(jì)算出了已經(jīng)滑動(dòng)的時(shí)間蟆豫,若已經(jīng)滑動(dòng)的時(shí)間小于總滑動(dòng)的時(shí)間,則說明滑動(dòng)沒有結(jié)束十减;不然就說明滑動(dòng)結(jié)束了帮辟,設(shè)置標(biāo)記mFinished = true;。而在滑動(dòng)未結(jié)束里面又分為了兩個(gè)mode由驹,不過這兩個(gè)mode都干了差不多的事,大致就是根據(jù)剛才的時(shí)間timePassed和插補(bǔ)器來計(jì)算出該時(shí)間點(diǎn)滾動(dòng)的距離mCurrXmCurrY并炮。也就是上面“三部曲”中第二部的mScroller.getCurrX(), mScroller.getCurrY()的值润樱。

然后在第二部曲中調(diào)用scrollTo()方法滾動(dòng)到指定點(diǎn)(即上面的mCurrX, mCurrY)。之后又調(diào)用了postInvalidate();嗅钻,讓View重繪并重新調(diào)用computeScroll()以此循環(huán)下去,一直到View滾動(dòng)到指定位置為止养篓,至此Scroller滾動(dòng)結(jié)束柳弄。

其實(shí)Scroller的原理還是比較通俗易懂的。我們?cè)賮砝砬逡幌滤悸废ィ砸粡垐D的形式來終結(jié)今天的Scroller解析:

Scroller的原理圖

好了萍丐,如果有什么問題可以在下面留言。

Goodbye!

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市壳影,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌根灯,老刑警劉巖悠汽,帶你破解...
    沈念sama閱讀 218,607評(píng)論 6 507
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件柿冲,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡怎栽,警方通過查閱死者的電腦和手機(jī)宿饱,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,239評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門谬以,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人邮丰,你說我怎么就攤上這事⊥扪” “怎么了斗蒋?”我有些...
    開封第一講書人閱讀 164,960評(píng)論 0 355
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)捞蚂。 經(jīng)常有香客問我爆哑,道長(zhǎng)舆吮,這世上最難降的妖魔是什么色冀? 我笑而不...
    開封第一講書人閱讀 58,750評(píng)論 1 294
  • 正文 為了忘掉前任,我火速辦了婚禮屯换,結(jié)果婚禮上与学,老公的妹妹穿的比我還像新娘。我一直安慰自己晕窑,他們只是感情好卵佛,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,764評(píng)論 6 392
  • 文/花漫 我一把揭開白布截汪。 她就那樣靜靜地躺著,像睡著了一般阳柔。 火紅的嫁衣襯著肌膚如雪蚓峦。 梳的紋絲不亂的頭發(fā)上医咨,一...
    開封第一講書人閱讀 51,604評(píng)論 1 305
  • 那天拟淮,我揣著相機(jī)與錄音谴忧,去河邊找鬼。 笑死委造,一個(gè)胖子當(dāng)著我的面吹牛均驶,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播爬虱,決...
    沈念sama閱讀 40,347評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼跑筝,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼瞒滴!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起虏两,我...
    開封第一講書人閱讀 39,253評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤世剖,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后引颈,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體境蜕,經(jīng)...
    沈念sama閱讀 45,702評(píng)論 1 315
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡粱年,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,893評(píng)論 3 336
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了赐俗。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片珊燎。...
    茶點(diǎn)故事閱讀 40,015評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖娩梨,靈堂內(nèi)的尸體忽然破棺而出叔扼,到底是詐尸還是另有隱情漫雷,我是刑警寧澤瓜富,帶...
    沈念sama閱讀 35,734評(píng)論 5 346
  • 正文 年R本政府宣布与柑,位于F島的核電站蓄坏,受9級(jí)特大地震影響剑辫,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜妹蔽,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,352評(píng)論 3 330
  • 文/蒙蒙 一胳岂、第九天 我趴在偏房一處隱蔽的房頂上張望舔稀。 院中可真熱鬧,春花似錦产园、人聲如沸夜郁。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,934評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)技俐。三九已至,卻和暖如春雕擂,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背谣拣。 一陣腳步聲響...
    開封第一講書人閱讀 33,052評(píng)論 1 270
  • 我被黑心中介騙來泰國(guó)打工族展, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人贵涵。 一個(gè)月前我還...
    沈念sama閱讀 48,216評(píng)論 3 371
  • 正文 我出身青樓宾茂,卻偏偏與公主長(zhǎng)得像拴还,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子端盆,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,969評(píng)論 2 355

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

  • 內(nèi)容是博主照著書敲出來的焕妙,博主碼字挺辛苦的弓摘,轉(zhuǎn)載請(qǐng)注明出處,后序內(nèi)容陸續(xù)會(huì)碼出末患。 當(dāng)了解了Android坐標(biāo)系和觸...
    Blankj閱讀 6,641評(píng)論 3 61
  • 什么是View View 是 Android 中所有控件的基類。 View的位置參數(shù) View 的位置由它的四個(gè)頂...
    acc8226閱讀 1,177評(píng)論 0 7
  • 一锤窑、Android開發(fā)初體驗(yàn) 二璧针、Android與MVC設(shè)計(jì)模式模型對(duì)象存儲(chǔ)著應(yīng)用的數(shù)據(jù)和業(yè)務(wù)邏輯。模型類通常用來...
    為夢(mèng)想戰(zhàn)斗閱讀 886評(píng)論 0 3
  • 女兒買了房子 果复。我閑著沒事會(huì)經(jīng)常去已完成主體建筑的工地看看陈莽,自封為“建筑質(zhì)量監(jiān)工”。前天天氣好我便又去“...
    一路春風(fēng)撲面來閱讀 300評(píng)論 0 1
  • (一) 谷雨記得也是這樣一個(gè)風(fēng)清月朗的夜晚, 長(zhǎng)滿四葉草和許多夏花的小土坡走搁,薔薇花倦怠著冰涼的花瓣躺在草叢里...
    彌生不知鹿閱讀 325評(píng)論 0 2