diff --git a/src/android/app/build.gradle b/src/android/app/build.gradle deleted file mode 100644 index df7e43b5ea..0000000000 --- a/src/android/app/build.gradle +++ /dev/null @@ -1,192 +0,0 @@ -apply plugin: 'com.android.application' - -/** - * Use the number of seconds/10 since Jan 1 2016 as the versionCode. - * This lets us upload a new build at most every 10 seconds for the - * next 680 years. - */ -def autoVersion = (int) (((new Date().getTime() / 1000) - 1451606400) / 10) -def buildType -def abiFilter = "arm64-v8a" //, "x86" - -android { - compileSdkVersion 33 - ndkVersion "25.1.8937393" - - compileOptions { - sourceCompatibility JavaVersion.VERSION_1_8 - targetCompatibility JavaVersion.VERSION_1_8 - } - - lintOptions { - // This is important as it will run lint but not abort on error - // Lint has some overly obnoxious "errors" that should really be warnings - abortOnError false - - //Uncomment disable lines for test builds... - //disable 'MissingTranslation'bin - //disable 'ExtraTranslation' - } - - defaultConfig { - // TODO If this is ever modified, change application_id in strings.xml - applicationId "org.citra.citra_emu" - minSdkVersion 28 - targetSdkVersion 31 - versionCode autoVersion - versionName getVersion() - ndk.abiFilters abiFilter - } - - def keystoreFile = System.getenv('ANDROID_KEYSTORE_FILE') - if (keystoreFile != null) { - signingConfigs { - release { - storeFile file(keystoreFile) - storePassword System.getenv('ANDROID_KEYSTORE_PASS') - keyAlias System.getenv('ANDROID_KEY_ALIAS') - keyPassword System.getenv('ANDROID_KEYSTORE_PASS') - } - } - } - - applicationVariants.all { variant -> - buildType = variant.buildType.name // sets the current build type - } - - // Define build types, which are orthogonal to product flavors. - buildTypes { - - // Signed by release key, allowing for upload to Play Store. - release { - signingConfig keystoreFile != null ? signingConfigs.release : signingConfigs.debug - } - - // builds a release build that doesn't need signing - // Attaches 'debug' suffix to version and package name, allowing installation alongside the release build. - relWithDebInfo { - initWith release - applicationIdSuffix ".debug" - versionNameSuffix '-debug' - signingConfig signingConfigs.debug - minifyEnabled false - testCoverageEnabled false - debuggable true - jniDebuggable true - } - - // Signed by debug key disallowing distribution on Play Store. - // Attaches 'debug' suffix to version and package name, allowing installation alongside the release build. - debug { - // TODO If this is ever modified, change application_id in debug/strings.xml - applicationIdSuffix ".debug" - versionNameSuffix '-debug' - debuggable true - jniDebuggable true - } - } - - flavorDimensions "version" - productFlavors { - canary { - dimension "version" - applicationIdSuffix ".canary" - } - nightly { - dimension "version" - } - } - - externalNativeBuild { - cmake { - version "3.22.1" - path "../../../CMakeLists.txt" - } - } - namespace 'org.citra.citra_emu' - - defaultConfig { - externalNativeBuild { - cmake { - arguments "-DENABLE_QT=0", // Don't use QT - "-DENABLE_SDL2=0", // Don't use SDL - "-DANDROID_ARM_NEON=true" // cryptopp requires Neon to work - - abiFilters abiFilter - } - } - } -} - -dependencies { - implementation "androidx.activity:activity:1.5.1" - implementation "androidx.fragment:fragment:1.5.5" - implementation 'androidx.appcompat:appcompat:1.5.1' - implementation 'androidx.exifinterface:exifinterface:1.3.4' - implementation 'androidx.cardview:cardview:1.0.0' - implementation 'androidx.recyclerview:recyclerview:1.2.1' - implementation "androidx.documentfile:documentfile:1.0.1" - implementation 'androidx.constraintlayout:constraintlayout:2.1.4' - implementation 'androidx.lifecycle:lifecycle-viewmodel:2.5.1' - implementation 'androidx.fragment:fragment:1.5.3' - implementation "androidx.slidingpanelayout:slidingpanelayout:1.2.0" - implementation 'com.google.android.material:material:1.6.1' - implementation 'androidx.core:core-splashscreen:1.0.0' - implementation "androidx.work:work-runtime:2.8.1" - - // For loading huge screenshots from the disk. - implementation 'com.squareup.picasso:picasso:2.71828' - - // Allows FRP-style asynchronous operations in Android. - implementation 'io.reactivex:rxandroid:1.2.1' - implementation 'org.ini4j:ini4j:0.5.4' - implementation 'androidx.constraintlayout:constraintlayout:2.1.4' - implementation 'androidx.localbroadcastmanager:localbroadcastmanager:1.1.0' - implementation 'androidx.swiperefreshlayout:swiperefreshlayout:1.1.0' - - // Please don't upgrade the billing library as the newer version is not GPL-compatible - implementation 'com.android.billingclient:billing:2.0.3' - - // To use the androidx.test.core APIs - androidTestImplementation "androidx.test:core:1.5.0" - androidTestImplementation "androidx.test.ext:junit:1.1.5" -} - -def getVersion() { - def versionName = '0.0' - - try { - versionName = 'git describe --always --long'.execute([], project.rootDir).text - .trim() - .replaceAll(/(-0)?-[^-]+$/, "") - } catch (Exception) { - logger.error('Cannot find git, defaulting to dummy version number') - } - - if (System.getenv("GITHUB_ACTIONS") != null) { - def gitTag = System.getenv("GIT_TAG_NAME") - versionName = gitTag ?: versionName - } - - return versionName -} - -// Add task to each variant for copying output APKs to bundle directory. -android.applicationVariants.all { variant -> - def capitalizedName = variant.name.capitalize() - def copyTask = tasks.register("copyBundle${capitalizedName}") { - doLast { - project.copy { - from variant.outputs[0].outputFile.parentFile - include '*.apk' - into layout.buildDirectory.dir("bundle") - } - project.copy { - from layout.buildDirectory.dir("outputs/bundle/${variant.name}") - include '*.aab' - into layout.buildDirectory.dir("bundle") - } - } - } - tasks.named("bundle${capitalizedName}").get().configure { finalizedBy copyTask } -} diff --git a/src/android/app/build.gradle.kts b/src/android/app/build.gradle.kts new file mode 100644 index 0000000000..f92ce0301d --- /dev/null +++ b/src/android/app/build.gradle.kts @@ -0,0 +1,203 @@ +// Copyright 2023 Citra Emulator Project +// Licensed under GPLv2 or any later version +// Refer to the license.txt file included. + +import android.databinding.tool.ext.capitalizeUS + +plugins { + id("com.android.application") + id("org.jetbrains.kotlin.android") +} + +/** + * Use the number of seconds/10 since Jan 1 2016 as the versionCode. + * This lets us upload a new build at most every 10 seconds for the + * next 680 years. + */ +val autoVersion = (((System.currentTimeMillis() / 1000) - 1451606400) / 10).toInt() +val abiFilter = listOf("arm64-v8a"/*, "x86", "x86_64"*/) + +@Suppress("UnstableApiUsage") +android { + namespace = "org.citra.citra_emu" + + compileSdkVersion = "android-33" + ndkVersion = "25.2.9519653" + + compileOptions { + sourceCompatibility = JavaVersion.VERSION_17 + targetCompatibility = JavaVersion.VERSION_17 + } + + kotlinOptions { + jvmTarget = "17" + } + + buildFeatures { + viewBinding = true + } + + lint { + // This is important as it will run lint but not abort on error + // Lint has some overly obnoxious "errors" that should really be warnings + abortOnError = false + } + + defaultConfig { + // TODO If this is ever modified, change application_id in strings.xml + applicationId = "org.citra.citra_emu" + minSdk = 28 + targetSdk = 33 + versionCode = autoVersion + versionName = getGitVersion() + + ndk { + //noinspection ChromeOsAbiSupport + abiFilters += abiFilter + } + + externalNativeBuild { + cmake { + arguments( + "-DENABLE_QT=0", // Don't use QT + "-DENABLE_SDL2=0", // Don't use SDL + "-DANDROID_ARM_NEON=true" // cryptopp requires Neon to work + ) + } + } + } + + val keystoreFile = System.getenv("ANDROID_KEYSTORE_FILE") + if (keystoreFile != null) { + signingConfigs { + create("release") { + storeFile = file(keystoreFile) + storePassword = System.getenv("ANDROID_KEYSTORE_PASS") + keyAlias = System.getenv("ANDROID_KEY_ALIAS") + keyPassword = System.getenv("ANDROID_KEYSTORE_PASS") + } + } + } + + // Define build types, which are orthogonal to product flavors. + buildTypes { + // Signed by release key, allowing for upload to Play Store. + release { + signingConfig = if (keystoreFile != null) { + signingConfigs.getByName("release") + } else { + signingConfigs.getByName("debug") + } + } + + // builds a release build that doesn't need signing + // Attaches 'debug' suffix to version and package name, allowing installation alongside the release build. + register("relWithDebInfo") { + initWith(getByName("release")) + applicationIdSuffix = ".debug" + versionNameSuffix = "-debug" + signingConfig = signingConfigs.getByName("debug") + isMinifyEnabled = false + isDebuggable = true + isJniDebuggable = true + } + + // Signed by debug key disallowing distribution on Play Store. + // Attaches 'debug' suffix to version and package name, allowing installation alongside the release build. + debug { + // TODO If this is ever modified, change application_id in debug/strings.xml + applicationIdSuffix = ".debug" + versionNameSuffix = "-debug" + isDebuggable = true + isJniDebuggable = true + } + } + + flavorDimensions.add("version") + productFlavors { + create("canary") { + dimension = "version" + applicationIdSuffix = ".canary" + } + + create("nightly") { + dimension = "version" + } + } + + externalNativeBuild { + cmake { + version = "3.22.1" + path = file("../../../CMakeLists.txt") + } + } +} + +dependencies { + implementation("androidx.activity:activity-ktx:1.7.2") + implementation("androidx.fragment:fragment-ktx:1.6.0") + implementation("androidx.appcompat:appcompat:1.6.1") + implementation("androidx.documentfile:documentfile:1.0.1") + implementation("androidx.lifecycle:lifecycle-viewmodel-ktx:2.6.1") + implementation("androidx.slidingpanelayout:slidingpanelayout:1.2.0") + implementation("com.google.android.material:material:1.9.0") + implementation("androidx.core:core-splashscreen:1.0.1") + implementation("androidx.work:work-runtime:2.8.1") + + // For loading huge screenshots from the disk. + implementation("com.squareup.picasso:picasso:2.71828") + + // Allows FRP-style asynchronous operations in Android. + implementation("io.reactivex:rxandroid:1.2.1") + + implementation("org.ini4j:ini4j:0.5.4") + implementation("androidx.localbroadcastmanager:localbroadcastmanager:1.1.0") + implementation("androidx.swiperefreshlayout:swiperefreshlayout:1.1.0") + + // Please don't upgrade the billing library as the newer version is not GPL-compatible + implementation("com.android.billingclient:billing:2.0.3") +} + +fun getGitVersion(): String { + var versionName = "0.0" + + try { + versionName = ProcessBuilder("git", "describe", "--always", "--long") + .directory(project.rootDir) + .redirectOutput(ProcessBuilder.Redirect.PIPE) + .redirectError(ProcessBuilder.Redirect.PIPE) + .start().inputStream.bufferedReader().use { it.readText() } + .trim() + .replace(Regex("(-0)?-[^-]+$"), "") + } catch (e: Exception) { + logger.error("Cannot find git, defaulting to dummy version number") + } + + if (System.getenv("GITHUB_ACTIONS") != null) { + val gitTag = System.getenv("GIT_TAG_NAME") + versionName = gitTag ?: versionName + } + + return versionName +} + +android.applicationVariants.configureEach { + val variant = this + val capitalizedName = variant.name.capitalizeUS() + + val copyTask = tasks.register("copyBundle${capitalizedName}") { + doLast { + project.copy { + from(variant.outputs.first().outputFile.parentFile) + include("*.apk") + into(layout.buildDirectory.dir("bundle")) + } + project.copy { + from(layout.buildDirectory.dir("outputs/bundle/${variant.name}")) + include("*.aab") + into(layout.buildDirectory.dir("bundle")) + } + } + } + tasks.named("bundle${capitalizedName}").configure { finalizedBy(copyTask) } +} diff --git a/src/android/app/src/androidTest/java/org/citra/citra_emu/ExampleInstrumentedTest.java b/src/android/app/src/androidTest/java/org/citra/citra_emu/ExampleInstrumentedTest.java deleted file mode 100644 index b7c32a2f14..0000000000 --- a/src/android/app/src/androidTest/java/org/citra/citra_emu/ExampleInstrumentedTest.java +++ /dev/null @@ -1,26 +0,0 @@ -package org.citra.citra_emu; - -import android.content.Context; -import androidx.test.core.app.ApplicationProvider; -import androidx.test.ext.junit.runners.AndroidJUnit4; - -import org.junit.Test; -import org.junit.runner.RunWith; - -import static org.junit.Assert.*; - -/** - * Instrumented test, which will execute on an Android device. - * - * @see Testing documentation - */ -@RunWith(AndroidJUnit4.class) -public class ExampleInstrumentedTest { - @Test - public void useAppContext() { - // Context of the app under test. - Context appContext = ApplicationProvider.getApplicationContext(); - - assertEquals("org.citra.citra_emu", appContext.getPackageName()); - } -} diff --git a/src/android/app/src/main/AndroidManifest.xml b/src/android/app/src/main/AndroidManifest.xml index ec83e022e9..3df8d45d62 100644 --- a/src/android/app/src/main/AndroidManifest.xml +++ b/src/android/app/src/main/AndroidManifest.xml @@ -1,6 +1,5 @@ - + diff --git a/src/android/app/src/test/java/org/citra/citra_emu/ExampleUnitTest.java b/src/android/app/src/test/java/org/citra/citra_emu/ExampleUnitTest.java deleted file mode 100644 index 4e4bb317fb..0000000000 --- a/src/android/app/src/test/java/org/citra/citra_emu/ExampleUnitTest.java +++ /dev/null @@ -1,17 +0,0 @@ -package org.citra.citra_emu; - -import org.junit.Test; - -import static org.junit.Assert.*; - -/** - * Example local unit test, which will execute on the development machine (host). - * - * @see Testing documentation - */ -public class ExampleUnitTest { - @Test - public void addition_isCorrect() { - assertEquals(4, 2 + 2); - } -} \ No newline at end of file diff --git a/src/android/build.gradle b/src/android/build.gradle deleted file mode 100644 index cfece10624..0000000000 --- a/src/android/build.gradle +++ /dev/null @@ -1,28 +0,0 @@ -// Top-level build file where you can add configuration options common to all sub-projects/modules. - -buildscript { - - repositories { - google() - mavenCentral() - jcenter() - } - dependencies { - classpath 'com.android.tools.build:gradle:8.0.2' - - // NOTE: Do not place your application dependencies here; they belong - // in the individual module build.gradle files - } -} - -allprojects { - repositories { - google() - mavenCentral() - jcenter() - } -} - -task clean(type: Delete) { - delete rootProject.buildDir -} diff --git a/src/android/build.gradle.kts b/src/android/build.gradle.kts new file mode 100644 index 0000000000..d2bde7bab4 --- /dev/null +++ b/src/android/build.gradle.kts @@ -0,0 +1,14 @@ +// Copyright 2023 Citra Emulator Project +// Licensed under GPLv2 or any later version +// Refer to the license.txt file included. + +// Top-level build file where you can add configuration options common to all sub-projects/modules. +plugins { + id("com.android.application") version "8.0.2" apply false + id("com.android.library") version "8.0.2" apply false + id("org.jetbrains.kotlin.android") version "1.8.21" apply false +} + +tasks.register("clean").configure { + delete(rootProject.buildDir) +} diff --git a/src/android/settings.gradle b/src/android/settings.gradle deleted file mode 100644 index e7b4def49c..0000000000 --- a/src/android/settings.gradle +++ /dev/null @@ -1 +0,0 @@ -include ':app' diff --git a/src/android/settings.gradle.kts b/src/android/settings.gradle.kts new file mode 100644 index 0000000000..e5b8fcabea --- /dev/null +++ b/src/android/settings.gradle.kts @@ -0,0 +1,23 @@ +// Copyright 2023 Citra Emulator Project +// Licensed under GPLv2 or any later version +// Refer to the license.txt file included. + +pluginManagement { + repositories { + gradlePluginPortal() + google() + mavenCentral() + } +} + +@Suppress("UnstableApiUsage") +dependencyResolutionManagement { + repositoriesMode.set(RepositoriesMode.FAIL_ON_PROJECT_REPOS) + repositories { + google() + mavenCentral() + jcenter() + } +} + +include(":app")