Android 動畫框架詳解,第 2 部分

Android launcher 的平滑和立體翻頁效果

咱們這裏把 Android launcher 程序的 Workspace 相關的代碼抽取出來,以一個比較簡單的代碼來展現 launcher 程序是如何實現多頁以及不一樣頁面之間的切換效果。本示例代碼在 SDK 2.1 中運行,設置的是 WVGA 的屏幕大小。android

首先咱們來看一下程序運行的效果來一些感性的認識。canvas

圖 1:平滑移動效果函數

圖 1:平滑移動效果

圖 2:立體翻頁效果佈局

圖 2:立體翻頁效果

窗口頁面的佈局

接着咱們來看一下程序 UI(即 View 和 ViewGroup)的佈局,Activity 的 ContentView 是 layout 中的 main.xml。它的內容以下:post

清單 1.this

清單 1.

其中 FlatWorkspace 的基類是 Workspace,它繼承自 ViewGroup,是一個容器類,其中包含三個子 View,子 View 是 ImageView。三個 ImageView 就是三個頁面。這三個 ImageView 的建立是在 WorkspaceActivity 的 onCreate 函數中調用 Workspace 的 initScreens 函數完成的,代碼以下:spa

清單 2rest

ViewGroup.LayoutParams p = new iewGroup.LayoutParams(ViewGroup.LayoutParams.FILL_PARENT,ViewGroup.LayoutParams.FILL_PARENT); 	
        	 for (int i = 0; i < 3; i++) { 
			 this.addView(new ImageView(this.getContext()), i, p); 
		 } 
		 ((ImageView)this.getChildAt(0)).setImageResource(R.drawable.image_search); 
		 ((ImageView)this.getChildAt(1)).setImageResource(R.drawable.image_system); 
		 ((ImageView)this.getChildAt(2)).setImageResource(R.drawable.image_top);

圖 3:Workspace 和頁面佈局圖xml

圖 3:Workspace 和頁面佈局圖

爲了讓三個頁面達到上圖的窗口布局,咱們對 Workspace 的 onMeasure 和 onLayout 函數進行了重載,重點在 onLayout 代碼中。onLayout 函數調用 layoutScreens 函數完成佈局,FlatWorkspace 中的 layoutScreens 實現以下:繼承

清單 3

protected void layoutScreens() { 
        int childLeft = 0; 
        final int count = getChildCount(); 
        for (int i = 0; i < count; i++) { 
            final View child = getChildAt(i); 
            if (child.getVisibility() != View.GONE) { 
                final int childWidth = child.getMeasuredWidth(); 
                child.layout(childLeft, 0, childLeft + childWidth, child.getMeasuredHeight()); 
                childLeft += childWidth; 
            } 
        } 
 }

上面 child.layout 部分的代碼把三個頁面分別佈局到了 X 和 Y 座標系中的((0,0)-(ScreenWidth,ScreenHeight))和((ScreenWidth,0)-(2*ScreenWidth,ScreenHeight))以及((2*ScreenWidth,0)-(3*ScreenWidth,ScreenHeight))三個矩形區域中,這裏用矩形區域的左上角頂點座標和右下角的頂點座標來表示矩陣。

至此咱們已經完成了整個窗口頁面的佈局,窗口頁面的佈局大小是實際可視屏幕寬度的三倍,因此要顯示全部頁面須要讓頁面滾動。

頁面的平滑移動的實現

下面來看用戶 touch move 的時候程序如何讓頁面進行滑動,而且繪製他們。

頁面的滑動能夠調用 View 的 scrollBy 或 ScrollTo 函數,在 Workspace 的 onTouchEvent 函數中取得用戶的手指移動的距離,而後調用 scrollBy(它的參數就是 X 和 Y 軸上須要移動的距離)來讓 Workspace 這個 View(也是 ViewGroup)移動用戶手指移動的距離,固然 View 移動以前得判斷一下用戶手指移動的距離和速度是否足夠才進行移動,以此減小用戶的誤操做。這部分代碼簡單就不進行深刻分析了,請你們本身看看代碼。

