Himi 原創, 轉載請注明! 謝謝。
原文地址: http://blog.csdn.net/xiaominghimi/archive/2011/01/04/6116089.aspx
之前在【Android2D游戲開發之四】中我給大家介紹了一張13幀的png的圖,利用設置可視區域的方式來實現動畫效果,但是這些屬于我們自己來實現動畫的方式,其實Android給我們的有兩類自定義動畫方式:
第一類:Frame By Frame 幀動畫 ( 不推薦游戲開發中使用)
所謂幀動畫,就是順序播放事先做好的圖像,類似于放電影;
分析: 此種方式類似我之前的那種利用設置可視區域的方式來實現動畫效果,不僅類似而且還不如!所以此種方式在此不予分析;
第二類:Tween Animation 漸變動畫
即通過對對象不斷做圖像變換 ( 平移、縮放、旋轉 ) 產生動畫效果!實現方式其實就是預先定義一組指令,這些指令指定了圖形變換的類型、觸發時間、持續時間。這些指令可以是以 XML 文件方式定義,也可以是以源代碼方式定義。程序沿著時間線執行這些指令就可以實現動畫 效果。
總結: 那么在Android 游戲開發中我們優先選用兩種方式:第一種設置可視區域的方式來實現動畫效果(幀動畫),需要童鞋們手動實現,那么在之前我的博文【Android2D游戲開發之四】中已經有了相應的源碼!大家可以去下載研究;那么這里就主要為大家詳細分析 Tween Animation!
在講述SurfaceView添加動畫之前,我們先來看看在View中如何實現TweenAnimation以及Tween 中的四種效果;
MyViewAnimation .java
補充:有童鞋說對三種相對位置不太理解,那么我簡單說補充下:
// Animation.ABSOLUTE 相對位置是屏幕左上角 ,絕對位置! // Animation.RELATIVE_TO_SELF 相對位置是自身View;取值為0,是自身左上角,取值為1是自身的右下角; // Animation.RELATIVE_TO_PARENT 相對父類View的位置
當設定了位置類型之后,會讓你傳入X或者Y的值,這里的X,Y可以理解成為一個點坐標!比如是旋轉動畫,那么這個(X,Y)就是旋轉中心點!
OK,對于Tween Animation下的每種動畫效果的實例化的每個參數都解釋的很詳細了!其實動畫的實現不光用代碼可以實現,在xml中注冊實現也是可以的,這里就不多寫了,大家可以自己去嘗試寫一下,那么在view中我們播放一種特效動畫,只要實例化其對象,然后設置下參數,然后startAnimation()就好了,步驟很簡單,只是每個動畫實例化的參數確有著千變萬化的改法,這些我也沒法子一一來給大家演示,大家可以自己改改參數看看實際的效果!當然對于每種動畫我們不光有設置播放的時候,還有一些屬性和方法可以調用,比如Animation.restart()重放動畫, getTransformation()此方法返回 假,說明動畫完成等等很多屬性,請各位童鞋自定實驗 o(∩_∩)o 哈哈~
順便先解釋下MyViewAnimation .java 類中onDraw()方法里的 ( 備注1) !其實這里我是想跟大家說明下Android Animation實現機制
【啟動任意一種動畫效果之前 和 之后 的對比圖】
很明顯、"Himi"字樣在動畫開始前和開始后出現了移動,而且在MyViewAnimation.java中我沒有使用Runnable接口,也沒有調用刷新的函數,那么我來給各位童鞋解釋下原因:
動畫的每種變換其實內部都是一次矩陣運算。在Android 中, Canvas 類中包含當前矩陣,當調用 Canvas.drawBitmap (bmp, x, y, Paint) 繪制時, android 會先把 bmp 做一次矩陣運算,然后將運算的結果顯示在 Canvas 上,然后不斷修改 Canvas 的矩陣并刷新屏幕, View 里的對象就會不停的做圖形變換,動畫就形成了。
還有一點提醒大家: 動畫的播放是對整個游戲畫布進行的操作,這一點要知道喲~
那么下面就要給大家介紹如何在我們的SurfaceView中運用Tween Animation!
MySurfaceViewAnimation.java
動畫代碼實現跟View中的做法一樣,運行模擬器發現按鍵沒效果,不是按鍵沒觸發是本來就存在問題, - -。但是!大家可以把此類里有一行,也就是 (備注2) 的注釋打開,我們給設置背景圖,然后在模擬器上的運行效果如下圖:
很明顯的看到,我們的動畫正常運行了,雖然效果并不是我們想到的!但是這里可以說明一點問題:
SurfaceView 本身具備雙緩沖機制!!!!!
有些文章里說“給SurfaceView添加雙緩沖”,其實是在畫蛇添足 - -,而且介紹的時候拿著單線程與雙線程例子來解釋雙緩沖更高效的實現方法;我想弱弱的問什么是雙緩沖??? 如果SurfaceView不具備雙緩沖,那敢問上面這張截圖如何解釋????
其實要實現雙緩沖,只需要是新建一個Bitmap和Canvas,用這個新建的Canvas把正弦波畫到新建的Bitmap,畫完再通過sfh.lockCanvas獲取SurfaceView對應的Canvas,用這個Canvas把新建的Bitmap畫到SurfaceView上去,這才叫雙緩沖; 還有雙緩存和多線程沒關系!
那么View中動畫的實現機制是在不斷的刷屏不斷的重復調用重寫的onDraw()方法、而在Surfaceview的那張截圖確實也正常的動畫操作了,原因又何在?而且我們設置的背景圖覆蓋我們draw出來的字體!!效果很不理想;那么經過考慮我決定利用布局把View和SurfaceView都一并顯示,用View主要去完成動畫部分,(那么關于如何一并顯示,或者說同時在SurfaceView中添加組件,在之前的【Android 2D開發之六】 和【Android 2D開發之七】都有了詳細講解,那么在這里),當然一并顯示也會有問題,比如我們存在了view和Surfaceiew,那么按鍵的時候觸發的哪個?或者說如何去控制這兩個View?放心,我下面就跟大家一一來講解!
下面先讓我們把我們的view 和 Surfaceview 先同時顯示出來:【 黑色的是MyView (View),白色是MySurfaceView(SurfaceView) 】
先上張運行截圖: (圖4)
main.xml中的代碼
xml中我們注冊了我們自定義的view-MyView 和 SurfaceView-MySurfaceView;
需要強調的有兩點:
1 : 當我們xml中注冊我們的View時,我們View類中的構造函數必須要用
public MyView(Context context, AttributeSet attrs) {} 兩個參數的形式,以前的文章有講解。
2 : 當我們在Xml中注冊兩個View的時候,它們顯示的次序就是根據xml注冊的順序來顯示,比如上面我們先注冊了 MySurfaceView, 然后注冊的 MyView ,那么顯示的時候會把后添加進去的MyView顯示在最上層!
下面我們來看MySurfaceView.java中的代碼:
代碼都很熟悉了, 主要我們來給大家解釋下 備注1,備注2:
備注1:
我在兩個MyView 和 MySurfaceView中都定義了本類一個靜態對象,然后在初始化的時候都利用 =this 的形式進行了實例化;
注意:=this; 的這種實例形式要注意!只能在當前程序中僅存在一個本類對象才可使用!
為什么要實例兩個View的實例而且定義成靜態,這樣做主要為了類之間方便調用和操作!比如在我們這個項目中,我這樣做是為了在MainActivity中去管理兩個View按鍵焦點!下面我會給出MainActivity的代碼,大家一看便知;
備注2:
我在兩個MyView 和 MySurfaceView中都對獲取按鍵焦點注釋掉了,而是在別的類中的調用其View的靜態實例對象就可以任意類中對其設置!這樣就可以很容易去控制到底誰來響應按鍵了。
這里還要強調一下:當xml中注冊多個 View的時候,當我們點擊按鍵之后,Android會先判定哪個ViewsetFocusable(true)設置焦點了,如果都設置了,那么Android 會默認響應在xml中第一個注冊的view ,而不是兩個都會響應。那么為什么不同時響應呢?我解釋下:
上面這截圖是Android SDK Api的樹狀圖,很明顯SurfaceView繼承了View,它倆是基繼承關系,那么不管是子類還是基類一旦響應了按鍵,其基類或者父類就不會再去響應;
下面我們來看MainActivity.java:
備注1:
這里是當程序運行的時候我們默認讓我們的MyView(View)來響應按鍵。通過類名調用對應的View實例,然后設置獲取焦點的函數;
備注2:
這里要注意:不管你在xml中注冊了多少個View ,也不管View是否都設置了獲取焦點,只要你在 MainActivity 中重寫onKeyDown()函數,Android 就會調用此函數。
那么直接在SurfaceView中進行實現動畫的想法這里沒有得到很好的解決,而是我利用布局的方式來一同顯示的方式,希望各位童鞋如果有好的方法,在SurfaceView中直接能使用動畫的建議和想法,希望留言給我,大家一起學習 討論,謝謝 下面給出項目源碼:
源碼下載地址: http://download.csdn.net/source/2969602
(歡迎各位童鞋訂閱本博客,因為咱的更新速度可是很快的~娃哈哈)
【Android游戲開發十四】深入Animation,在SurfaceView中照樣使用Android—Tween Animation!
更多文章、技術交流、商務合作、聯系博主
微信掃碼或搜索:z360901061

微信掃一掃加我為好友
QQ號聯系: 360901061
您的支持是博主寫作最大的動力,如果您喜歡我的文章,感覺我的文章對您有幫助,請用微信掃描下面二維碼支持博主2元、5元、10元、20元等您想捐的金額吧,狠狠點擊下面給點支持吧,站長非常感激您!手機微信長按不能支付解決辦法:請將微信支付二維碼保存到相冊,切換到微信,然后點擊微信右上角掃一掃功能,選擇支付二維碼完成支付。
【本文對您有幫助就好】元
