基于SceneForm實(shí)現(xiàn)子彈射擊(繪制子彈運(yùn)行軌跡)
基于 SceneForm 實(shí)現(xiàn)的子彈射擊(繪制子彈運(yùn)行軌跡)
Sceneform 框架很強(qiáng)大,不了解 Sceneform 的時(shí)候,覺得要想做 3D 場(chǎng)景需要會(huì) OpenGL,而 OpenGL 的學(xué)習(xí)曲線很陡;接觸到這個(gè)框架之后覺得小白也可以很快上手,甚至可以實(shí)現(xiàn)第一人稱射擊的效果
注:自己學(xué)習(xí) SceneForm 有一段時(shí)間了,不過沒有發(fā)現(xiàn)模擬重力場(chǎng)的接口,不知道是不是自己漏掉了
模擬射擊效果的思路其實(shí)很簡(jiǎn)單
1、加載一個(gè)子彈模型
2、規(guī)劃子彈由近及遠(yuǎn)的軌跡
3、繪制子彈的運(yùn)行軌跡
子彈運(yùn)行軌跡的邏輯代碼;代碼中涉及的 CleanArFragment 在之前的《ARCore 的 SceneForm 框架在沒有 Plane 情況下的繪制 3D 模型》已經(jīng)給出;另外需要自行提供一個(gè)紋理圖片,即代碼中的 R.drawable.texture。
class MainActivity : AppCompatActivity() { var arFragment : CleanArFragment? = null var camera : Camera? = null var size = Point(); //屏幕尺寸,控制子彈發(fā)射的初始位置 var bullet : ModelRenderable? = null var scene : Scene? = null val SHOT = 0x1101 //繪制過程軌跡信號(hào) val SHOT_OVER = 0x1102 //清除子彈模型信號(hào) var handler = object : Handler() { override fun handleMessage(msg : Message) { if (msg.what == SHOT) { //繪制移動(dòng)過程中的軌跡 var currentStatus = msg.obj as CurrentStatus currentStatus.node.worldPosition = currentStatus.status } else if (msg.what == SHOT_OVER) { //一次射擊完成,清除屏幕的子彈 var node = msg.obj as Node scene!!.removeChild(node) } } } override fun onCreate(savedInstanceState: Bundle?) { super.onCreate(savedInstanceState) setContentView(R.layout.activity_main) // 獲取屏幕尺寸 val display = windowManager.defaultDisplay display.getRealSize(size) arFragment = this.supportFragmentManager.findFragmentById(R.id.arFragment) as CleanArFragment arFragment!!.arSceneView.planeRenderer.isEnabled = false //禁止 sceneform 框架的平面繪制 scene = arFragment!!.arSceneView.scene camera = scene!!.camera initbullet() shootButton.setOnClickListener(listener) } var listener : View.OnClickListener = object : View.OnClickListener{ override fun onClick(v: View?) { shoot() } } @TargetApi(Build.VERSION_CODES.N) //初始化子彈模型 private fun initbullet() { Texture.builder().setSource(this@MainActivity, R.drawable.texture).build() .thenAccept( { texture -> MaterialFactory.makeOpaqueWithTexture(this@MainActivity, texture) .thenAccept { material -> // 設(shè)置子彈模型為球體 bullet = ShapeFactory.makeSphere(0.1f, Vector3(0f, 0f, 0f), material) } } ) } private fun shoot() { //從屏幕發(fā)出的射線,對(duì)應(yīng)子彈的運(yùn)行軌跡 var ray = camera!!.screenPointToRay(size.x / 2f, size.y / 2f); var node = Node() //子彈節(jié)點(diǎn) node.renderable = bullet //子彈節(jié)點(diǎn)加載子彈模型 scene!!.addChild(node) Thread(object : Runnable{ override fun run() { //子彈射擊過程中的軌跡,子線程處理軌跡事件,主線程改變軌跡位置 for (i in 1 .. 200 ) { //子彈射程 20 m var stepLen = i; var currentPoint = ray.getPoint(stepLen * 0.1f) var msg = handler.obtainMessage() msg.what = SHOT msg.obj = CurrentStatus(node, currentPoint) handler.sendMessage(msg) } //子彈超出距離后,從屏幕清除掉 var msg = handler.obtainMessage() msg.what = SHOT_OVER msg.obj = node handler.sendMessage(msg) } }).start() } // 子線程和主線程穿點(diǎn)的數(shù)據(jù)類 data class CurrentStatus(var node : Node, var status : Vector3) }
界面布局
<?xml version="1.0" encoding="utf-8"?> <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent" android:layout_height="match_parent" tools:context=".MainActivity"> <fragment android:layout_width="match_parent" android:layout_height="match_parent" android:id="@+id/arFragment" android:name="com.hosh.shootapplication.CleanArFragment"/> <View android:layout_width="35dp" android:layout_height="2dp" android:background="#ff0000" android:layout_centerInParent="true" /> <View android:layout_width="2dp" android:layout_height="35dp" android:background="#ff0000" android:layout_centerInParent="true" /> <Button android:layout_width="wrap_content" android:layout_height="wrap_content" android:id="@+id/shootButton" android:layout_alignParentBottom="true" android:layout_centerHorizontal="true" android:layout_marginBottom="8dp" android:text="@string/shoot" /> </RelativeLayout>
實(shí)現(xiàn)效果如下,因?yàn)閯?dòng)圖的偏差,子彈不是很清晰,子彈由中心的紅色十字向遠(yuǎn)處射擊
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持我們。
上一篇:android實(shí)現(xiàn)清理緩存功能
欄 目:Android
下一篇:Android Selector獲取焦點(diǎn)后文本背景修改的實(shí)現(xiàn)代碼
本文標(biāo)題:基于SceneForm實(shí)現(xiàn)子彈射擊(繪制子彈運(yùn)行軌跡)
本文地址:http://mengdiqiu.com.cn/a1/Android/9062.html
您可能感興趣的文章
- 01-10Android實(shí)現(xiàn)雙擊返回鍵退出應(yīng)用實(shí)現(xiàn)方法詳解
- 01-10android實(shí)現(xiàn)記住用戶名和密碼以及自動(dòng)登錄
- 01-10android實(shí)現(xiàn)簡(jiǎn)單計(jì)算器功能
- 01-10Android 友盟第三方登錄與分享的實(shí)現(xiàn)代碼
- 01-10C++自定義API函數(shù)實(shí)現(xiàn)大數(shù)相乘算法
- 01-10如何給Flutter界面切換實(shí)現(xiàn)點(diǎn)特效
- 01-10android實(shí)現(xiàn)指紋識(shí)別功能
- 01-10Android實(shí)現(xiàn)圓形漸變加載進(jìn)度條
- 01-10RecyclerView仿應(yīng)用列表實(shí)現(xiàn)網(wǎng)格布局
- 01-10Android實(shí)現(xiàn)底部導(dǎo)航欄的主界面