當 Workspace 這個 View 調用 scrollBy 進行 View 的滾動時,必然致使這個 View 無效,從而被系統從新繪製,因此它的 dispatchDraw 函數會被調用來進行子 View(ImageView)的繪製,它自己沒有什麼東西要繪製,因此就不用關心 Workspace 的 onDraw 函數了。dispatchDraw 函數會調用 drawScreens(canvas) 來對子 View 進行繪製。咱們來看一下 FlatWorkspace 的實現:

清單 4

protected void drawScreens(Canvas canvas) { 
        final long drawingTime = getDrawingTime(); 
        final int count = getChildCount(); 
        for (int i = 0; i < count; i++) { 
            drawChild(canvas, getChildAt(i), drawingTime); 
        } 
    }

這裏的 canvas 寬高就是屏幕可視範圍的大小(如 HVGA 屏幕的 320 × 480 大小),而三個子 ImageView 的佈局要超出屏幕的範圍,不在屏幕可視範圍以內的部分是不會被繪製的。這個繪製三個子 ImageView 的函數很重要,是製做立方體翻頁等特效的關鍵地方,FlatWorkspace 實現的是平滑滑動效果,因此咱們直接繪製三個子 ImageView。若是要實現立方體的效果,在繪製三個子 ImageView 的時候就要讓它們被繪製的時候有立體感,這個在 android 中咱們能夠經過上文提到的 Camera 類沿 Y 軸旋轉必定的角度實現。

程序讓用戶進行 touch move 操做的目的是讓用戶選擇一個頁面,若是按照上面的實現,當用戶最後擡起手指時,頁面切換不會很完全,而是象圖 1 同樣停留在兩個頁面之間。因此當用戶擡起手指時程序需判斷一下移動到下一個完整的頁面還有多大距離,而後讓 Workspace 這個 View 再移動這個距離一遍完整的切換到下一頁。在這個移動的過程當中,爲了給用戶一個平滑的感受,不能一下就移動這個距離,而是須要給必定的時間間隔,在這個時間段裏逐漸的移動到位,因此這裏咱們使用 Scroller 類的方法實現逐漸的移動。具體過程是在 Workspace 的 onTouchEvent 函數中檢測到用戶 touch up(擡起手指)時進行應該調整到哪一個頁面的判斷,而後調用 snapToScreen(targetScreen) 跳轉到須要目的頁面,而後它調用 scrollToScreen(screen) 讓 Workspace 這個 View 進行須要的滾動,這個函數在 FlatWorkspace 中的實現以下:

清單 5

public void scrollToScreen(int screen) { 
        final int newX = screen * getWidth(); 
        final int deltaX = newX - getScrollX(); 
        Log.e("FlatWorkspace","scrollToScreen call mScroller.startScroll"); 
        mScroller.startScroll(getScrollX(), getScrollY(), deltaX, getScrollY(), Math.abs(deltaX) * 2); 
        invalidate(); 
  }

這裏的重點是 mScroler.startScroll 部分的代碼,它讓 Workspace view 在時間段 Math.abs(deltaX) * 2 裏移動下一個目標頁面可視化須要移動的距離 deltaX(及目的頁面的座標減去目前已經移動的距離),你們請好好看一下這個 deltaX 的計算,這裏不細說了。這個 mScroller.startScroll 並不會致使 Workspace 當即進行移動,它只會致使當前 View 無效,從而從新繪製,在 Workspace 被它的父親 View 調用繪製的時候,它的 computeScroll 函數會被調用,因此會在這個函數中讓 Workspace 調用 scrollTo 函數進行實際的移動。代碼以下:

清單 6

public void computeScroll() { 
		 if (mScroller.computeScrollOffset()) { 	
			 scrollTo(mScroller.getCurrX(), mScroller.getCurrY()); 
			 //postInvalidate(); 
		 } else if (mNextScreen != INVALID_SCREEN) { 
			 mCurrentScreen = mNextScreen; 
			 mNextScreen = INVALID_SCREEN; 

		 } 
	 }

至此,咱們對 Workspace 的整個運行機制和平滑移動的效果是如何實現的已經介紹完成了。下面咱們來具體談談立體翻頁效果是如何實現的。

立體翻頁效果的實現

