Michael Bestas | dd2506a | 2023-07-20 20:53:02 +0300 | [diff] [blame] | 1 | /* |
| 2 | * SPDX-FileCopyrightText: 2022-2023 The LineageOS Project |
| 3 | * SPDX-License-Identifier: Apache-2.0 |
| 4 | */ |
| 5 | |
LuK1337 | 26b7902 | 2022-09-05 23:32:37 +0200 | [diff] [blame] | 6 | import groovy.util.Node |
| 7 | import groovy.util.NodeList |
| 8 | import groovy.xml.XmlParser |
| 9 | |
LuK1337 | 9c4638c | 2022-09-05 13:37:03 +0200 | [diff] [blame] | 10 | plugins { |
| 11 | id("com.android.application") |
| 12 | id("kotlin-android") |
| 13 | } |
| 14 | |
| 15 | android { |
| 16 | compileSdk = 33 |
| 17 | |
| 18 | defaultConfig { |
Sebastiano Barezzi | f5a9a78 | 2022-11-07 01:06:37 +0100 | [diff] [blame] | 19 | applicationId = "org.lineageos.aperture" |
Sebastiano Barezzi | d11139c | 2022-10-30 16:37:13 +0100 | [diff] [blame] | 20 | minSdk = 26 |
LuK1337 | 9c4638c | 2022-09-05 13:37:03 +0200 | [diff] [blame] | 21 | targetSdk = 33 |
| 22 | versionCode = 1 |
| 23 | versionName = "1.0" |
LuK1337 | 9c4638c | 2022-09-05 13:37:03 +0200 | [diff] [blame] | 24 | } |
| 25 | |
| 26 | buildTypes { |
Sebastiano Barezzi | f0b4640 | 2022-11-07 01:05:51 +0100 | [diff] [blame] | 27 | getByName("release") { |
LuK1337 | 9c4638c | 2022-09-05 13:37:03 +0200 | [diff] [blame] | 28 | // Enables code shrinking, obfuscation, and optimization. |
| 29 | isMinifyEnabled = true |
| 30 | |
| 31 | // Enables resource shrinking. |
| 32 | isShrinkResources = true |
| 33 | |
| 34 | // Includes the default ProGuard rules files. |
| 35 | setProguardFiles( |
| 36 | listOf( |
| 37 | getDefaultProguardFile("proguard-android-optimize.txt"), |
| 38 | "proguard-rules.pro" |
| 39 | ) |
| 40 | ) |
| 41 | } |
Sebastiano Barezzi | f5a9a78 | 2022-11-07 01:06:37 +0100 | [diff] [blame] | 42 | getByName("debug") { |
| 43 | // Append .dev to package name so we won't conflict with AOSP build. |
| 44 | applicationIdSuffix = ".dev" |
| 45 | } |
LuK1337 | 9c4638c | 2022-09-05 13:37:03 +0200 | [diff] [blame] | 46 | } |
| 47 | |
| 48 | compileOptions { |
| 49 | sourceCompatibility = JavaVersion.VERSION_1_8 |
| 50 | targetCompatibility = JavaVersion.VERSION_1_8 |
| 51 | } |
| 52 | |
| 53 | kotlinOptions { |
| 54 | jvmTarget = "1.8" |
| 55 | } |
| 56 | } |
| 57 | |
| 58 | dependencies { |
LuK1337 | 00ac9b1 | 2023-01-25 20:14:54 +0100 | [diff] [blame] | 59 | // Align versions of all Kotlin components |
| 60 | implementation(platform("org.jetbrains.kotlin:kotlin-bom:1.8.0")) |
| 61 | |
LuK1337 | 4ca4060 | 2023-05-24 20:34:45 +0200 | [diff] [blame] | 62 | implementation("androidx.core:core-ktx:1.10.1") |
Sebastiano Barezzi | 1a512eb | 2023-06-22 23:20:38 +0200 | [diff] [blame] | 63 | implementation("androidx.activity:activity-ktx:1.7.2") |
LuK1337 | 6d39d8a | 2023-02-22 20:30:20 +0100 | [diff] [blame] | 64 | implementation("androidx.appcompat:appcompat:1.6.1") |
LuK1337 | 9c4638c | 2022-09-05 13:37:03 +0200 | [diff] [blame] | 65 | implementation("androidx.constraintlayout:constraintlayout:2.1.4") |
Tommy Webb | 636f225 | 2023-01-27 13:33:42 -0500 | [diff] [blame] | 66 | implementation("androidx.exifinterface:exifinterface:1.3.6") |
LuK1337 | 4ca4060 | 2023-05-24 20:34:45 +0200 | [diff] [blame] | 67 | implementation("androidx.lifecycle:lifecycle-viewmodel-ktx:2.6.1") |
LuK1337 | afc94a3 | 2022-09-05 23:30:33 +0200 | [diff] [blame] | 68 | implementation("androidx.preference:preference:1.2.0") |
LuK1337 | bc3d67a | 2023-06-04 19:54:31 +0200 | [diff] [blame] | 69 | implementation("com.google.android.material:material:1.9.0") |
LuK1337 | 9c4638c | 2022-09-05 13:37:03 +0200 | [diff] [blame] | 70 | |
| 71 | // CameraX core library using the camera2 implementation |
LuK1337 | c9be184 | 2023-07-27 15:07:40 +0200 | [diff] [blame^] | 72 | val cameraxVersion = "1.3.0-beta02" |
LuK1337 | 9c4638c | 2022-09-05 13:37:03 +0200 | [diff] [blame] | 73 | // The following line is optional, as the core library is included indirectly by camera-camera2 |
| 74 | implementation("androidx.camera:camera-core:${cameraxVersion}") |
| 75 | implementation("androidx.camera:camera-camera2:${cameraxVersion}") |
| 76 | // If you want to additionally use the CameraX Lifecycle library |
| 77 | implementation("androidx.camera:camera-lifecycle:${cameraxVersion}") |
| 78 | // If you want to additionally use the CameraX VideoCapture library |
| 79 | implementation("androidx.camera:camera-video:${cameraxVersion}") |
| 80 | // If you want to additionally use the CameraX View class |
| 81 | implementation("androidx.camera:camera-view:${cameraxVersion}") |
| 82 | // If you want to additionally use the CameraX Extensions library |
| 83 | implementation("androidx.camera:camera-extensions:${cameraxVersion}") |
| 84 | |
Sebastiano Barezzi | 6b71eb5 | 2022-11-10 00:05:27 +0100 | [diff] [blame] | 85 | // Media3 |
LuK1337 | 4ca4060 | 2023-05-24 20:34:45 +0200 | [diff] [blame] | 86 | val media3Version = "1.0.2" |
Sebastiano Barezzi | 6b71eb5 | 2022-11-10 00:05:27 +0100 | [diff] [blame] | 87 | // For media playback using ExoPlayer |
| 88 | implementation("androidx.media3:media3-exoplayer:$media3Version") |
| 89 | // For building media playback UIs |
| 90 | implementation("androidx.media3:media3-ui:$media3Version") |
| 91 | |
LuK1337 | 9c4638c | 2022-09-05 13:37:03 +0200 | [diff] [blame] | 92 | // ZXing |
| 93 | implementation("com.google.zxing:core:3.5.0") |
| 94 | |
| 95 | // Coil |
LuK1337 | 5c420fa | 2022-10-10 13:22:45 +0200 | [diff] [blame] | 96 | implementation("io.coil-kt:coil:2.2.2") |
| 97 | implementation("io.coil-kt:coil-video:2.2.2") |
LuK1337 | 9c4638c | 2022-09-05 13:37:03 +0200 | [diff] [blame] | 98 | } |
LuK1337 | 26b7902 | 2022-09-05 23:32:37 +0200 | [diff] [blame] | 99 | |
| 100 | tasks.register("generateBp") { |
| 101 | val project = project(":app") |
| 102 | val configuration = project.configurations["debugRuntimeClasspath"] |
| 103 | |
| 104 | val libsBase = File("${project.projectDir.absolutePath}/libs") |
| 105 | libsBase.deleteRecursively() |
| 106 | |
| 107 | val moduleString = { it: ModuleVersionIdentifier -> "${it.group}:${it.name}:${it.version}" } |
| 108 | val modulePath = |
| 109 | { it: ModuleVersionIdentifier -> "${it.group.replace(".", "/")}/${it.name}/${it.version}" } |
| 110 | |
| 111 | val spaces = { it: Int -> |
| 112 | var ret = "" |
| 113 | for (i in it downTo 1) { |
| 114 | ret += ' ' |
| 115 | } |
| 116 | ret |
| 117 | } |
| 118 | |
| 119 | val moduleName = { it: Any -> |
| 120 | when (it) { |
| 121 | is ModuleVersionIdentifier -> { |
| 122 | "${rootProject.name}_${it.group}_${it.name}" |
| 123 | } |
| 124 | is String -> { |
| 125 | if (it.contains(":")) { |
| 126 | val (group, artifactId) = it.split(":") |
| 127 | "${rootProject.name}_${group}_${artifactId}" |
| 128 | } else { |
| 129 | "${rootProject.name}_${it}" |
| 130 | } |
| 131 | } |
| 132 | else -> { |
| 133 | throw Exception("Invalid `it` type") |
| 134 | } |
| 135 | } |
| 136 | } |
| 137 | |
| 138 | val moduleNameAosp = { it: String -> |
| 139 | when (it) { |
| 140 | "androidx.constraintlayout:constraintlayout" -> "androidx-constraintlayout_constraintlayout" |
| 141 | "com.google.auto.value:auto-value-annotations" -> "auto_value_annotations" |
Sebastiano Barezzi | 6b71eb5 | 2022-11-10 00:05:27 +0100 | [diff] [blame] | 142 | "com.google.guava:guava" -> "guava" |
LuK1337 | 26b7902 | 2022-09-05 23:32:37 +0200 | [diff] [blame] | 143 | "com.google.guava:listenablefuture" -> "guava" |
| 144 | "org.jetbrains.kotlin:kotlin-stdlib" -> "kotlin-stdlib" |
| 145 | "org.jetbrains.kotlin:kotlin-stdlib-jdk8" -> "kotlin-stdlib-jdk8" |
| 146 | "org.jetbrains.kotlinx:kotlinx-coroutines-android" -> "kotlinx-coroutines-android" |
| 147 | else -> it.replace(":", "_") |
| 148 | } |
| 149 | } |
| 150 | |
| 151 | val isAvailableInAosp = { group: String, artifactId: String -> |
| 152 | when { |
| 153 | group.startsWith("androidx") -> { |
Sebastiano Barezzi | 6b71eb5 | 2022-11-10 00:05:27 +0100 | [diff] [blame] | 154 | // We provide our own androidx.{camera,media3} & lifecycle-common |
| 155 | !group.startsWith("androidx.camera") && |
| 156 | !group.startsWith("androidx.media3") && |
| 157 | artifactId != "lifecycle-common" |
LuK1337 | 26b7902 | 2022-09-05 23:32:37 +0200 | [diff] [blame] | 158 | } |
LuK1337 | 5c420fa | 2022-10-10 13:22:45 +0200 | [diff] [blame] | 159 | group.startsWith("org.jetbrains") -> true |
LuK1337 | 26b7902 | 2022-09-05 23:32:37 +0200 | [diff] [blame] | 160 | group == "com.google.auto.value" -> true |
| 161 | group == "com.google.guava" -> true |
| 162 | group == "junit" -> true |
| 163 | else -> false |
| 164 | } |
| 165 | } |
| 166 | |
| 167 | // Update app/Android.bp |
| 168 | File("${project.projectDir.absolutePath}/Android.bp").let { file -> |
| 169 | // Read dependencies |
| 170 | val dependencies = "${spaces(8)}// DO NOT EDIT THIS SECTION MANUALLY\n".plus( |
LuK1337 | 00ac9b1 | 2023-01-25 20:14:54 +0100 | [diff] [blame] | 171 | configuration.allDependencies.filter { |
| 172 | // kotlin-bom does not need to be added to dependencies |
| 173 | it.group != "org.jetbrains.kotlin" && it.name != "kotlin-bom" |
| 174 | }.joinToString("\n") { |
LuK1337 | 26b7902 | 2022-09-05 23:32:37 +0200 | [diff] [blame] | 175 | if (isAvailableInAosp(it.group!!, it.name)) { |
| 176 | "${spaces(8)}\"${moduleNameAosp("${it.group}:${it.name}")}\"," |
| 177 | } else { |
| 178 | "${spaces(8)}\"${moduleName("${it.group}:${it.name}")}\"," |
| 179 | } |
| 180 | } |
| 181 | ) |
| 182 | |
| 183 | // Replace existing dependencies with newly generated ones |
| 184 | file.writeText( |
| 185 | file.readText().replace( |
| 186 | "static_libs: \\[.*?\\]".toRegex(RegexOption.DOT_MATCHES_ALL), |
| 187 | "static_libs: [%s]".format("\n$dependencies\n${spaces(4)}") |
| 188 | ) |
| 189 | ) |
| 190 | } |
| 191 | |
| 192 | // Update app/libs |
| 193 | configuration.resolvedConfiguration.resolvedArtifacts.sortedBy { |
| 194 | moduleString(it.moduleVersion.id) |
| 195 | }.distinctBy { |
| 196 | moduleString(it.moduleVersion.id) |
| 197 | }.forEach { |
| 198 | val id = it.moduleVersion.id |
| 199 | |
| 200 | // Skip modules that are available in AOSP |
| 201 | if (isAvailableInAosp(id.group, it.name)) { |
| 202 | return@forEach |
| 203 | } |
| 204 | |
| 205 | // Get file path |
| 206 | val dirPath = "${libsBase}/${modulePath(id)}" |
| 207 | val filePath = "${dirPath}/${it.file.name}" |
| 208 | |
| 209 | // Copy artifact to app/libs |
| 210 | it.file.copyTo(File(filePath)) |
| 211 | |
| 212 | // Parse dependencies |
| 213 | val dependencies = |
Sebastiano Barezzi | fa391d0 | 2022-10-31 14:36:20 +0100 | [diff] [blame] | 214 | it.file.parentFile.parentFile.walk().filter { file -> file.extension == "pom" } |
| 215 | .map { file -> |
LuK1337 | 26b7902 | 2022-09-05 23:32:37 +0200 | [diff] [blame] | 216 | val ret = mutableListOf<String>() |
| 217 | |
| 218 | val pom = XmlParser().parse(file) |
| 219 | val dependencies = (pom["dependencies"] as NodeList).firstOrNull() as Node? |
| 220 | |
| 221 | dependencies?.children()?.forEach { node -> |
| 222 | val dependency = node as Node |
| 223 | ret.add( |
| 224 | "${ |
| 225 | (dependency.get("groupId") as NodeList).text() |
| 226 | }:${ |
| 227 | (dependency.get("artifactId") as NodeList).text() |
| 228 | }" |
| 229 | ) |
| 230 | } |
| 231 | |
| 232 | ret |
Sebastiano Barezzi | fa391d0 | 2022-10-31 14:36:20 +0100 | [diff] [blame] | 233 | }.flatten() |
LuK1337 | 26b7902 | 2022-09-05 23:32:37 +0200 | [diff] [blame] | 234 | |
LuK1337 | 6c560fa | 2022-10-31 22:40:11 +0100 | [diff] [blame] | 235 | var targetSdkVersion = android.defaultConfig.targetSdk |
LuK1337 | 26b7902 | 2022-09-05 23:32:37 +0200 | [diff] [blame] | 236 | var minSdkVersion = 14 |
| 237 | |
| 238 | // Extract AndroidManifest.xml for AARs |
| 239 | if (it.file.extension == "aar") { |
| 240 | copy { |
| 241 | from(zipTree(filePath).matching { include("/AndroidManifest.xml") }.singleFile) |
| 242 | into(dirPath) |
| 243 | } |
| 244 | |
| 245 | val androidManifest = XmlParser().parse(File("${dirPath}/AndroidManifest.xml")) |
| 246 | |
| 247 | val usesSdk = (androidManifest["uses-sdk"] as NodeList).first() as Node |
| 248 | targetSdkVersion = (usesSdk.get("@targetSdkVersion") as Int?) ?: targetSdkVersion |
| 249 | minSdkVersion = (usesSdk.get("@minSdkVersion") as Int?) ?: minSdkVersion |
| 250 | } |
| 251 | |
| 252 | // Write Android.bp |
| 253 | File("$libsBase/Android.bp").let { file -> |
| 254 | // Add autogenerated header if file is empty |
| 255 | if (file.length() == 0L) { |
| 256 | file.writeText("// DO NOT EDIT THIS FILE MANUALLY") |
| 257 | } |
| 258 | |
| 259 | val formatDeps = { addNoDeps: Boolean -> |
| 260 | val deps = dependencies.filter { dep -> |
| 261 | when { |
| 262 | configuration.resolvedConfiguration.resolvedArtifacts.firstOrNull { artifact -> |
| 263 | dep == "${artifact.moduleVersion.id.group}:${artifact.moduleVersion.id.name}" |
| 264 | } == null -> { |
| 265 | val moduleName = if (addNoDeps) { |
| 266 | moduleName(id) |
| 267 | } else { |
| 268 | "${moduleName(id)}-nodeps" |
| 269 | } |
| 270 | println("$moduleName: Skipping $dep because it's not in resolvedArtifacts") |
| 271 | false |
| 272 | } |
| 273 | dep == "org.jetbrains.kotlin:kotlin-stdlib-common" -> false |
| 274 | else -> true |
| 275 | } |
LuK1337 | 47cafc3 | 2023-01-25 23:20:18 +0100 | [diff] [blame] | 276 | }.distinct().toMutableList() |
LuK1337 | 26b7902 | 2022-09-05 23:32:37 +0200 | [diff] [blame] | 277 | |
| 278 | if (addNoDeps) { |
| 279 | // Add -nodeps dependency for android_library/java_library_static |
| 280 | deps.add(0, "${id.group}_${id.name}-nodeps") |
| 281 | } |
| 282 | |
| 283 | var ret = "" |
| 284 | |
| 285 | if (deps.isNotEmpty()) { |
| 286 | deps.forEach { dep -> |
| 287 | ret += if (dep.contains(":")) { |
| 288 | val (group, artifactId) = dep.split(":") |
| 289 | if (isAvailableInAosp(group, artifactId)) { |
| 290 | "\n${spaces(8)}\"${moduleNameAosp(dep)}\"," |
| 291 | } else { |
| 292 | "\n${spaces(8)}\"${moduleName(dep)}\"," |
| 293 | } |
| 294 | } else { |
| 295 | "\n${spaces(8)}\"${moduleName(dep)}\"," |
| 296 | } |
| 297 | } |
| 298 | ret += "\n${spaces(4)}" |
| 299 | } |
| 300 | |
| 301 | ret |
| 302 | } |
| 303 | |
| 304 | when (it.extension) { |
| 305 | "aar" -> { |
| 306 | file.appendText( |
| 307 | """ |
| 308 | |
| 309 | android_library_import { |
| 310 | name: "${moduleName(id)}-nodeps", |
| 311 | aars: ["${modulePath(id)}/${it.file.name}"], |
| 312 | sdk_version: "$targetSdkVersion", |
| 313 | min_sdk_version: "$minSdkVersion", |
| 314 | apex_available: [ |
| 315 | "//apex_available:platform", |
| 316 | "//apex_available:anyapex", |
| 317 | ], |
| 318 | static_libs: [%s], |
| 319 | } |
| 320 | |
| 321 | android_library { |
| 322 | name: "${moduleName(id)}", |
| 323 | sdk_version: "$targetSdkVersion", |
| 324 | min_sdk_version: "$minSdkVersion", |
| 325 | apex_available: [ |
| 326 | "//apex_available:platform", |
| 327 | "//apex_available:anyapex", |
| 328 | ], |
| 329 | manifest: "${modulePath(id)}/AndroidManifest.xml", |
| 330 | static_libs: [%s], |
| 331 | java_version: "1.7", |
| 332 | } |
| 333 | |
| 334 | """.trimIndent().format(formatDeps(false), formatDeps(true)) |
| 335 | ) |
| 336 | } |
| 337 | "jar" -> { |
| 338 | file.appendText( |
| 339 | """ |
| 340 | |
| 341 | java_import { |
| 342 | name: "${moduleName(id)}-nodeps", |
| 343 | jars: ["${modulePath(id)}/${it.file.name}"], |
| 344 | sdk_version: "$targetSdkVersion", |
| 345 | min_sdk_version: "$minSdkVersion", |
| 346 | apex_available: [ |
| 347 | "//apex_available:platform", |
| 348 | "//apex_available:anyapex", |
| 349 | ], |
| 350 | } |
| 351 | |
| 352 | java_library_static { |
| 353 | name: "${moduleName(id)}", |
| 354 | sdk_version: "$targetSdkVersion", |
| 355 | min_sdk_version: "$minSdkVersion", |
| 356 | apex_available: [ |
| 357 | "//apex_available:platform", |
| 358 | "//apex_available:anyapex", |
| 359 | ], |
| 360 | static_libs: [%s], |
| 361 | java_version: "1.7", |
| 362 | } |
| 363 | |
| 364 | """.trimIndent().format(formatDeps(true)) |
| 365 | ) |
| 366 | } |
| 367 | else -> throw Exception("Unknown file extension: ${it.extension}") |
| 368 | } |
| 369 | } |
| 370 | } |
| 371 | } |