1
app/.gitignore
vendored
Normal file
1
app/.gitignore
vendored
Normal file
@@ -0,0 +1 @@
|
||||
/build
|
||||
478
app/build.gradle.kts
Normal file
478
app/build.gradle.kts
Normal file
@@ -0,0 +1,478 @@
|
||||
/*
|
||||
* Copyright (c) 田梓萱[小草林] 2021-2024.
|
||||
* All Rights Reserved.
|
||||
* All codes are protected by China's regulations on the protection of computer software, and infringement must be investigated.
|
||||
* 版权所有 (c) 田梓萱[小草林] 2021-2024.
|
||||
* 所有代码均受中国《计算机软件保护条例》保护,侵权必究.
|
||||
*/
|
||||
@file:Suppress("UnstableApiUsage")
|
||||
import java.time.LocalDate
|
||||
import java.time.format.DateTimeFormatter.ofPattern
|
||||
import java.util.Properties
|
||||
|
||||
plugins {
|
||||
id("com.android.application")
|
||||
id("com.didiglobal.booster")
|
||||
id("io.github.leavesczy.trace")
|
||||
id("com.google.devtools.ksp")
|
||||
id("androidx.room")
|
||||
id("kotlin-android")
|
||||
id("kotlin-parcelize")
|
||||
id("kotlinx-serialization")
|
||||
id("androidx.navigation.safeargs.kotlin")
|
||||
id("com.google.gms.google-services")
|
||||
id("com.google.firebase.firebase-perf")
|
||||
// id("com.google.firebase.crashlytics")
|
||||
id("version-catalog")
|
||||
id("io.sentry.android.gradle")
|
||||
id("io.sentry.kotlin.compiler.gradle")
|
||||
id("com.mikepenz.aboutlibraries.plugin")
|
||||
id("android.aop")
|
||||
}
|
||||
|
||||
val properties = Properties()
|
||||
val isDebug = gradle.startParameter.taskNames.any { it.contains("debug", true) }
|
||||
|
||||
properties.load(project.rootProject.file("gradle.properties").inputStream())
|
||||
|
||||
room { schemaDirectory("$projectDir/schemas") }
|
||||
|
||||
android {
|
||||
namespace = "ink.xcl.saitra"
|
||||
compileSdk = 34
|
||||
ndkVersion = "26.1.10909125"
|
||||
// compileSdkPreview = "UpsideDownCake"
|
||||
// flavorDimensions += listOf("arch")
|
||||
// productFlavors {
|
||||
// create("arm64") {
|
||||
// dimension = "arch"
|
||||
// ndk {
|
||||
// abiFilters += listOf("arm64-v8a") // 仅64位
|
||||
// }
|
||||
// externalNativeBuild {
|
||||
// cmake {
|
||||
// cFlags += listOf("-Os -flto -ffunction-sections -fdata-sections -fvisibility=hidden")
|
||||
// abiFilters += listOf("arm64-v8a")
|
||||
// cppFlags += listOf("-Os -flto -ffunction-sections -fdata-sections
|
||||
// -fvisibility=hidden")
|
||||
// arguments += listOf("-Wl,--gc-sections", "-DANDROID_STL=c++_static")
|
||||
// }
|
||||
// }
|
||||
// }
|
||||
// create("arm32") {
|
||||
// dimension = "arch"
|
||||
// ndk {
|
||||
// abiFilters += listOf("armeabi-v7a") // 仅32位
|
||||
// }
|
||||
// externalNativeBuild {
|
||||
// cmake {
|
||||
// cFlags += listOf("-Os -flto -ffunction-sections -fdata-sections -fvisibility=hidden")
|
||||
// abiFilters += listOf("armeabi-v7a")
|
||||
// cppFlags += listOf("-Os -flto -ffunction-sections -fdata-sections
|
||||
// -fvisibility=hidden")
|
||||
// arguments += listOf("-Wl,--gc-sections", "-DANDROID_STL=c++_static")
|
||||
// }
|
||||
// }
|
||||
// }
|
||||
// }
|
||||
lint {
|
||||
quiet = true
|
||||
ignoreWarnings = true
|
||||
abortOnError = false
|
||||
checkReleaseBuilds = true
|
||||
}
|
||||
experimentalProperties["android.experimental.art-profile-r8-rewriting"] = true
|
||||
experimentalProperties["android.experimental.r8.dex-startup-optimization"] = true
|
||||
defaultConfig {
|
||||
applicationId = "ink.xcl.saitra"
|
||||
multiDexEnabled = true
|
||||
ndk {
|
||||
abiFilters +=
|
||||
listOf(
|
||||
"armeabi-v7a", // 兼容32位
|
||||
"arm64-v8a", // 兼容64位
|
||||
// "x86",//不兼容x86
|
||||
// "x86_64",//不兼容x86_64
|
||||
) // 仅兼容64位
|
||||
}
|
||||
minSdk = 22
|
||||
targetSdk = 34
|
||||
versionCode = LocalDate.now().format(ofPattern("yyyyMMdd")).toInt()
|
||||
versionName = "0.0.1"
|
||||
proguardFiles(getDefaultProguardFile("proguard-defaults.txt"), "proguard-rules.pro")
|
||||
multiDexKeepProguard = file("multidex-config.pro")
|
||||
testInstrumentationRunner = "androidx.test.runner.AndroidJUnitRunner"
|
||||
vectorDrawables { useSupportLibrary = true }
|
||||
// externalNativeBuild {
|
||||
// cmake {
|
||||
// cFlags += listOf("-Os -flto -ffunction-sections -fdata-sections -fvisibility=hidden")
|
||||
// abiFilters += listOf("armeabi-v7a", "arm64-v8a")
|
||||
// cppFlags += listOf("-Os -flto -ffunction-sections -fdata-sections -fvisibility=hidden")
|
||||
// arguments += listOf("-Wl,--gc-sections", "-DANDROID_STL=c++_static")
|
||||
// }
|
||||
// }
|
||||
}
|
||||
bundle {
|
||||
abi {
|
||||
// This property is set to true by default.
|
||||
enableSplit = true
|
||||
}
|
||||
}
|
||||
// signingConfigs {
|
||||
// // We use a bundled debug keystore, to allow debug builds from CI to be upgradable
|
||||
// create("release") {
|
||||
// storeFile = rootProject.file("./key/saitra.keystore")
|
||||
// storePassword = "2025Tzx."
|
||||
// keyAlias = "XCL_TZX_EI_AChatI"
|
||||
// keyPassword = "2025Tzx."
|
||||
// enableV1Signing = true
|
||||
// enableV2Signing = true
|
||||
// enableV3Signing = true
|
||||
// enableV4Signing = true
|
||||
// }
|
||||
// }
|
||||
composeOptions { kotlinCompilerExtensionVersion = libs.versions.composeCompiler.get() }
|
||||
packaging {
|
||||
// jniLibs {
|
||||
// excludes += "**/libchat.so"
|
||||
// }//动态加载-不合规-暂时移除
|
||||
resources {
|
||||
excludes += "META-INF/DEPENDENCIES"
|
||||
excludes += "META-INF/NOTICE"
|
||||
excludes += "META-INF/LICENSE"
|
||||
excludes += "META-INF/LICENSE.txt"
|
||||
excludes += "META-INF/NOTICE.txt"
|
||||
excludes += "META-INF/LICENSE.md"
|
||||
excludes += "META-INF/LICENSE-notice.md"
|
||||
excludes += "META-INF/{AL2.0,LGPL2.1,LICENSE,NOTICE,DEPENDENCIES}"
|
||||
excludes += "DebugProbesKt.bin"
|
||||
excludes += "META-INF/*.kotlin_module"
|
||||
excludes += "**/*.kotlin_module"
|
||||
excludes += "**/*.version"
|
||||
}
|
||||
}
|
||||
buildTypes {
|
||||
debug {
|
||||
isDebuggable = true
|
||||
isCrunchPngs = true
|
||||
isJniDebuggable = true
|
||||
isMinifyEnabled = false
|
||||
isShrinkResources = false
|
||||
// signingConfig = signingConfigs.getByName("release")
|
||||
// versionNameSuffix = "-debug-${LocalDate.now().format(ofPattern("yyyy-MM-dd"))}"
|
||||
}
|
||||
|
||||
release {
|
||||
isDebuggable = false
|
||||
isCrunchPngs = true
|
||||
isJniDebuggable = false
|
||||
isMinifyEnabled = true
|
||||
isShrinkResources = true
|
||||
// signingConfig = signingConfigs.getByName("release")
|
||||
// versionNameSuffix = "-release-${LocalDate.now().format(ofPattern("yyyy-MM-dd"))}"
|
||||
}
|
||||
// create("benchmark") {
|
||||
// signingConfig = signingConfigs.getByName("release")
|
||||
// matchingFallbacks += listOf("release")
|
||||
// isDebuggable = false
|
||||
// }
|
||||
}
|
||||
compileOptions {
|
||||
isCoreLibraryDesugaringEnabled = true
|
||||
sourceCompatibility = JavaVersion.VERSION_17
|
||||
targetCompatibility = JavaVersion.VERSION_17
|
||||
}
|
||||
kotlinOptions {
|
||||
jvmTarget = "17"
|
||||
val arguments =
|
||||
listOf(
|
||||
// https://kotlinlang.org/docs/compiler-reference.html#progressive
|
||||
"-progressive",
|
||||
// Enable Java default method generation.
|
||||
"-Xjvm-default=all",
|
||||
"-Xjsr305=strict",
|
||||
// Generate smaller bytecode by not generating runtime not-null assertions.
|
||||
"-Xno-call-assertions",
|
||||
"-Xno-param-assertions",
|
||||
"-Xno-receiver-assertions",
|
||||
)
|
||||
freeCompilerArgs += arguments
|
||||
}
|
||||
splits {
|
||||
abi {
|
||||
isEnable = true
|
||||
reset()
|
||||
include("armeabi-v7a", "arm64-v8a")
|
||||
isUniversalApk = true
|
||||
}
|
||||
}
|
||||
buildFeatures {
|
||||
compose = true
|
||||
// dataBinding = false
|
||||
viewBinding = true
|
||||
buildConfig = true
|
||||
aidl = true
|
||||
}
|
||||
// externalNativeBuild {
|
||||
// cmake {
|
||||
// path = file("CMakeLists.txt")
|
||||
// version = "3.22.1"
|
||||
// }
|
||||
// }
|
||||
// 屏蔽 task (uploadCrashlyticsMappingFileRelease)-仅限国内构建、使用Mini打包不需要
|
||||
// gradle.taskGraph.whenReady {
|
||||
// tasks.find { it.name == "uploadCrashlyticsMappingFileRelease" }?.enabled = false
|
||||
// }
|
||||
|
||||
applicationVariants.all {
|
||||
outputs.all {
|
||||
if (this is com.android.build.gradle.internal.api.ApkVariantOutputImpl) {
|
||||
val filterData = getFilter(com.android.build.OutputFile.ABI)
|
||||
// println("EI_v${versionName}(${versionCode})_${filterData}_${buildType.name}.apk")
|
||||
outputFileName =
|
||||
if (filterData.isNullOrEmpty()) {
|
||||
"Tra_v${versionName}(${versionCode})_${buildType.name}.apk"
|
||||
} else {
|
||||
"Tra_v${versionName}(${versionCode})_(${filterData}|${buildType.name}).apk"
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
sourceSets { getByName("main") { jniLibs.srcDir("libs") } }
|
||||
}
|
||||
|
||||
aboutLibraries {
|
||||
// - `./gradlew app:exportLibraryDefinitions -PaboutLibraries.exportPath=src/main/res/raw`
|
||||
registerAndroidTasks = false
|
||||
outputFileName = "aboutlibraries.json"
|
||||
configPath = "config"
|
||||
offlineMode = false
|
||||
fetchRemoteLicense = true
|
||||
fetchRemoteFunding = true
|
||||
excludeFields = arrayOf("developers", "funding")
|
||||
includePlatform = true
|
||||
strictMode = com.mikepenz.aboutlibraries.plugin.StrictMode.WARN
|
||||
duplicationMode = com.mikepenz.aboutlibraries.plugin.DuplicateMode.MERGE
|
||||
duplicationRule = com.mikepenz.aboutlibraries.plugin.DuplicateRule.GROUP
|
||||
prettyPrint = true
|
||||
}
|
||||
|
||||
dependencies {
|
||||
// implementation(fileTree(mapOf("dir" to "libs", "include" to listOf("*.jar"))))
|
||||
implementation(libs.material) {
|
||||
exclude(group = "org.jetbrains.kotlin", module = "kotlin-bom")
|
||||
exclude(group = "androidx.activity", module = "activity")
|
||||
exclude(group = "androidx.appcompat", module = "appcompat")
|
||||
}
|
||||
implementation(libs.material3)
|
||||
implementation(libs.material3.adaptive)
|
||||
implementation(libs.material3.adaptive.navigation.suite)
|
||||
implementation(libs.material3.window.size)
|
||||
implementation(libs.androidx.foundation)
|
||||
implementation(libs.androidx.foundation.layout)
|
||||
implementation(libs.androidx.ui)
|
||||
implementation(libs.androidx.ui.util)
|
||||
implementation(libs.androidx.ui.graphics)
|
||||
debugImplementation(libs.androidx.ui.tooling)
|
||||
debugImplementation(libs.androidx.ui.tooling.preview)
|
||||
debugImplementation(libs.squareup.leakcanary)
|
||||
implementation(libs.androidx.material)
|
||||
implementation(libs.androidx.material.icons.core)
|
||||
implementation(libs.androidx.material.icons.extended)
|
||||
implementation(libs.androidx.runtime)
|
||||
implementation(libs.androidx.animation)
|
||||
implementation(libs.androidx.recyclerview)
|
||||
implementation(libs.androidx.recyclerview.selection)
|
||||
// androidTestImplementation(libs.androidx.runtime.tracing)
|
||||
// androidTestImplementation(libs.androidx.ui.test.manifest)
|
||||
// androidTestImplementation(libs.androidx.ui.test.junit4)
|
||||
implementation(libs.androidx.glance)
|
||||
implementation(libs.androidx.glance.appwidget)
|
||||
// implementation(libs.androidx.glance.material)
|
||||
implementation(libs.androidx.glance.material3)
|
||||
debugImplementation(libs.glance.appwidget.viewer)
|
||||
implementation(libs.glance.appwidget.host)
|
||||
implementation(libs.glance.appwidget.configuration)
|
||||
// implementation(libs.accompanist.permissions)
|
||||
implementation(libs.accompanist.navigation.material) {
|
||||
exclude(group = "androidx.navigation", module = "navigation-compose")
|
||||
}
|
||||
implementation(libs.accompanist.drawablepainter)
|
||||
implementation(libs.accompanist.adaptive)
|
||||
implementation(libs.androidx.navigation.ui.ktx)
|
||||
implementation(libs.androidx.navigation.compose)
|
||||
// androidTestImplementation(libs.androidx.navigation.testing)
|
||||
implementation(libs.androidx.window)
|
||||
implementation(libs.androidx.window.java)
|
||||
// androidTestImplementation(libs.androidx.window.testing)
|
||||
implementation(libs.androidx.lifecycle.viewmodel.ktx)
|
||||
implementation(libs.androidx.lifecycle.viewmodel.compose)
|
||||
implementation(libs.androidx.lifecycle.runtime.compose)
|
||||
implementation(libs.androidx.lifecycle.viewmodel.savedstate)
|
||||
implementation(libs.androidx.lifecycle.common.java8)
|
||||
implementation(libs.androidx.lifecycle.service)
|
||||
implementation(libs.androidx.lifecycle.process)
|
||||
implementation(libs.androidx.lifecycle.reactivestreams.ktx)
|
||||
implementation(libs.androidx.lifecycle.livedata.ktx)
|
||||
// androidTestImplementation(libs.androidx.core.testing)
|
||||
// androidTestImplementation(libs.androidx.lifecycle.runtime.testing)
|
||||
// room数据库
|
||||
implementation(libs.androidx.room.runtime)
|
||||
// annotationProcessor(libs.androidx.room.compiler)
|
||||
ksp(libs.androidx.room.compiler)
|
||||
implementation(libs.androidx.room.ktx)
|
||||
implementation(libs.androidx.room.guava)
|
||||
// testImplementation(libs.androidx.room.testing)
|
||||
implementation(libs.sqlcipher.android) { exclude(group = "androidx.sqlite", module = "sqlite") }
|
||||
implementation(libs.androidx.sqlite)
|
||||
implementation(libs.androidx.sqlite.ktx)
|
||||
implementation(libs.androidx.sqlite.framework)
|
||||
// work
|
||||
implementation(libs.androidx.work.runtime.ktx)
|
||||
implementation(libs.androidx.work.gcm)
|
||||
// androidTestImplementation(libs.androidx.work.testing)
|
||||
implementation(libs.androidx.work.multiprocess)
|
||||
// sentry
|
||||
implementation(libs.sentry.android)
|
||||
implementation(libs.sentry.compose.android)
|
||||
implementation(libs.sentry.android.okhttp) {
|
||||
exclude(group = "com.squareup.okhttp3", module = "okhttp")
|
||||
}
|
||||
// implementation(libs.firebase.perf.ktx)
|
||||
// implementation(libs.firebase.crashlytics.ktx)
|
||||
// implementation(libs.firebase.crashlytics.ndk)
|
||||
// implementation(libs.firebase.config.ktx)
|
||||
// implementation(libs.firebase.analytics.ktx)
|
||||
// implementation(libs.firebase.messaging.ktx)
|
||||
// Ktor
|
||||
implementation(libs.ktor.client.core)
|
||||
implementation(libs.ktor.client.okhttp) {
|
||||
exclude(group = "com.squareup.okhttp3", module = "okhttp")
|
||||
}
|
||||
implementation(libs.ktor.client.logging)
|
||||
implementation(libs.ktor.client.content.negotiation)
|
||||
implementation(libs.ktor.client.encoding)
|
||||
implementation(libs.ktor.serialization.kotlinx.protobuf)
|
||||
// androidTestImplementation(libs.ktor.client.mock)
|
||||
// Okhttp
|
||||
implementation(libs.okhttp)
|
||||
implementation(libs.okhttp.sse)
|
||||
debugImplementation(libs.okhttp.monitor)
|
||||
releaseImplementation(libs.okhttp.monitor.noop)
|
||||
// Grpc
|
||||
// runtimeOnly(libs.grpc.okhttp)
|
||||
// api(libs.grpc.stub)
|
||||
// api(libs.grpc.protobuf.lite)
|
||||
// api(libs.grpc.kotlin.stub)
|
||||
// implementation(libs.protobuf.kotlinlite) // protobuf-kotlin解析
|
||||
implementation(libs.kotlinx.serialization.protobuf) {
|
||||
exclude(group = "com.google.protobuf", module = "protobuf-javalite")
|
||||
} // protobuf-serialization解析
|
||||
implementation(libs.protobuf.javalite) // protobuf-java解析
|
||||
// 图片渲染-Coil
|
||||
implementation(libs.coil3)
|
||||
implementation(libs.coil3.compose)
|
||||
implementation(libs.coil3.gif)
|
||||
implementation(libs.coil3.svg) { exclude(group = "com.caverock", module = "androidsvg-aar") }
|
||||
implementation(libs.coil3.video)
|
||||
implementation(libs.coil3.network) { exclude(group = "com.squareup.okhttp3", module = "okhttp") }
|
||||
implementation(libs.zoomimage.compose.coil)
|
||||
implementation(libs.coil)
|
||||
implementation(libs.coil.compose)
|
||||
// Koin依赖注入
|
||||
implementation(libs.koin.core)
|
||||
implementation(libs.koin.core.coroutines)
|
||||
implementation(libs.koin.android)
|
||||
implementation(libs.koin.android.compat)
|
||||
// implementation(libs.koin.androidx.workmanager)
|
||||
implementation(libs.koin.androidx.compose)
|
||||
implementation(libs.koin.androidx.compose.navigation)
|
||||
// androidTestImplementation(libs.koin.android.test)
|
||||
// androidTestImplementation(libs.androidx.tracing.perfetto)
|
||||
// androidTestImplementation(libs.androidx.tracing.perfetto.binary)
|
||||
implementation(libs.androidx.core)
|
||||
implementation(libs.androidx.core.google.shortcuts)
|
||||
implementation(libs.androidx.core.ktx)
|
||||
implementation(libs.androidx.core.animation)
|
||||
implementation(libs.androidx.core.performance)
|
||||
implementation(libs.androidx.core.splashscreen)
|
||||
implementation(libs.androidx.startup.runtime)
|
||||
implementation(libs.androidx.appcompat) {
|
||||
exclude(group = "androidx.fragment", module = "fragment")
|
||||
}
|
||||
implementation(libs.androidx.appcompat.resources)
|
||||
implementation(libs.androidx.emoji2)
|
||||
implementation(libs.androidx.emoji2.bundled)
|
||||
// implementation(libs.androidx.emoji2.emojipicker)
|
||||
// implementation(libs.androidx.gridlayout)
|
||||
// implementation(libs.billing.ktx)
|
||||
// implementation(libs.alipaysdk.android)//应用内支付-支付宝-暂时移除
|
||||
implementation(libs.androidx.activity.ktx)
|
||||
implementation(libs.androidx.activity.compose)
|
||||
implementation(libs.annotation)
|
||||
implementation(libs.androidx.concurrent.futures.ktx)
|
||||
implementation(libs.androidx.annotation.experimental)
|
||||
implementation(libs.androidx.constraintlayout)
|
||||
implementation(libs.androidx.constraintlayout.compose)
|
||||
implementation(libs.androidx.asynclayoutinflater)
|
||||
implementation(libs.androidx.collection.ktx)
|
||||
// exoplayer
|
||||
implementation(libs.media3.exoplayer)
|
||||
// agora
|
||||
implementation(libs.agora.rtc.basic)
|
||||
// implementation(libs.material)
|
||||
implementation(libs.androidx.multidex)
|
||||
coreLibraryDesugaring(libs.desugar.jdk.libs)
|
||||
implementation(libs.bcprov.jdk15on)
|
||||
implementation(libs.relinker)
|
||||
implementation(libs.guava)
|
||||
// implementation(libs.lottie.compose) // AE动画
|
||||
implementation(libs.androidx.transition.ktx)
|
||||
// 协程
|
||||
implementation(libs.kotlinx.coroutines.core)
|
||||
implementation(libs.kotlinx.coroutines.play.services)
|
||||
implementation(libs.kotlinx.coroutines.android)
|
||||
implementation(libs.kotlin.reflect)
|
||||
implementation(libs.kotlin.stdlib)
|
||||
implementation(libs.kotlin.stdlib.jdk7)
|
||||
implementation(libs.kotlin.stdlib.jdk8)
|
||||
// androidTestImplementation(libs.kotlinx.coroutines.test)
|
||||
// androidTestImplementation(libs.androidx.junit)
|
||||
// androidTestImplementation(libs.androidx.espresso.core)
|
||||
// testImplementation(libs.junit.jupiter.api)
|
||||
// androidTestImplementation(libs.junit.jupiter.api)
|
||||
// Acra 异常全局捕获
|
||||
implementation(libs.acra.core)
|
||||
implementation(libs.acra.toast)
|
||||
implementation(libs.acra.advanced.scheduler)
|
||||
implementation(libs.auto.service.annotations)
|
||||
ksp(libs.auto.service.ksp)
|
||||
implementation(libs.androidx.fragment)
|
||||
// GMS
|
||||
implementation(libs.play.services.location)
|
||||
// implementation(libs.integrity)
|
||||
// TODO:暂时取消掉翻译的依赖
|
||||
// implementation(libs.playstore.dynamic.feature.support)
|
||||
// {
|
||||
// exclude(group = "com.google.android.play", module = "core")
|
||||
// }
|
||||
// implementation(libs.translate)//文本翻译
|
||||
// Play Core
|
||||
implementation(libs.asset.delivery.ktx)
|
||||
implementation(libs.feature.delivery.ktx)
|
||||
// implementation(libs.review.ktx)
|
||||
implementation(libs.app.update.ktx)
|
||||
implementation(libs.android.aop.core)
|
||||
implementation(libs.android.aop.annotation)
|
||||
ksp(libs.android.aop.ksp)
|
||||
implementation(libs.aboutlibraries.core)
|
||||
implementation(libs.aboutlibraries.compose.m3)
|
||||
// QrCode
|
||||
// implementation(libs.qrose)
|
||||
// 拼音
|
||||
// implementation(libs.pinyin4j)
|
||||
// profile优化
|
||||
implementation(libs.androidx.profileinstaller)
|
||||
}
|
||||
|
||||
configurations.all { resolutionStrategy.cacheChangingModulesFor(10, TimeUnit.SECONDS) }
|
||||
21
app/proguard-rules.pro
vendored
Normal file
21
app/proguard-rules.pro
vendored
Normal file
@@ -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
|
||||
28
app/src/main/AndroidManifest.xml
Normal file
28
app/src/main/AndroidManifest.xml
Normal file
@@ -0,0 +1,28 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
|
||||
xmlns:tools="http://schemas.android.com/tools">
|
||||
|
||||
<application
|
||||
android:allowBackup="true"
|
||||
android:dataExtractionRules="@xml/data_extraction_rules"
|
||||
android:fullBackupContent="@xml/backup_rules"
|
||||
android:icon="@mipmap/ic_launcher"
|
||||
android:label="@string/app_name"
|
||||
android:roundIcon="@mipmap/ic_launcher_round"
|
||||
android:supportsRtl="true"
|
||||
android:theme="@style/Theme.AiTra"
|
||||
tools:targetApi="31">
|
||||
<activity
|
||||
android:name=".MainActivity"
|
||||
android:exported="true"
|
||||
android:label="@string/app_name"
|
||||
android:theme="@style/Theme.AiTra">
|
||||
<intent-filter>
|
||||
<action android:name="android.intent.action.MAIN" />
|
||||
|
||||
<category android:name="android.intent.category.LAUNCHER" />
|
||||
</intent-filter>
|
||||
</activity>
|
||||
</application>
|
||||
|
||||
</manifest>
|
||||
38
app/src/main/java/ink/xcl/saitra/MainActivity.kt
Normal file
38
app/src/main/java/ink/xcl/saitra/MainActivity.kt
Normal file
@@ -0,0 +1,38 @@
|
||||
package ink.xcl.saitra
|
||||
|
||||
import android.os.Bundle
|
||||
import androidx.activity.ComponentActivity
|
||||
import androidx.activity.compose.setContent
|
||||
import androidx.compose.foundation.layout.fillMaxSize
|
||||
import androidx.compose.material3.MaterialTheme
|
||||
import androidx.compose.material3.Surface
|
||||
import androidx.compose.material3.Text
|
||||
import androidx.compose.runtime.Composable
|
||||
import androidx.compose.ui.Modifier
|
||||
import androidx.compose.ui.tooling.preview.Preview
|
||||
import ink.xcl.saitra.ui.theme.AiTraTheme
|
||||
|
||||
class MainActivity : ComponentActivity() {
|
||||
override fun onCreate(savedInstanceState: Bundle?) {
|
||||
super.onCreate(savedInstanceState)
|
||||
setContent {
|
||||
AiTraTheme {
|
||||
// A surface container using the 'background' color from the theme
|
||||
Surface(
|
||||
modifier = Modifier.fillMaxSize(),
|
||||
color = MaterialTheme.colorScheme.background
|
||||
) {
|
||||
Greeting("Android")
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@Composable
|
||||
fun Greeting(name: String, modifier: Modifier = Modifier) {
|
||||
Text(
|
||||
text = "Hello $name!",
|
||||
modifier = modifier
|
||||
)
|
||||
}
|
||||
11
app/src/main/java/ink/xcl/saitra/ui/theme/Color.kt
Normal file
11
app/src/main/java/ink/xcl/saitra/ui/theme/Color.kt
Normal file
@@ -0,0 +1,11 @@
|
||||
package ink.xcl.saitra.ui.theme
|
||||
|
||||
import androidx.compose.ui.graphics.Color
|
||||
|
||||
val Purple80 = Color(0xFFD0BCFF)
|
||||
val PurpleGrey80 = Color(0xFFCCC2DC)
|
||||
val Pink80 = Color(0xFFEFB8C8)
|
||||
|
||||
val Purple40 = Color(0xFF6650a4)
|
||||
val PurpleGrey40 = Color(0xFF625b71)
|
||||
val Pink40 = Color(0xFF7D5260)
|
||||
70
app/src/main/java/ink/xcl/saitra/ui/theme/Theme.kt
Normal file
70
app/src/main/java/ink/xcl/saitra/ui/theme/Theme.kt
Normal file
@@ -0,0 +1,70 @@
|
||||
package ink.xcl.saitra.ui.theme
|
||||
|
||||
import android.app.Activity
|
||||
import android.os.Build
|
||||
import androidx.compose.foundation.isSystemInDarkTheme
|
||||
import androidx.compose.material3.MaterialTheme
|
||||
import androidx.compose.material3.darkColorScheme
|
||||
import androidx.compose.material3.dynamicDarkColorScheme
|
||||
import androidx.compose.material3.dynamicLightColorScheme
|
||||
import androidx.compose.material3.lightColorScheme
|
||||
import androidx.compose.runtime.Composable
|
||||
import androidx.compose.runtime.SideEffect
|
||||
import androidx.compose.ui.graphics.toArgb
|
||||
import androidx.compose.ui.platform.LocalContext
|
||||
import androidx.compose.ui.platform.LocalView
|
||||
import androidx.core.view.WindowCompat
|
||||
|
||||
private val DarkColorScheme = darkColorScheme(
|
||||
primary = Purple80,
|
||||
secondary = PurpleGrey80,
|
||||
tertiary = Pink80
|
||||
)
|
||||
|
||||
private val LightColorScheme = lightColorScheme(
|
||||
primary = Purple40,
|
||||
secondary = PurpleGrey40,
|
||||
tertiary = Pink40
|
||||
|
||||
/* Other default colors to override
|
||||
background = Color(0xFFFFFBFE),
|
||||
surface = Color(0xFFFFFBFE),
|
||||
onPrimary = Color.White,
|
||||
onSecondary = Color.White,
|
||||
onTertiary = Color.White,
|
||||
onBackground = Color(0xFF1C1B1F),
|
||||
onSurface = Color(0xFF1C1B1F),
|
||||
*/
|
||||
)
|
||||
|
||||
@Composable
|
||||
fun AiTraTheme(
|
||||
darkTheme: Boolean = isSystemInDarkTheme(),
|
||||
// Dynamic color is available on Android 12+
|
||||
dynamicColor: Boolean = true,
|
||||
content: @Composable () -> Unit
|
||||
) {
|
||||
val colorScheme = when {
|
||||
dynamicColor && Build.VERSION.SDK_INT >= Build.VERSION_CODES.S -> {
|
||||
val context = LocalContext.current
|
||||
if (darkTheme) dynamicDarkColorScheme(context) else dynamicLightColorScheme(context)
|
||||
}
|
||||
|
||||
darkTheme -> DarkColorScheme
|
||||
else -> LightColorScheme
|
||||
}
|
||||
val view = LocalView.current
|
||||
if (!view.isInEditMode) {
|
||||
SideEffect {
|
||||
val window = (view.context as Activity).window
|
||||
window.statusBarColor = colorScheme.primary.toArgb()
|
||||
WindowCompat.getInsetsController(window, view).isAppearanceLightStatusBars = darkTheme
|
||||
}
|
||||
}
|
||||
|
||||
MaterialTheme(
|
||||
colorScheme = colorScheme,
|
||||
typography = Typography,
|
||||
content = content
|
||||
)
|
||||
}
|
||||
34
app/src/main/java/ink/xcl/saitra/ui/theme/Type.kt
Normal file
34
app/src/main/java/ink/xcl/saitra/ui/theme/Type.kt
Normal file
@@ -0,0 +1,34 @@
|
||||
package ink.xcl.saitra.ui.theme
|
||||
|
||||
import androidx.compose.material3.Typography
|
||||
import androidx.compose.ui.text.TextStyle
|
||||
import androidx.compose.ui.text.font.FontFamily
|
||||
import androidx.compose.ui.text.font.FontWeight
|
||||
import androidx.compose.ui.unit.sp
|
||||
|
||||
// Set of Material typography styles to start with
|
||||
val Typography = Typography(
|
||||
bodyLarge = TextStyle(
|
||||
fontFamily = FontFamily.Default,
|
||||
fontWeight = FontWeight.Normal,
|
||||
fontSize = 16.sp,
|
||||
lineHeight = 24.sp,
|
||||
letterSpacing = 0.5.sp
|
||||
)
|
||||
/* Other default text styles to override
|
||||
titleLarge = TextStyle(
|
||||
fontFamily = FontFamily.Default,
|
||||
fontWeight = FontWeight.Normal,
|
||||
fontSize = 22.sp,
|
||||
lineHeight = 28.sp,
|
||||
letterSpacing = 0.sp
|
||||
),
|
||||
labelSmall = TextStyle(
|
||||
fontFamily = FontFamily.Default,
|
||||
fontWeight = FontWeight.Medium,
|
||||
fontSize = 11.sp,
|
||||
lineHeight = 16.sp,
|
||||
letterSpacing = 0.5.sp
|
||||
)
|
||||
*/
|
||||
)
|
||||
30
app/src/main/res/drawable-v24/ic_launcher_foreground.xml
Normal file
30
app/src/main/res/drawable-v24/ic_launcher_foreground.xml
Normal file
@@ -0,0 +1,30 @@
|
||||
<vector xmlns:android="http://schemas.android.com/apk/res/android"
|
||||
xmlns:aapt="http://schemas.android.com/aapt"
|
||||
android:width="108dp"
|
||||
android:height="108dp"
|
||||
android:viewportWidth="108"
|
||||
android:viewportHeight="108">
|
||||
<path android:pathData="M31,63.928c0,0 6.4,-11 12.1,-13.1c7.2,-2.6 26,-1.4 26,-1.4l38.1,38.1L107,108.928l-32,-1L31,63.928z">
|
||||
<aapt:attr name="android:fillColor">
|
||||
<gradient
|
||||
android:endX="85.84757"
|
||||
android:endY="92.4963"
|
||||
android:startX="42.9492"
|
||||
android:startY="49.59793"
|
||||
android:type="linear">
|
||||
<item
|
||||
android:color="#44000000"
|
||||
android:offset="0.0" />
|
||||
<item
|
||||
android:color="#00000000"
|
||||
android:offset="1.0" />
|
||||
</gradient>
|
||||
</aapt:attr>
|
||||
</path>
|
||||
<path
|
||||
android:fillColor="#FFFFFF"
|
||||
android:fillType="nonZero"
|
||||
android:pathData="M65.3,45.828l3.8,-6.6c0.2,-0.4 0.1,-0.9 -0.3,-1.1c-0.4,-0.2 -0.9,-0.1 -1.1,0.3l-3.9,6.7c-6.3,-2.8 -13.4,-2.8 -19.7,0l-3.9,-6.7c-0.2,-0.4 -0.7,-0.5 -1.1,-0.3C38.8,38.328 38.7,38.828 38.9,39.228l3.8,6.6C36.2,49.428 31.7,56.028 31,63.928h46C76.3,56.028 71.8,49.428 65.3,45.828zM43.4,57.328c-0.8,0 -1.5,-0.5 -1.8,-1.2c-0.3,-0.7 -0.1,-1.5 0.4,-2.1c0.5,-0.5 1.4,-0.7 2.1,-0.4c0.7,0.3 1.2,1 1.2,1.8C45.3,56.528 44.5,57.328 43.4,57.328L43.4,57.328zM64.6,57.328c-0.8,0 -1.5,-0.5 -1.8,-1.2s-0.1,-1.5 0.4,-2.1c0.5,-0.5 1.4,-0.7 2.1,-0.4c0.7,0.3 1.2,1 1.2,1.8C66.5,56.528 65.6,57.328 64.6,57.328L64.6,57.328z"
|
||||
android:strokeWidth="1"
|
||||
android:strokeColor="#00000000" />
|
||||
</vector>
|
||||
170
app/src/main/res/drawable/ic_launcher_background.xml
Normal file
170
app/src/main/res/drawable/ic_launcher_background.xml
Normal file
@@ -0,0 +1,170 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<vector xmlns:android="http://schemas.android.com/apk/res/android"
|
||||
android:width="108dp"
|
||||
android:height="108dp"
|
||||
android:viewportWidth="108"
|
||||
android:viewportHeight="108">
|
||||
<path
|
||||
android:fillColor="#3DDC84"
|
||||
android:pathData="M0,0h108v108h-108z" />
|
||||
<path
|
||||
android:fillColor="#00000000"
|
||||
android:pathData="M9,0L9,108"
|
||||
android:strokeWidth="0.8"
|
||||
android:strokeColor="#33FFFFFF" />
|
||||
<path
|
||||
android:fillColor="#00000000"
|
||||
android:pathData="M19,0L19,108"
|
||||
android:strokeWidth="0.8"
|
||||
android:strokeColor="#33FFFFFF" />
|
||||
<path
|
||||
android:fillColor="#00000000"
|
||||
android:pathData="M29,0L29,108"
|
||||
android:strokeWidth="0.8"
|
||||
android:strokeColor="#33FFFFFF" />
|
||||
<path
|
||||
android:fillColor="#00000000"
|
||||
android:pathData="M39,0L39,108"
|
||||
android:strokeWidth="0.8"
|
||||
android:strokeColor="#33FFFFFF" />
|
||||
<path
|
||||
android:fillColor="#00000000"
|
||||
android:pathData="M49,0L49,108"
|
||||
android:strokeWidth="0.8"
|
||||
android:strokeColor="#33FFFFFF" />
|
||||
<path
|
||||
android:fillColor="#00000000"
|
||||
android:pathData="M59,0L59,108"
|
||||
android:strokeWidth="0.8"
|
||||
android:strokeColor="#33FFFFFF" />
|
||||
<path
|
||||
android:fillColor="#00000000"
|
||||
android:pathData="M69,0L69,108"
|
||||
android:strokeWidth="0.8"
|
||||
android:strokeColor="#33FFFFFF" />
|
||||
<path
|
||||
android:fillColor="#00000000"
|
||||
android:pathData="M79,0L79,108"
|
||||
android:strokeWidth="0.8"
|
||||
android:strokeColor="#33FFFFFF" />
|
||||
<path
|
||||
android:fillColor="#00000000"
|
||||
android:pathData="M89,0L89,108"
|
||||
android:strokeWidth="0.8"
|
||||
android:strokeColor="#33FFFFFF" />
|
||||
<path
|
||||
android:fillColor="#00000000"
|
||||
android:pathData="M99,0L99,108"
|
||||
android:strokeWidth="0.8"
|
||||
android:strokeColor="#33FFFFFF" />
|
||||
<path
|
||||
android:fillColor="#00000000"
|
||||
android:pathData="M0,9L108,9"
|
||||
android:strokeWidth="0.8"
|
||||
android:strokeColor="#33FFFFFF" />
|
||||
<path
|
||||
android:fillColor="#00000000"
|
||||
android:pathData="M0,19L108,19"
|
||||
android:strokeWidth="0.8"
|
||||
android:strokeColor="#33FFFFFF" />
|
||||
<path
|
||||
android:fillColor="#00000000"
|
||||
android:pathData="M0,29L108,29"
|
||||
android:strokeWidth="0.8"
|
||||
android:strokeColor="#33FFFFFF" />
|
||||
<path
|
||||
android:fillColor="#00000000"
|
||||
android:pathData="M0,39L108,39"
|
||||
android:strokeWidth="0.8"
|
||||
android:strokeColor="#33FFFFFF" />
|
||||
<path
|
||||
android:fillColor="#00000000"
|
||||
android:pathData="M0,49L108,49"
|
||||
android:strokeWidth="0.8"
|
||||
android:strokeColor="#33FFFFFF" />
|
||||
<path
|
||||
android:fillColor="#00000000"
|
||||
android:pathData="M0,59L108,59"
|
||||
android:strokeWidth="0.8"
|
||||
android:strokeColor="#33FFFFFF" />
|
||||
<path
|
||||
android:fillColor="#00000000"
|
||||
android:pathData="M0,69L108,69"
|
||||
android:strokeWidth="0.8"
|
||||
android:strokeColor="#33FFFFFF" />
|
||||
<path
|
||||
android:fillColor="#00000000"
|
||||
android:pathData="M0,79L108,79"
|
||||
android:strokeWidth="0.8"
|
||||
android:strokeColor="#33FFFFFF" />
|
||||
<path
|
||||
android:fillColor="#00000000"
|
||||
android:pathData="M0,89L108,89"
|
||||
android:strokeWidth="0.8"
|
||||
android:strokeColor="#33FFFFFF" />
|
||||
<path
|
||||
android:fillColor="#00000000"
|
||||
android:pathData="M0,99L108,99"
|
||||
android:strokeWidth="0.8"
|
||||
android:strokeColor="#33FFFFFF" />
|
||||
<path
|
||||
android:fillColor="#00000000"
|
||||
android:pathData="M19,29L89,29"
|
||||
android:strokeWidth="0.8"
|
||||
android:strokeColor="#33FFFFFF" />
|
||||
<path
|
||||
android:fillColor="#00000000"
|
||||
android:pathData="M19,39L89,39"
|
||||
android:strokeWidth="0.8"
|
||||
android:strokeColor="#33FFFFFF" />
|
||||
<path
|
||||
android:fillColor="#00000000"
|
||||
android:pathData="M19,49L89,49"
|
||||
android:strokeWidth="0.8"
|
||||
android:strokeColor="#33FFFFFF" />
|
||||
<path
|
||||
android:fillColor="#00000000"
|
||||
android:pathData="M19,59L89,59"
|
||||
android:strokeWidth="0.8"
|
||||
android:strokeColor="#33FFFFFF" />
|
||||
<path
|
||||
android:fillColor="#00000000"
|
||||
android:pathData="M19,69L89,69"
|
||||
android:strokeWidth="0.8"
|
||||
android:strokeColor="#33FFFFFF" />
|
||||
<path
|
||||
android:fillColor="#00000000"
|
||||
android:pathData="M19,79L89,79"
|
||||
android:strokeWidth="0.8"
|
||||
android:strokeColor="#33FFFFFF" />
|
||||
<path
|
||||
android:fillColor="#00000000"
|
||||
android:pathData="M29,19L29,89"
|
||||
android:strokeWidth="0.8"
|
||||
android:strokeColor="#33FFFFFF" />
|
||||
<path
|
||||
android:fillColor="#00000000"
|
||||
android:pathData="M39,19L39,89"
|
||||
android:strokeWidth="0.8"
|
||||
android:strokeColor="#33FFFFFF" />
|
||||
<path
|
||||
android:fillColor="#00000000"
|
||||
android:pathData="M49,19L49,89"
|
||||
android:strokeWidth="0.8"
|
||||
android:strokeColor="#33FFFFFF" />
|
||||
<path
|
||||
android:fillColor="#00000000"
|
||||
android:pathData="M59,19L59,89"
|
||||
android:strokeWidth="0.8"
|
||||
android:strokeColor="#33FFFFFF" />
|
||||
<path
|
||||
android:fillColor="#00000000"
|
||||
android:pathData="M69,19L69,89"
|
||||
android:strokeWidth="0.8"
|
||||
android:strokeColor="#33FFFFFF" />
|
||||
<path
|
||||
android:fillColor="#00000000"
|
||||
android:pathData="M79,19L79,89"
|
||||
android:strokeWidth="0.8"
|
||||
android:strokeColor="#33FFFFFF" />
|
||||
</vector>
|
||||
6
app/src/main/res/mipmap-anydpi-v26/ic_launcher.xml
Normal file
6
app/src/main/res/mipmap-anydpi-v26/ic_launcher.xml
Normal file
@@ -0,0 +1,6 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<adaptive-icon xmlns:android="http://schemas.android.com/apk/res/android">
|
||||
<background android:drawable="@drawable/ic_launcher_background" />
|
||||
<foreground android:drawable="@drawable/ic_launcher_foreground" />
|
||||
<monochrome android:drawable="@drawable/ic_launcher_foreground" />
|
||||
</adaptive-icon>
|
||||
6
app/src/main/res/mipmap-anydpi-v26/ic_launcher_round.xml
Normal file
6
app/src/main/res/mipmap-anydpi-v26/ic_launcher_round.xml
Normal file
@@ -0,0 +1,6 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<adaptive-icon xmlns:android="http://schemas.android.com/apk/res/android">
|
||||
<background android:drawable="@drawable/ic_launcher_background" />
|
||||
<foreground android:drawable="@drawable/ic_launcher_foreground" />
|
||||
<monochrome android:drawable="@drawable/ic_launcher_foreground" />
|
||||
</adaptive-icon>
|
||||
BIN
app/src/main/res/mipmap-xxxhdpi/ic_launcher.webp
Normal file
BIN
app/src/main/res/mipmap-xxxhdpi/ic_launcher.webp
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 3.8 KiB |
BIN
app/src/main/res/mipmap-xxxhdpi/ic_launcher_round.webp
Normal file
BIN
app/src/main/res/mipmap-xxxhdpi/ic_launcher_round.webp
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 7.6 KiB |
10
app/src/main/res/values/colors.xml
Normal file
10
app/src/main/res/values/colors.xml
Normal file
@@ -0,0 +1,10 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<resources>
|
||||
<color name="purple_200">#FFBB86FC</color>
|
||||
<color name="purple_500">#FF6200EE</color>
|
||||
<color name="purple_700">#FF3700B3</color>
|
||||
<color name="teal_200">#FF03DAC5</color>
|
||||
<color name="teal_700">#FF018786</color>
|
||||
<color name="black">#FF000000</color>
|
||||
<color name="white">#FFFFFFFF</color>
|
||||
</resources>
|
||||
3
app/src/main/res/values/strings.xml
Normal file
3
app/src/main/res/values/strings.xml
Normal file
@@ -0,0 +1,3 @@
|
||||
<resources>
|
||||
<string name="app_name">AiTra</string>
|
||||
</resources>
|
||||
5
app/src/main/res/values/themes.xml
Normal file
5
app/src/main/res/values/themes.xml
Normal file
@@ -0,0 +1,5 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<resources>
|
||||
|
||||
<style name="Theme.AiTra" parent="android:Theme.Material.Light.NoActionBar" />
|
||||
</resources>
|
||||
13
app/src/main/res/xml/backup_rules.xml
Normal file
13
app/src/main/res/xml/backup_rules.xml
Normal file
@@ -0,0 +1,13 @@
|
||||
<?xml version="1.0" encoding="utf-8"?><!--
|
||||
Sample backup rules file; uncomment and customize as necessary.
|
||||
See https://developer.android.com/guide/topics/data/autobackup
|
||||
for details.
|
||||
Note: This file is ignored for devices older that API 31
|
||||
See https://developer.android.com/about/versions/12/backup-restore
|
||||
-->
|
||||
<full-backup-content>
|
||||
<!--
|
||||
<include domain="sharedpref" path="."/>
|
||||
<exclude domain="sharedpref" path="device.xml"/>
|
||||
-->
|
||||
</full-backup-content>
|
||||
19
app/src/main/res/xml/data_extraction_rules.xml
Normal file
19
app/src/main/res/xml/data_extraction_rules.xml
Normal file
@@ -0,0 +1,19 @@
|
||||
<?xml version="1.0" encoding="utf-8"?><!--
|
||||
Sample data extraction rules file; uncomment and customize as necessary.
|
||||
See https://developer.android.com/about/versions/12/backup-restore#xml-changes
|
||||
for details.
|
||||
-->
|
||||
<data-extraction-rules>
|
||||
<cloud-backup>
|
||||
<!-- TODO: Use <include> and <exclude> to control what is backed up.
|
||||
<include .../>
|
||||
<exclude .../>
|
||||
-->
|
||||
</cloud-backup>
|
||||
<!--
|
||||
<device-transfer>
|
||||
<include .../>
|
||||
<exclude .../>
|
||||
</device-transfer>
|
||||
-->
|
||||
</data-extraction-rules>
|
||||
Reference in New Issue
Block a user