閱讀排行
- 1C語言 while語句的用法詳解
- 2java 實(shí)現(xiàn)簡(jiǎn)單圣誕樹的示例代碼(圣誕
- 3利用C語言實(shí)現(xiàn)“百馬百擔(dān)”問題方法
- 4C語言中計(jì)算正弦的相關(guān)函數(shù)總結(jié)
- 5c語言計(jì)算三角形面積代碼
- 6什么是 WSH(腳本宿主)的詳細(xì)解釋
- 7C++ 中隨機(jī)函數(shù)random函數(shù)的使用方法
- 8正則表達(dá)式匹配各種特殊字符
- 9C語言十進(jìn)制轉(zhuǎn)二進(jìn)制代碼實(shí)例
- 10C語言查找數(shù)組里數(shù)字重復(fù)次數(shù)的方法
本欄相關(guān)
- 01-10Android自定義View之繪制圓形頭像功能
- 01-10Android實(shí)現(xiàn)雙擊返回鍵退出應(yīng)用實(shí)現(xiàn)方
- 01-10android實(shí)現(xiàn)簡(jiǎn)單計(jì)算器功能
- 01-10android實(shí)現(xiàn)記住用戶名和密碼以及自動(dòng)
- 01-10C++自定義API函數(shù)實(shí)現(xiàn)大數(shù)相乘算法
- 01-10Android 友盟第三方登錄與分享的實(shí)現(xiàn)代
- 01-10android實(shí)現(xiàn)指紋識(shí)別功能
- 01-10如何給Flutter界面切換實(shí)現(xiàn)點(diǎn)特效
- 01-10Android實(shí)現(xiàn)圓形漸變加載進(jìn)度條
- 01-10Emoji表情在Android JNI中的兼容性問題詳
隨機(jī)閱讀
- 01-11ajax實(shí)現(xiàn)頁面的局部加載
- 08-05織夢(mèng)dedecms什么時(shí)候用欄目交叉功能?
- 08-05dedecms(織夢(mèng))副欄目數(shù)量限制代碼修改
- 01-10delphi制作wav文件的方法
- 01-11Mac OSX 打開原生自帶讀寫NTFS功能(圖文
- 08-05DEDE織夢(mèng)data目錄下的sessions文件夾有什
- 01-10SublimeText編譯C開發(fā)環(huán)境設(shè)置
- 01-10使用C語言求解撲克牌的順子及n個(gè)骰子
- 04-02jquery與jsp,用jquery
- 01-10C#中split用法實(shí)例總結(jié)