經過前面的分析可知,立體翻頁效果能夠在平滑翻頁效果的基礎上經過改寫三個子 ImageView 的繪製來完成。同時可知,翻頁時用戶操做過程分爲三步:放下手指觸摸屏幕,移動手指,擡起手指。手指觸摸屏幕表示頁面之間的滑動要開始了;移動手指的時候頁面應該跟着用戶手指的移動距離進行對應距離的移動,同時系統會根據頁面的移動位置對 Workspace 裏面的三個子 View(即頁面)進行繪製;擡起手指的時候判斷應該移動到哪一個頁面,還須要移動多少距離,而後平滑的移動須要的距離來跳轉到目的頁面上。

爲了顯示立體效果,對每一個子 ImageView 的繪製時得想辦法讓它沿 Y 軸旋轉必定的角度,前面已經提到 android 經過 Camera 這個類提供了這個功能,不須要使用 opengl ES 的東西,固然若是要作出更好的 3D 效果,咱們就須要 opengl ES 的強大功能了。既然要旋轉必定的角度,那這個角度怎麼計算呢?咱們把這個角度和用戶手指移動的距離關聯起來。由於這個立方體只會沿着 Y 軸旋轉,咱們只看這三個面的立方體的頂部就夠了,它的頂部沿着 Y 軸的往其箭頭指示的方向看是一個等邊三角形,每一個面相對於手機屏幕的沿着 Y 軸旋轉的角度的計算方法以下圖所示:

圖 4:初始屏幕位置示意圖

圖 4: 初始屏幕位置示意圖

下圖爲屏幕 1 沿 Y 軸旋轉 45 讀後其餘兩個屏幕須要沿 Y 軸旋轉的角度。

圖 5:旋轉 45 度後屏幕位置示意圖

圖 5: 旋轉 45 度後屏幕位置示意圖

這個變換的部分請看代碼 CubeWorkspace 中函數 drawScreen 的代碼,以下:

清單 7

protected void drawScreen(Canvas canvas, int screen, long drawingTime) { 
        final int width = getWidth(); 
        final int scrollWidth = screen * width; 
        final int scrollX = this.getScrollX();  
        if(scrollWidth > scrollX + width || scrollWidth + width < scrollX) { 
            return; 
        } 
        final View child = getChildAt(screen); 
        final int faceIndex = screen; 
        final float faceDegree = currentDegree - faceIndex * preFaceDegree; 
        if(faceDegree > 90  faceDegree < -90) { 
            return; 
        } 
        final float centerX = (scrollWidth < scrollX)?scrollWidth + width:scrollWidth; 
        final float centerY = getHeight()/2; 
        final Camera camera = mCamera; 
        final Matrix matrix = mMatrix; 
        canvas.save(); 
        camera.save(); 
        camera.rotateY(-faceDegree); 
        camera.getMatrix(matrix); 
        camera.restore(); 
        matrix.preTranslate(-centerX, -centerY); 
        matrix.postTranslate(centerX, centerY); 
        canvas.concat(matrix); 
        drawChild(canvas, child, drawingTime); 
        child.setBackgroundColor(Color.TRANSPARENT); 
        canvas.restore(); 
    }

上面函數中的 currentDegree 變量是變化的,不是一個固定的值,改變這個變量值的方法比較隱蔽,在 AngelBaseWorkspace 的 scrollTo 函數中。AngelBaseWorkspace 中的 scrollTo 函數把 View 類中的函數重載了,這個函數會被 View 中的 scrollBy 函數調用,因此每次 touch 屏幕而且 move 的時候 AngelBaseWorkspace 中的 scrollTo 函數會被調用(onTouchEvent 調用 scrollBy,scrollBy 調用 scrollTo),它會根據用戶 touch move 移動的距離來更改當前頁面的角度,即變量 currentDegree 的值。具體請看以下代碼:

清單 8

public void scrollTo(int x, int y) { 
        if (getScrollX() != x || getScrollY() != y) { 
            int oldX = getScrollX(); 
            int oldY = getScrollY(); 

            super.scrollTo(x, y); 
            //x is the touch action X direction move distance 
            currentDegree = x * degreeOffset;            
            onScrollChanged(x, y, oldX, oldY); 
            invalidate(); 
        } 
    }

這個立方體特效部分的代碼介紹到這裏。

相關文章
相關標籤/搜索