From 1e50024ffdaf66a57ac0c5689baa75998328a452 Mon Sep 17 00:00:00 2001 From: feie9456 Date: Tue, 2 Dec 2025 12:19:49 +0800 Subject: [PATCH] init --- .gitignore | 15 + .idea/.gitignore | 3 + app/.gitignore | 1 + app/build.gradle.kts | 59 +++ app/proguard-rules.pro | 21 + .../pinball/ExampleInstrumentedTest.kt | 24 + app/src/main/AndroidManifest.xml | 27 ++ .../java/com/example/pinball/MainActivity.kt | 430 ++++++++++++++++++ .../com/example/pinball/ui/theme/Color.kt | 11 + .../com/example/pinball/ui/theme/Theme.kt | 58 +++ .../java/com/example/pinball/ui/theme/Type.kt | 34 ++ .../res/drawable/ic_launcher_background.xml | 170 +++++++ .../res/drawable/ic_launcher_foreground.xml | 30 ++ app/src/main/res/layout/activity_main.xml | 63 +++ .../res/mipmap-anydpi-v26/ic_launcher.xml | 6 + .../mipmap-anydpi-v26/ic_launcher_round.xml | 6 + app/src/main/res/mipmap-hdpi/ic_launcher.webp | Bin 0 -> 1404 bytes .../res/mipmap-hdpi/ic_launcher_round.webp | Bin 0 -> 2898 bytes app/src/main/res/mipmap-mdpi/ic_launcher.webp | Bin 0 -> 982 bytes .../res/mipmap-mdpi/ic_launcher_round.webp | Bin 0 -> 1772 bytes .../main/res/mipmap-xhdpi/ic_launcher.webp | Bin 0 -> 1900 bytes .../res/mipmap-xhdpi/ic_launcher_round.webp | Bin 0 -> 3918 bytes .../main/res/mipmap-xxhdpi/ic_launcher.webp | Bin 0 -> 2884 bytes .../res/mipmap-xxhdpi/ic_launcher_round.webp | Bin 0 -> 5914 bytes .../main/res/mipmap-xxxhdpi/ic_launcher.webp | Bin 0 -> 3844 bytes .../res/mipmap-xxxhdpi/ic_launcher_round.webp | Bin 0 -> 7778 bytes app/src/main/res/values/colors.xml | 10 + app/src/main/res/values/dimens.xml | 7 + app/src/main/res/values/strings.xml | 3 + app/src/main/res/values/themes.xml | 5 + app/src/main/res/xml/backup_rules.xml | 13 + .../main/res/xml/data_extraction_rules.xml | 19 + .../com/example/pinball/ExampleUnitTest.kt | 17 + build.gradle.kts | 6 + gradle.properties | 23 + gradle/libs.versions.toml | 32 ++ gradle/wrapper/gradle-wrapper.jar | Bin 0 -> 59203 bytes gradle/wrapper/gradle-wrapper.properties | 6 + gradlew | 185 ++++++++ gradlew.bat | 89 ++++ settings.gradle.kts | 24 + 41 files changed, 1397 insertions(+) create mode 100644 .gitignore create mode 100644 .idea/.gitignore create mode 100644 app/.gitignore create mode 100644 app/build.gradle.kts create mode 100644 app/proguard-rules.pro create mode 100644 app/src/androidTest/java/com/example/pinball/ExampleInstrumentedTest.kt create mode 100644 app/src/main/AndroidManifest.xml create mode 100644 app/src/main/java/com/example/pinball/MainActivity.kt create mode 100644 app/src/main/java/com/example/pinball/ui/theme/Color.kt create mode 100644 app/src/main/java/com/example/pinball/ui/theme/Theme.kt create mode 100644 app/src/main/java/com/example/pinball/ui/theme/Type.kt create mode 100644 app/src/main/res/drawable/ic_launcher_background.xml create mode 100644 app/src/main/res/drawable/ic_launcher_foreground.xml create mode 100644 app/src/main/res/layout/activity_main.xml create mode 100644 app/src/main/res/mipmap-anydpi-v26/ic_launcher.xml create mode 100644 app/src/main/res/mipmap-anydpi-v26/ic_launcher_round.xml create mode 100644 app/src/main/res/mipmap-hdpi/ic_launcher.webp create mode 100644 app/src/main/res/mipmap-hdpi/ic_launcher_round.webp create mode 100644 app/src/main/res/mipmap-mdpi/ic_launcher.webp create mode 100644 app/src/main/res/mipmap-mdpi/ic_launcher_round.webp create mode 100644 app/src/main/res/mipmap-xhdpi/ic_launcher.webp create mode 100644 app/src/main/res/mipmap-xhdpi/ic_launcher_round.webp create mode 100644 app/src/main/res/mipmap-xxhdpi/ic_launcher.webp create mode 100644 app/src/main/res/mipmap-xxhdpi/ic_launcher_round.webp create mode 100644 app/src/main/res/mipmap-xxxhdpi/ic_launcher.webp create mode 100644 app/src/main/res/mipmap-xxxhdpi/ic_launcher_round.webp create mode 100644 app/src/main/res/values/colors.xml create mode 100644 app/src/main/res/values/dimens.xml create mode 100644 app/src/main/res/values/strings.xml create mode 100644 app/src/main/res/values/themes.xml create mode 100644 app/src/main/res/xml/backup_rules.xml create mode 100644 app/src/main/res/xml/data_extraction_rules.xml create mode 100644 app/src/test/java/com/example/pinball/ExampleUnitTest.kt create mode 100644 build.gradle.kts create mode 100644 gradle.properties create mode 100644 gradle/libs.versions.toml create mode 100644 gradle/wrapper/gradle-wrapper.jar create mode 100644 gradle/wrapper/gradle-wrapper.properties create mode 100644 gradlew create mode 100644 gradlew.bat create mode 100644 settings.gradle.kts diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..aa724b7 --- /dev/null +++ b/.gitignore @@ -0,0 +1,15 @@ +*.iml +.gradle +/local.properties +/.idea/caches +/.idea/libraries +/.idea/modules.xml +/.idea/workspace.xml +/.idea/navEditor.xml +/.idea/assetWizardSettings.xml +.DS_Store +/build +/captures +.externalNativeBuild +.cxx +local.properties diff --git a/.idea/.gitignore b/.idea/.gitignore new file mode 100644 index 0000000..26d3352 --- /dev/null +++ b/.idea/.gitignore @@ -0,0 +1,3 @@ +# Default ignored files +/shelf/ +/workspace.xml diff --git a/app/.gitignore b/app/.gitignore new file mode 100644 index 0000000..42afabf --- /dev/null +++ b/app/.gitignore @@ -0,0 +1 @@ +/build \ No newline at end of file diff --git a/app/build.gradle.kts b/app/build.gradle.kts new file mode 100644 index 0000000..d2eb9b3 --- /dev/null +++ b/app/build.gradle.kts @@ -0,0 +1,59 @@ +plugins { + alias(libs.plugins.android.application) + alias(libs.plugins.kotlin.android) + alias(libs.plugins.kotlin.compose) +} + +android { + namespace = "com.example.pinball" + compileSdk = 36 + + defaultConfig { + applicationId = "com.example.pinball" + minSdk = 24 + targetSdk = 36 + versionCode = 1 + versionName = "1.0" + + testInstrumentationRunner = "androidx.test.runner.AndroidJUnitRunner" + } + + buildTypes { + release { + isMinifyEnabled = false + proguardFiles( + getDefaultProguardFile("proguard-android-optimize.txt"), + "proguard-rules.pro" + ) + } + } + compileOptions { + sourceCompatibility = JavaVersion.VERSION_11 + targetCompatibility = JavaVersion.VERSION_11 + } + kotlinOptions { + jvmTarget = "11" + } + buildFeatures { + compose = true + } +} + +dependencies { + + implementation(libs.androidx.core.ktx) + implementation(libs.androidx.lifecycle.runtime.ktx) + implementation(libs.androidx.activity.compose) + implementation(platform(libs.androidx.compose.bom)) + implementation(libs.androidx.compose.ui) + implementation(libs.androidx.compose.ui.graphics) + implementation(libs.androidx.compose.ui.tooling.preview) + implementation(libs.androidx.compose.material3) + testImplementation(libs.junit) + androidTestImplementation(libs.androidx.junit) + androidTestImplementation(libs.androidx.espresso.core) + androidTestImplementation(platform(libs.androidx.compose.bom)) + androidTestImplementation(libs.androidx.compose.ui.test.junit4) + debugImplementation(libs.androidx.compose.ui.tooling) + debugImplementation(libs.androidx.compose.ui.test.manifest) +} \ No newline at end of file diff --git a/app/proguard-rules.pro b/app/proguard-rules.pro new file mode 100644 index 0000000..481bb43 --- /dev/null +++ b/app/proguard-rules.pro @@ -0,0 +1,21 @@ +# Add project specific ProGuard rules here. +# You can control the set of applied configuration files using the +# proguardFiles setting in build.gradle. +# +# For more details, see +# http://developer.android.com/guide/developing/tools/proguard.html + +# If your project uses WebView with JS, uncomment the following +# and specify the fully qualified class name to the JavaScript interface +# class: +#-keepclassmembers class fqcn.of.javascript.interface.for.webview { +# public *; +#} + +# Uncomment this to preserve the line number information for +# debugging stack traces. +#-keepattributes SourceFile,LineNumberTable + +# If you keep the line number information, uncomment this to +# hide the original source file name. +#-renamesourcefileattribute SourceFile \ No newline at end of file diff --git a/app/src/androidTest/java/com/example/pinball/ExampleInstrumentedTest.kt b/app/src/androidTest/java/com/example/pinball/ExampleInstrumentedTest.kt new file mode 100644 index 0000000..0fef4ee --- /dev/null +++ b/app/src/androidTest/java/com/example/pinball/ExampleInstrumentedTest.kt @@ -0,0 +1,24 @@ +package com.example.pinball + +import androidx.test.platform.app.InstrumentationRegistry +import androidx.test.ext.junit.runners.AndroidJUnit4 + +import org.junit.Test +import org.junit.runner.RunWith + +import org.junit.Assert.* + +/** + * Instrumented test, which will execute on an Android device. + * + * See [testing documentation](http://d.android.com/tools/testing). + */ +@RunWith(AndroidJUnit4::class) +class ExampleInstrumentedTest { + @Test + fun useAppContext() { + // Context of the app under test. + val appContext = InstrumentationRegistry.getInstrumentation().targetContext + assertEquals("com.example.pinball", appContext.packageName) + } +} \ No newline at end of file diff --git a/app/src/main/AndroidManifest.xml b/app/src/main/AndroidManifest.xml new file mode 100644 index 0000000..c138e4e --- /dev/null +++ b/app/src/main/AndroidManifest.xml @@ -0,0 +1,27 @@ + + + + + + + + + + + + + + \ No newline at end of file diff --git a/app/src/main/java/com/example/pinball/MainActivity.kt b/app/src/main/java/com/example/pinball/MainActivity.kt new file mode 100644 index 0000000..b6d9598 --- /dev/null +++ b/app/src/main/java/com/example/pinball/MainActivity.kt @@ -0,0 +1,430 @@ +package com.example.pinball + +import android.app.Activity +import android.content.Context +import android.graphics.* +import android.os.Bundle +import android.view.Choreographer +import android.view.MotionEvent +import android.view.View +import android.view.Window +import android.view.WindowManager +import android.widget.Button +import android.widget.FrameLayout +import android.widget.TextView +import android.widget.Toast +import java.util.* +import kotlin.math.abs + +class MainActivity : Activity() { + // Screen dimensions + private var tableWidth = 0 + private var tableHeight = 0 + + // Paddle properties + private var racketY = 0f + private var racketHeight = 0 + private var racketWidth = 0 + private var racketX = 0f + + // Ball properties + private var ballSize = 0 + private var ballX = 0f + private var ballY = 0f + + // Physics (Pixels per second) + private var velocityX = 0f + private var velocityY = 0f + private val baseSpeed = 1000f // Base speed in pixels/sec + private var lastFrameTimeNanos = 0L + + // Game State + private var isLose = false + private var score = 0 + private var gameTimeSeconds = 0f + private var currentLevel = 1 + + // Bricks + data class Brick( + val rect: RectF, + var isBroken: Boolean = false, + val color: Int + ) + private var bricks = mutableListOf() + + private lateinit var gameView: GameView + private lateinit var scoreText: TextView + private lateinit var btnRestart: Button + private val rand = Random() + + // Game Loop + private val frameCallback = object : Choreographer.FrameCallback { + override fun doFrame(frameTimeNanos: Long) { + if (isLose) return + + if (lastFrameTimeNanos != 0L) { + val dt = (frameTimeNanos - lastFrameTimeNanos) / 1_000_000_000f + updateGame(dt) + } + lastFrameTimeNanos = frameTimeNanos + + gameView.invalidate() + scoreText.text = "Score: $score Level: $currentLevel" + + Choreographer.getInstance().postFrameCallback(this) + } + } + + override fun onCreate(savedInstanceState: Bundle?) { + super.onCreate(savedInstanceState) + requestWindowFeature(Window.FEATURE_NO_TITLE) + window.setFlags( + WindowManager.LayoutParams.FLAG_FULLSCREEN, + WindowManager.LayoutParams.FLAG_FULLSCREEN + ) + + setContentView(R.layout.activity_main) + + // Initialize dimensions + racketHeight = resources.getDimensionPixelSize(R.dimen.racket_height) + racketWidth = resources.getDimensionPixelSize(R.dimen.racket_width) + ballSize = resources.getDimensionPixelSize(R.dimen.ball_size) + + val displayMetrics = resources.displayMetrics + tableWidth = displayMetrics.widthPixels + tableHeight = displayMetrics.heightPixels + + // Move paddle up (approx 20% from bottom) + racketY = tableHeight * 0.80f + + gameView = GameView(this) + findViewById(R.id.game_container).addView(gameView) + + scoreText = findViewById(R.id.score_text) + btnRestart = findViewById(R.id.btn_restart) + val btnSave = findViewById