您好,登錄后才能下訂單哦!
這篇文章將為大家詳細講解有關基于SceneForm如何實現子彈射擊,小編覺得挺實用的,因此分享給大家做個參考,希望大家閱讀完這篇文章后可以有所收獲。
基于 SceneForm 實現的子彈射擊(繪制子彈運行軌跡)
Sceneform 框架很強大,不了解 Sceneform 的時候,覺得要想做 3D 場景需要會 OpenGL,而 OpenGL 的學習曲線很陡;接觸到這個框架之后覺得小白也可以很快上手,甚至可以實現第一人稱射擊的效果
注:自己學習 SceneForm 有一段時間了,不過沒有發現模擬重力場的接口,不知道是不是自己漏掉了
模擬射擊效果的思路其實很簡單
1、加載一個子彈模型
2、規劃子彈由近及遠的軌跡
3、繪制子彈的運行軌跡
子彈運行軌跡的邏輯代碼;代碼中涉及的 CleanArFragment 在之前的《ARCore 的 SceneForm 框架在沒有 Plane 情況下的繪制 3D 模型》已經給出;另外需要自行提供一個紋理圖片,即代碼中的 R.drawable.texture。
class MainActivity : AppCompatActivity() { var arFragment : CleanArFragment? = null var camera : Camera? = null var size = Point(); //屏幕尺寸,控制子彈發射的初始位置 var bullet : ModelRenderable? = null var scene : Scene? = null val SHOT = 0x1101 //繪制過程軌跡信號 val SHOT_OVER = 0x1102 //清除子彈模型信號 var handler = object : Handler() { override fun handleMessage(msg : Message) { if (msg.what == SHOT) { //繪制移動過程中的軌跡 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 -> // 設置子彈模型為球體 bullet = ShapeFactory.makeSphere(0.1f, Vector3(0f, 0f, 0f), material) } } ) } private fun shoot() { //從屏幕發出的射線,對應子彈的運行軌跡 var ray = camera!!.screenPointToRay(size.x / 2f, size.y / 2f); var node = Node() //子彈節點 node.renderable = bullet //子彈節點加載子彈模型 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() } // 子線程和主線程穿點的數據類 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>
實現效果如下,因為動圖的偏差,子彈不是很清晰,子彈由中心的紅色十字向遠處射擊
關于“基于SceneForm如何實現子彈射擊”這篇文章就分享到這里了,希望以上內容可以對大家有一定的幫助,使各位可以學到更多知識,如果覺得文章不錯,請把它分享出去讓更多的人看到。
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。