Compare commits
17 Commits
Author | SHA1 | Date |
---|---|---|
|
db2f338acd | |
|
c9d15ccc99 | |
|
2026845237 | |
|
e120975f26 | |
|
89eb7a13fa | |
|
7459af2f0a | |
|
636375fe15 | |
|
d721fa025c | |
|
6971784f98 | |
|
c0480c12e6 | |
|
817e7bf467 | |
|
821f00386e | |
|
e65ed84e03 | |
|
cdc13082de | |
|
2426f6b8bd | |
|
bb87a3d138 | |
|
b535a7667d |
|
@ -0,0 +1,17 @@
|
||||||
|
*.iml
|
||||||
|
.gradle
|
||||||
|
/local.properties
|
||||||
|
/.idea/caches
|
||||||
|
/.idea/libraries
|
||||||
|
/.idea/modules.xml
|
||||||
|
/.idea/workspace.xml
|
||||||
|
/.idea/navEditor.xml
|
||||||
|
/.idea/assetWizardSettings.xml
|
||||||
|
.DS_Store
|
||||||
|
/build
|
||||||
|
*/build/
|
||||||
|
**/build/
|
||||||
|
/captures
|
||||||
|
.externalNativeBuild
|
||||||
|
.cxx
|
||||||
|
local.properties
|
|
@ -0,0 +1,3 @@
|
||||||
|
# Default ignored files
|
||||||
|
/shelf/
|
||||||
|
/workspace.xml
|
|
@ -0,0 +1,6 @@
|
||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<project version="4">
|
||||||
|
<component name="AndroidProjectSystem">
|
||||||
|
<option name="providerId" value="com.android.tools.idea.GradleProjectSystem" />
|
||||||
|
</component>
|
||||||
|
</project>
|
|
@ -0,0 +1,123 @@
|
||||||
|
<component name="ProjectCodeStyleConfiguration">
|
||||||
|
<code_scheme name="Project" version="173">
|
||||||
|
<JetCodeStyleSettings>
|
||||||
|
<option name="CODE_STYLE_DEFAULTS" value="KOTLIN_OFFICIAL" />
|
||||||
|
</JetCodeStyleSettings>
|
||||||
|
<codeStyleSettings language="XML">
|
||||||
|
<option name="FORCE_REARRANGE_MODE" value="1" />
|
||||||
|
<indentOptions>
|
||||||
|
<option name="CONTINUATION_INDENT_SIZE" value="4" />
|
||||||
|
</indentOptions>
|
||||||
|
<arrangement>
|
||||||
|
<rules>
|
||||||
|
<section>
|
||||||
|
<rule>
|
||||||
|
<match>
|
||||||
|
<AND>
|
||||||
|
<NAME>xmlns:android</NAME>
|
||||||
|
<XML_ATTRIBUTE />
|
||||||
|
<XML_NAMESPACE>^$</XML_NAMESPACE>
|
||||||
|
</AND>
|
||||||
|
</match>
|
||||||
|
</rule>
|
||||||
|
</section>
|
||||||
|
<section>
|
||||||
|
<rule>
|
||||||
|
<match>
|
||||||
|
<AND>
|
||||||
|
<NAME>xmlns:.*</NAME>
|
||||||
|
<XML_ATTRIBUTE />
|
||||||
|
<XML_NAMESPACE>^$</XML_NAMESPACE>
|
||||||
|
</AND>
|
||||||
|
</match>
|
||||||
|
<order>BY_NAME</order>
|
||||||
|
</rule>
|
||||||
|
</section>
|
||||||
|
<section>
|
||||||
|
<rule>
|
||||||
|
<match>
|
||||||
|
<AND>
|
||||||
|
<NAME>.*:id</NAME>
|
||||||
|
<XML_ATTRIBUTE />
|
||||||
|
<XML_NAMESPACE>http://schemas.android.com/apk/res/android</XML_NAMESPACE>
|
||||||
|
</AND>
|
||||||
|
</match>
|
||||||
|
</rule>
|
||||||
|
</section>
|
||||||
|
<section>
|
||||||
|
<rule>
|
||||||
|
<match>
|
||||||
|
<AND>
|
||||||
|
<NAME>.*:name</NAME>
|
||||||
|
<XML_ATTRIBUTE />
|
||||||
|
<XML_NAMESPACE>http://schemas.android.com/apk/res/android</XML_NAMESPACE>
|
||||||
|
</AND>
|
||||||
|
</match>
|
||||||
|
</rule>
|
||||||
|
</section>
|
||||||
|
<section>
|
||||||
|
<rule>
|
||||||
|
<match>
|
||||||
|
<AND>
|
||||||
|
<NAME>name</NAME>
|
||||||
|
<XML_ATTRIBUTE />
|
||||||
|
<XML_NAMESPACE>^$</XML_NAMESPACE>
|
||||||
|
</AND>
|
||||||
|
</match>
|
||||||
|
</rule>
|
||||||
|
</section>
|
||||||
|
<section>
|
||||||
|
<rule>
|
||||||
|
<match>
|
||||||
|
<AND>
|
||||||
|
<NAME>style</NAME>
|
||||||
|
<XML_ATTRIBUTE />
|
||||||
|
<XML_NAMESPACE>^$</XML_NAMESPACE>
|
||||||
|
</AND>
|
||||||
|
</match>
|
||||||
|
</rule>
|
||||||
|
</section>
|
||||||
|
<section>
|
||||||
|
<rule>
|
||||||
|
<match>
|
||||||
|
<AND>
|
||||||
|
<NAME>.*</NAME>
|
||||||
|
<XML_ATTRIBUTE />
|
||||||
|
<XML_NAMESPACE>^$</XML_NAMESPACE>
|
||||||
|
</AND>
|
||||||
|
</match>
|
||||||
|
<order>BY_NAME</order>
|
||||||
|
</rule>
|
||||||
|
</section>
|
||||||
|
<section>
|
||||||
|
<rule>
|
||||||
|
<match>
|
||||||
|
<AND>
|
||||||
|
<NAME>.*</NAME>
|
||||||
|
<XML_ATTRIBUTE />
|
||||||
|
<XML_NAMESPACE>http://schemas.android.com/apk/res/android</XML_NAMESPACE>
|
||||||
|
</AND>
|
||||||
|
</match>
|
||||||
|
<order>ANDROID_ATTRIBUTE_ORDER</order>
|
||||||
|
</rule>
|
||||||
|
</section>
|
||||||
|
<section>
|
||||||
|
<rule>
|
||||||
|
<match>
|
||||||
|
<AND>
|
||||||
|
<NAME>.*</NAME>
|
||||||
|
<XML_ATTRIBUTE />
|
||||||
|
<XML_NAMESPACE>.*</XML_NAMESPACE>
|
||||||
|
</AND>
|
||||||
|
</match>
|
||||||
|
<order>BY_NAME</order>
|
||||||
|
</rule>
|
||||||
|
</section>
|
||||||
|
</rules>
|
||||||
|
</arrangement>
|
||||||
|
</codeStyleSettings>
|
||||||
|
<codeStyleSettings language="kotlin">
|
||||||
|
<option name="CODE_STYLE_DEFAULTS" value="KOTLIN_OFFICIAL" />
|
||||||
|
</codeStyleSettings>
|
||||||
|
</code_scheme>
|
||||||
|
</component>
|
|
@ -0,0 +1,5 @@
|
||||||
|
<component name="ProjectCodeStyleConfiguration">
|
||||||
|
<state>
|
||||||
|
<option name="USE_PER_PROJECT_SETTINGS" value="true" />
|
||||||
|
</state>
|
||||||
|
</component>
|
|
@ -0,0 +1,6 @@
|
||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<project version="4">
|
||||||
|
<component name="CompilerConfiguration">
|
||||||
|
<bytecodeTargetLevel target="21" />
|
||||||
|
</component>
|
||||||
|
</project>
|
|
@ -0,0 +1,10 @@
|
||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<project version="4">
|
||||||
|
<component name="deploymentTargetSelector">
|
||||||
|
<selectionStates>
|
||||||
|
<SelectionState runConfigName="app">
|
||||||
|
<option name="selectionMode" value="DROPDOWN" />
|
||||||
|
</SelectionState>
|
||||||
|
</selectionStates>
|
||||||
|
</component>
|
||||||
|
</project>
|
|
@ -0,0 +1,13 @@
|
||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<project version="4">
|
||||||
|
<component name="DeviceTable">
|
||||||
|
<option name="columnSorters">
|
||||||
|
<list>
|
||||||
|
<ColumnSorterState>
|
||||||
|
<option name="column" value="Name" />
|
||||||
|
<option name="order" value="ASCENDING" />
|
||||||
|
</ColumnSorterState>
|
||||||
|
</list>
|
||||||
|
</option>
|
||||||
|
</component>
|
||||||
|
</project>
|
|
@ -0,0 +1,35 @@
|
||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<project version="4">
|
||||||
|
<component name="GradleMigrationSettings" migrationVersion="1" />
|
||||||
|
<component name="GradleSettings">
|
||||||
|
<option name="linkedExternalProjectsSettings">
|
||||||
|
<GradleProjectSettings>
|
||||||
|
<compositeConfiguration>
|
||||||
|
<compositeBuild compositeDefinitionSource="SCRIPT">
|
||||||
|
<builds>
|
||||||
|
<build path="$PROJECT_DIR$/buildSrc" name="buildSrc">
|
||||||
|
<projects>
|
||||||
|
<project path="$PROJECT_DIR$/buildSrc" />
|
||||||
|
</projects>
|
||||||
|
</build>
|
||||||
|
</builds>
|
||||||
|
</compositeBuild>
|
||||||
|
</compositeConfiguration>
|
||||||
|
<option name="testRunner" value="CHOOSE_PER_TEST" />
|
||||||
|
<option name="externalProjectPath" value="$PROJECT_DIR$" />
|
||||||
|
<option name="gradleJvm" value="#GRADLE_LOCAL_JAVA_HOME" />
|
||||||
|
<option name="modules">
|
||||||
|
<set>
|
||||||
|
<option value="$PROJECT_DIR$" />
|
||||||
|
<option value="$PROJECT_DIR$/app" />
|
||||||
|
<option value="$PROJECT_DIR$/buildSrc" />
|
||||||
|
<option value="$PROJECT_DIR$/loadingstateview" />
|
||||||
|
<option value="$PROJECT_DIR$/loadingstateview-ktx" />
|
||||||
|
<option value="$PROJECT_DIR$/viewbinding-base" />
|
||||||
|
<option value="$PROJECT_DIR$/viewbinding-nonreflection-ktx" />
|
||||||
|
</set>
|
||||||
|
</option>
|
||||||
|
</GradleProjectSettings>
|
||||||
|
</option>
|
||||||
|
</component>
|
||||||
|
</project>
|
|
@ -0,0 +1,6 @@
|
||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<project version="4">
|
||||||
|
<component name="KotlinJpsPluginSettings">
|
||||||
|
<option name="version" value="2.0.21" />
|
||||||
|
</component>
|
||||||
|
</project>
|
|
@ -0,0 +1,10 @@
|
||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<project version="4">
|
||||||
|
<component name="ProjectMigrations">
|
||||||
|
<option name="MigrateToGradleLocalJavaHome">
|
||||||
|
<set>
|
||||||
|
<option value="$PROJECT_DIR$" />
|
||||||
|
</set>
|
||||||
|
</option>
|
||||||
|
</component>
|
||||||
|
</project>
|
|
@ -0,0 +1,9 @@
|
||||||
|
<project version="4">
|
||||||
|
<component name="ExternalStorageConfigurationManager" enabled="true" />
|
||||||
|
<component name="ProjectRootManager" version="2" languageLevel="JDK_21" default="true" project-jdk-name="jbr-21" project-jdk-type="JavaSDK">
|
||||||
|
<output url="file://$PROJECT_DIR$/build/classes" />
|
||||||
|
</component>
|
||||||
|
<component name="ProjectType">
|
||||||
|
<option name="id" value="Android" />
|
||||||
|
</component>
|
||||||
|
</project>
|
|
@ -0,0 +1,17 @@
|
||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<project version="4">
|
||||||
|
<component name="RunConfigurationProducerService">
|
||||||
|
<option name="ignoredProducers">
|
||||||
|
<set>
|
||||||
|
<option value="com.intellij.execution.junit.AbstractAllInDirectoryConfigurationProducer" />
|
||||||
|
<option value="com.intellij.execution.junit.AllInPackageConfigurationProducer" />
|
||||||
|
<option value="com.intellij.execution.junit.PatternConfigurationProducer" />
|
||||||
|
<option value="com.intellij.execution.junit.TestInClassConfigurationProducer" />
|
||||||
|
<option value="com.intellij.execution.junit.UniqueIdConfigurationProducer" />
|
||||||
|
<option value="com.intellij.execution.junit.testDiscovery.JUnitTestDiscoveryConfigurationProducer" />
|
||||||
|
<option value="org.jetbrains.kotlin.idea.junit.KotlinJUnitRunConfigurationProducer" />
|
||||||
|
<option value="org.jetbrains.kotlin.idea.junit.KotlinPatternConfigurationProducer" />
|
||||||
|
</set>
|
||||||
|
</option>
|
||||||
|
</component>
|
||||||
|
</project>
|
|
@ -0,0 +1,6 @@
|
||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<project version="4">
|
||||||
|
<component name="VcsDirectoryMappings">
|
||||||
|
<mapping directory="$PROJECT_DIR$/.." vcs="Git" />
|
||||||
|
</component>
|
||||||
|
</project>
|
|
@ -0,0 +1 @@
|
||||||
|
/build
|
|
@ -0,0 +1,310 @@
|
||||||
|
import com.android.build.api.dsl.ApplicationProductFlavor
|
||||||
|
|
||||||
|
plugins {
|
||||||
|
id("com.android.application")
|
||||||
|
kotlin("android")
|
||||||
|
id("kotlin-parcelize")
|
||||||
|
kotlin("kapt")
|
||||||
|
id("dagger.hilt.android.plugin")
|
||||||
|
//TODO - enable later: id("com.google.gms.google-services")
|
||||||
|
id("com.google.firebase.crashlytics")
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
android {
|
||||||
|
namespace = Version.applicationId
|
||||||
|
compileSdk = 36
|
||||||
|
|
||||||
|
|
||||||
|
defaultConfig {
|
||||||
|
applicationId = Version.applicationId
|
||||||
|
minSdk = Version.minSdk
|
||||||
|
targetSdk = Version.targetSdk
|
||||||
|
compileSdk = Version.targetSdk
|
||||||
|
versionCode = Version.versionCode
|
||||||
|
versionName = Version.versionName
|
||||||
|
testInstrumentationRunner = "androidx.test.runner.AndroidJUnitRunner"
|
||||||
|
setProperty("archivesBaseName", "${Version.applicationId}-build${Version.versionCode}")
|
||||||
|
multiDexEnabled = true
|
||||||
|
ndk {
|
||||||
|
abiFilters += listOf("armeabi", "armeabi-v7a", "arm64-v8a", "x86", "x86_64")
|
||||||
|
}
|
||||||
|
renderscriptTargetApi = 23
|
||||||
|
renderscriptSupportModeEnabled = true
|
||||||
|
}
|
||||||
|
|
||||||
|
kapt {
|
||||||
|
correctErrorTypes = true
|
||||||
|
arguments {
|
||||||
|
arg("AROUTER_MODULE_NAME", project.name)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
hilt {
|
||||||
|
enableExperimentalClasspathAggregation = true
|
||||||
|
enableAggregatingTask = false
|
||||||
|
}
|
||||||
|
|
||||||
|
buildFeatures {
|
||||||
|
dataBinding = true
|
||||||
|
viewBinding = true
|
||||||
|
}
|
||||||
|
|
||||||
|
kotlinOptions {
|
||||||
|
jvmTarget = JavaVersion.VERSION_17.toString()
|
||||||
|
}
|
||||||
|
|
||||||
|
signingConfigs {
|
||||||
|
val storeFileName = "storeFile"
|
||||||
|
val storePasswordName = "KEYSTORE_PWD"
|
||||||
|
val keyAliasName = "KEY_ALIAS"
|
||||||
|
val keyPasswordName = "KEY_PWD"
|
||||||
|
|
||||||
|
fun names(suffix: String = "") =
|
||||||
|
listOf(storeFileName + suffix, storePasswordName + suffix, keyAliasName + suffix, keyPasswordName + suffix)
|
||||||
|
|
||||||
|
create("release") {
|
||||||
|
storeFile = file(project.properties[names()[0]]?.toString() ?: "")
|
||||||
|
storePassword = project.properties[names()[1]] as? String?
|
||||||
|
keyAlias = project.properties[names()[2]] as? String?
|
||||||
|
keyPassword = project.properties[names()[3]] as? String?
|
||||||
|
}
|
||||||
|
getByName("debug") {
|
||||||
|
storeFile = file(project.properties[names()[0]]?.toString() ?: "")
|
||||||
|
storePassword = project.properties[names()[1]] as? String?
|
||||||
|
keyAlias = project.properties[names()[2]] as? String?
|
||||||
|
keyPassword = project.properties[names()[3]] as? String?
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
buildTypes {
|
||||||
|
release {
|
||||||
|
isMinifyEnabled = false
|
||||||
|
isShrinkResources = false
|
||||||
|
proguardFiles(
|
||||||
|
getDefaultProguardFile("proguard-android-optimize.txt"),
|
||||||
|
"proguard-rules.pro"
|
||||||
|
)
|
||||||
|
}
|
||||||
|
debug {
|
||||||
|
isMinifyEnabled = false
|
||||||
|
isShrinkResources = false
|
||||||
|
proguardFiles(
|
||||||
|
getDefaultProguardFile("proguard-android-optimize.txt"),
|
||||||
|
"proguard-rules.pro"
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
compileOptions {
|
||||||
|
isCoreLibraryDesugaringEnabled = true
|
||||||
|
targetCompatibility(JavaVersion.VERSION_17)
|
||||||
|
sourceCompatibility(JavaVersion.VERSION_17)
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
val flavorDimensionName = "VisualNovel"
|
||||||
|
flavorDimensions.add(flavorDimensionName)
|
||||||
|
productFlavors {
|
||||||
|
fun ApplicationProductFlavor.buildConfigString(name: String, value: String) =
|
||||||
|
buildConfigField("String", name, "\"$value\"")
|
||||||
|
|
||||||
|
fun ApplicationProductFlavor.buildConfigBoolean(name: String, value: String) =
|
||||||
|
buildConfigField("Boolean", name, value)
|
||||||
|
|
||||||
|
create("novelTest") {
|
||||||
|
dimension = flavorDimensionName
|
||||||
|
signingConfig = signingConfigs.getByName("debug")
|
||||||
|
|
||||||
|
buildConfigString("HOST", "https://www.xxxxx.ai/")
|
||||||
|
buildConfigString("ABOUT_US", "https://www.xxxxx.ai/about")
|
||||||
|
buildConfigString("API_FROG", "https://www.test-frog.xxxxx.ai")
|
||||||
|
buildConfigString("EPAL_TERMS_SERVICES", "https://www.xxxxx.ai/policy/tos")
|
||||||
|
buildConfigString("API_SHARK", "https://test-shark.xxxxx.ai")
|
||||||
|
buildConfigString("API_COW", "https://test-cow.xxxxx.ai")
|
||||||
|
buildConfigString("API_PIGEON", "https://test-pigeon.xxxx.ai")
|
||||||
|
buildConfigString("API_LION", "https://test-lion.xxxx.ai")
|
||||||
|
buildConfigString("RECHAEGE_SERVICES", "https://test.xxxxx.ai/policy/recharge")
|
||||||
|
|
||||||
|
buildConfigString("RTC_APP_ID", "689ade491323ae01797818e0-XXX-TODO")
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
create("product") {
|
||||||
|
dimension = flavorDimensionName
|
||||||
|
signingConfig = signingConfigs.getByName("release")
|
||||||
|
|
||||||
|
buildConfigString("HOST", "https://test.xxxxx.ai/")
|
||||||
|
buildConfigString("ABOUT_US", "https://test.xxxxx.ai/about")
|
||||||
|
buildConfigString("API_FROG", "https://test-frog.xxxxx.ai")
|
||||||
|
buildConfigString("EPAL_TERMS_SERVICES", "https://test.xxxxx.ai/policy/tos")
|
||||||
|
buildConfigString("API_SHARK", "https://test-shark.xxxxx.ai")
|
||||||
|
buildConfigString("API_COW", "https://test-cow.xxxxx.ai")
|
||||||
|
buildConfigString("API_PIGEON", "https://test-pigeon.xxxx.ai")
|
||||||
|
buildConfigString("API_LION", "https://test-lion.xxxx.ai")
|
||||||
|
buildConfigString("RECHAEGE_SERVICES", "https://test.xxxxx.ai/policy/recharge")
|
||||||
|
|
||||||
|
buildConfigString("RTC_APP_ID", "689ade491323ae01797818e0-XXX-TODO")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
dependencies {
|
||||||
|
implementation(fileTree("dir" to "libs", "include" to listOf("*.jar", "*.aar")))
|
||||||
|
coreLibraryDesugaring("com.android.tools:desugar_jdk_libs:2.1.5")
|
||||||
|
|
||||||
|
coreLibraryDesugaring(Deps.desugar)
|
||||||
|
implementation(libs.androidx.core.ktx)
|
||||||
|
implementation(libs.androidx.appcompat)
|
||||||
|
implementation(libs.material)
|
||||||
|
implementation(libs.androidx.activity)
|
||||||
|
implementation(libs.androidx.constraintlayout)
|
||||||
|
testImplementation(libs.junit)
|
||||||
|
androidTestImplementation(libs.androidx.junit)
|
||||||
|
androidTestImplementation(libs.androidx.espresso.core)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
//dagger hilt
|
||||||
|
implementation(Deps.hilt)
|
||||||
|
kapt(Deps.hiltAndroidCompiler)
|
||||||
|
kapt(Deps.hiltCompiler)
|
||||||
|
|
||||||
|
//Retrofit2
|
||||||
|
implementation(Deps.retrofit2)
|
||||||
|
implementation(Deps.retrofitAdapterRxjava2)
|
||||||
|
implementation(Deps.retrofit2ConverterGson)
|
||||||
|
implementation(Deps.okhttp3LoggingInterceptor)
|
||||||
|
|
||||||
|
|
||||||
|
// timber used for: recording logs
|
||||||
|
implementation(Deps.timber)
|
||||||
|
|
||||||
|
|
||||||
|
// tecent mmkv
|
||||||
|
implementation(Deps.mmkv)
|
||||||
|
implementation(Deps.material)
|
||||||
|
|
||||||
|
// gson
|
||||||
|
implementation(Deps.gson)
|
||||||
|
|
||||||
|
// viewpager2
|
||||||
|
implementation(Deps.viewpager2)
|
||||||
|
|
||||||
|
// for multiple dex support
|
||||||
|
implementation(Deps.multidex)
|
||||||
|
|
||||||
|
|
||||||
|
//google firebase
|
||||||
|
implementation(platform(Deps.firebaseBom))
|
||||||
|
implementation(Deps.firebaseMessageKtx)
|
||||||
|
implementation(Deps.firebaseAnalyticsKtx)
|
||||||
|
implementation(Deps.firebaseCrashlyticsKtx)
|
||||||
|
implementation(Deps.firebaseAuthKtx)
|
||||||
|
implementation(Deps.credentials)
|
||||||
|
implementation(Deps.credentialsAuth)
|
||||||
|
implementation(Deps.googleId)
|
||||||
|
|
||||||
|
|
||||||
|
// coroutine support
|
||||||
|
implementation(Deps.kotlinCoroutinesCore)
|
||||||
|
implementation(Deps.kotlinCoroutinesAndroid)
|
||||||
|
|
||||||
|
implementation(Deps.appcompat)
|
||||||
|
implementation(Deps.constraintlayout)
|
||||||
|
implementation(Deps.flexbox)
|
||||||
|
implementation(Deps.coreKtx)
|
||||||
|
implementation(Deps.activityKtx)
|
||||||
|
implementation(Deps.activityCompose)
|
||||||
|
implementation(Deps.fragmentKtx)
|
||||||
|
|
||||||
|
// vm and lifecycle
|
||||||
|
implementation(Deps.viewModel)
|
||||||
|
implementation(Deps.livedata)
|
||||||
|
implementation(Deps.lifecycleJava8)
|
||||||
|
implementation(Deps.lifecycleRuntime)
|
||||||
|
implementation(Deps.datastore)
|
||||||
|
|
||||||
|
|
||||||
|
//glide
|
||||||
|
implementation(Deps.glide)
|
||||||
|
kapt(Deps.glideCompiler)
|
||||||
|
implementation(Deps.glideTransformations)
|
||||||
|
implementation(Deps.glideWebpdecoder)
|
||||||
|
|
||||||
|
// eventbus
|
||||||
|
implementation(Deps.modularEventbus)
|
||||||
|
kapt(Deps.modularEventbusCompiler)
|
||||||
|
|
||||||
|
// indicator
|
||||||
|
implementation(Deps.magicIndicator)
|
||||||
|
|
||||||
|
// BlurView and Luban
|
||||||
|
implementation(Deps.Luban)
|
||||||
|
implementation(Deps.BlurView)
|
||||||
|
|
||||||
|
//ali aRouter
|
||||||
|
implementation(Deps.arouter)
|
||||||
|
kapt(Deps.arouterCompiler)
|
||||||
|
|
||||||
|
|
||||||
|
//Permission
|
||||||
|
implementation(Deps.permission)
|
||||||
|
// lottie
|
||||||
|
implementation(Deps.lottie)
|
||||||
|
// float window
|
||||||
|
implementation(Deps.easyFloat)
|
||||||
|
// guide
|
||||||
|
implementation(Deps.newbieGuide)
|
||||||
|
// refreshLayout
|
||||||
|
implementation(Deps.refreshLayout)
|
||||||
|
// apng
|
||||||
|
implementation(Deps.apng)
|
||||||
|
// softKey board
|
||||||
|
implementation(Deps.skbGlobal)
|
||||||
|
|
||||||
|
|
||||||
|
// baseRecyclerAdapter
|
||||||
|
implementation(Deps.brvah)
|
||||||
|
implementation(Deps.swipeMenuLayout)
|
||||||
|
implementation(Deps.BRV)
|
||||||
|
|
||||||
|
//banner
|
||||||
|
implementation(Deps.banner)
|
||||||
|
|
||||||
|
// spannable
|
||||||
|
implementation(Deps.spannablex)
|
||||||
|
|
||||||
|
// Media related libs
|
||||||
|
implementation(Deps.mp3Recorder)
|
||||||
|
implementation(Deps.photoView)
|
||||||
|
implementation(Deps.transition)
|
||||||
|
implementation(Deps.paging)
|
||||||
|
implementation(Deps.exoplayer)
|
||||||
|
implementation(Deps.subsamplingScaleImageView)
|
||||||
|
|
||||||
|
//s3图片上传 oss
|
||||||
|
implementation(Deps.awsS3)
|
||||||
|
implementation(Deps.awsCore)
|
||||||
|
|
||||||
|
// 网易 云信
|
||||||
|
implementation(Deps.nimBase)
|
||||||
|
implementation(Deps.nimPush)
|
||||||
|
|
||||||
|
//内购 / 充值
|
||||||
|
implementation(Deps.billing)
|
||||||
|
|
||||||
|
// RTC : 实时通信
|
||||||
|
implementation(Deps.BytePlusRTC)
|
||||||
|
|
||||||
|
|
||||||
|
implementation(project(mapOf("path" to ":loadingstateview")))
|
||||||
|
implementation(project(mapOf("path" to ":loadingstateview-ktx")))
|
||||||
|
implementation(project(mapOf("path" to ":viewbinding-base")))
|
||||||
|
implementation(project(mapOf("path" to ":viewbinding-nonreflection-ktx")))
|
||||||
|
}
|
Binary file not shown.
|
@ -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
|
|
@ -0,0 +1,24 @@
|
||||||
|
package com.remax.visualnovel
|
||||||
|
|
||||||
|
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.remax.visualnovel", appContext.packageName)
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,83 @@
|
||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
|
||||||
|
xmlns:tools="http://schemas.android.com/tools">
|
||||||
|
|
||||||
|
<uses-permission android:name="com.android.providers.media.MediaProvider" />
|
||||||
|
<uses-permission android:name="android.permission.EXPAND_STATUS_BAR" />
|
||||||
|
<uses-permission android:name="android.permission.INTERNET" />
|
||||||
|
<uses-permission android:name="android.permission.POST_NOTIFICATIONS" />
|
||||||
|
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" tools:ignore="ScopedStorage" />
|
||||||
|
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />
|
||||||
|
<uses-permission android:name="android.permission.READ_MEDIA_IMAGES" />
|
||||||
|
<uses-permission android:name="android.permission.READ_MEDIA_AUDIO" />
|
||||||
|
<uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION" />
|
||||||
|
<uses-permission android:name="android.permission.ACCESS_FINE_LOCATION" />
|
||||||
|
<uses-permission android:name="android.permission.CAMERA" />
|
||||||
|
<uses-permission android:name="com.android.vending.BILLING" />
|
||||||
|
<uses-permission android:name="android.permission.WAKE_LOCK" />
|
||||||
|
<uses-permission android:name="android.permission.READ_PHONE_STATE" />
|
||||||
|
<uses-permission android:name="android.permission.RECORD_AUDIO" />
|
||||||
|
<uses-permission android:name="android.permission.MODIFY_AUDIO_SETTINGS" />
|
||||||
|
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
|
||||||
|
<uses-permission android:name="android.permission.BLUETOOTH" />
|
||||||
|
<uses-permission android:name="android.permission.ACCESS_WIFI_STATE" />
|
||||||
|
<uses-permission android:name="android.permission.SYSTEM_ALERT_WINDOW" />
|
||||||
|
<uses-permission android:name="android.permission.CHANGE_WIFI_STATE" />
|
||||||
|
<uses-permission android:name="android.permission.FLASHLIGHT" />
|
||||||
|
<uses-permission android:name="android.permission.VIBRATE" />
|
||||||
|
<uses-permission android:name="android.permission.RECEIVE_USER_PRESENT" />
|
||||||
|
<uses-permission android:name="android.permission.FOREGROUND_SERVICE" />
|
||||||
|
<uses-permission android:name="android.permission.BLUETOOTH_CONNECT" />
|
||||||
|
|
||||||
|
<uses-feature android:name="android.hardware.camera" android:required="false" />
|
||||||
|
|
||||||
|
<application
|
||||||
|
android:name=".configs.NovelApplication"
|
||||||
|
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/AppTheme">
|
||||||
|
<activity
|
||||||
|
android:name=".ui.splash.SplashActivity"
|
||||||
|
android:theme="@style/AppTheme.Launcher"
|
||||||
|
android:exported="true" >
|
||||||
|
<intent-filter>
|
||||||
|
<action android:name="android.intent.action.MAIN" />
|
||||||
|
<category android:name="android.intent.category.LAUNCHER" />
|
||||||
|
</intent-filter>
|
||||||
|
</activity>
|
||||||
|
|
||||||
|
|
||||||
|
<activity
|
||||||
|
android:name=".ui.main.MainActivity"
|
||||||
|
android:exported="false" >
|
||||||
|
</activity>
|
||||||
|
|
||||||
|
<activity
|
||||||
|
android:name=".ui.search.SearchActivity"
|
||||||
|
android:exported="false" >
|
||||||
|
</activity>
|
||||||
|
|
||||||
|
<activity
|
||||||
|
android:name=".ui.discussion.DiscussionActivity"
|
||||||
|
android:exported="false" >
|
||||||
|
</activity>
|
||||||
|
|
||||||
|
<activity
|
||||||
|
android:name=".ui.checkin.CheckInActivity"
|
||||||
|
android:exported="false" >
|
||||||
|
</activity>
|
||||||
|
|
||||||
|
<activity
|
||||||
|
android:name=".ui.chat.ChatActivity"
|
||||||
|
android:exported="false" >
|
||||||
|
</activity>
|
||||||
|
|
||||||
|
|
||||||
|
</application>
|
||||||
|
|
||||||
|
</manifest>
|
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
|
@ -0,0 +1,202 @@
|
||||||
|
{
|
||||||
|
"color.context.vip.normal": "@GRA:$glo.deg.ltr,$glo.color.red.20,$glo.color.violet.20,$glo.color.mint.20",
|
||||||
|
"glo.color.orange.10": "$glo.color.orange.10",
|
||||||
|
|
||||||
|
|
||||||
|
"color.primary.normal": "$glo.color.magenta.50",
|
||||||
|
"color.primary.hover": "$glo.color.magenta.40",
|
||||||
|
"color.primary.press": "$glo.color.magenta.60",
|
||||||
|
"color.primary.disabled": "$color.surface.nest.disabled",
|
||||||
|
"color.primary.variant.normal": "$glo.color.magenta.40",
|
||||||
|
"color.primary.variant.hover": "$glo.color.magenta.30",
|
||||||
|
"color.primary.variant.press": "$glo.color.magenta.50",
|
||||||
|
"color.primary.variant.disabled": "$color.surface.nest.disabled",
|
||||||
|
"color.secondary.gradient.normal": "@GRA:$glo.deg.ltr,$glo.color.red.20,$glo.color.violet.20,$glo.color.mint.20",
|
||||||
|
"color.primary.gradient.normal": "@GRA:$glo.deg.ltr,$glo.color.magenta.30,$glo.color.purple.40",
|
||||||
|
"color.primary.gradient.hover": "@GRA:$glo.deg.ltr,$glo.color.magenta.20,$glo.color.purple.30",
|
||||||
|
"color.primary.gradient.press": "@GRA:$glo.deg.ltr,$glo.color.magenta.40,$glo.color.purple.50",
|
||||||
|
"color.primary.gradient.disabled": "$color.surface.nest.disabled",
|
||||||
|
"color.primary.onpic.normal": "$glo.color.violet.40,$glo.transparent.t85",
|
||||||
|
"color.primary.onpic.hover": "$glo.color.violet.30,$glo.transparent.t85",
|
||||||
|
"color.primary.onpic.press": "$glo.color.violet.50,$glo.transparent.t85",
|
||||||
|
"color.important.normal": "$glo.color.red.50",
|
||||||
|
"color.important.hover": "$glo.color.red.40",
|
||||||
|
"color.important.press": "$glo.color.red.60",
|
||||||
|
"color.important.disabled": "$color.surface.nest.disabled",
|
||||||
|
"color.important.variant.normal": "$glo.color.red.40",
|
||||||
|
"color.important.variant.hover": "$glo.color.red.30",
|
||||||
|
"color.important.variant.press": "$glo.color.red.50",
|
||||||
|
"color.important.variant.disabled": "$glo.color.blue.10,$glo.transparent.t25",
|
||||||
|
"color.important.gradient.normal": "@GRA:$glo.deg.ltr,$glo.color.orange.50,$glo.color.red.50",
|
||||||
|
"color.important.gradient.hover": "@GRA:$glo.deg.ltr,$glo.color.orange.40,$glo.color.red.40",
|
||||||
|
"color.important.gradient.press": "@GRA:$glo.deg.ltr,$glo.color.orange.60,$glo.color.red.60",
|
||||||
|
"color.important.gradient.disabled": "$color.surface.nest.disabled",
|
||||||
|
"color.important.onpic.normal": "$glo.color.red.50,$glo.transparent.t85",
|
||||||
|
"color.positive.normal": "$glo.color.mint.60",
|
||||||
|
"color.positive.hover": "$glo.color.mint.50",
|
||||||
|
"color.positive.press": "$glo.color.mint.70",
|
||||||
|
"color.positive.disabled": "$color.surface.nest.disabled",
|
||||||
|
"color.positive.variant.normal": "$glo.color.mint.40",
|
||||||
|
"color.positive.variant.hover": "$glo.color.mint.30",
|
||||||
|
"color.positive.variant.press": "$glo.color.mint.50",
|
||||||
|
"color.positive.variant.disabled": "$glo.color.blue.10,$glo.transparent.t25",
|
||||||
|
"color.positive.gradient.normal": "@GRA:$glo.deg.ltr,$glo.color.green.40,$glo.color.mint.60",
|
||||||
|
"color.positive.gradient.hover": "$glo.border.1",
|
||||||
|
"color.positive.gradient.press": "@GRA:$glo.deg.ltr,$glo.color.green.60,$glo.color.mint.70",
|
||||||
|
"color.positive.gradient.disabled": "$color.surface.nest.disabled",
|
||||||
|
"color.positive.onpic.normal": "$glo.color.mint.60,$glo.transparent.t85",
|
||||||
|
"color.warning.normal": "$glo.color.orange.50",
|
||||||
|
"color.warning.hover": "$glo.color.orange.40",
|
||||||
|
"color.warning.press": "$glo.color.orange.60",
|
||||||
|
"color.warning.disabled": "$color.surface.nest.disabled",
|
||||||
|
"color.warning.variant.normal": "$glo.color.orange.40",
|
||||||
|
"color.warning.variant.hover": "$glo.color.orange.30",
|
||||||
|
"color.warning.variant.press": "$glo.color.orange.50",
|
||||||
|
"color.warning.variant.disabled": "$glo.color.blue.10,$glo.transparent.t25",
|
||||||
|
"color.warning.gradient.normal": "@GRA:$glo.deg.ltr,$glo.color.yellow.40,$glo.color.orange.50",
|
||||||
|
"color.warning.gradient.hover": "@GRA:$glo.deg.ltr,$glo.color.yellow.30,$glo.color.orange.40",
|
||||||
|
"color.warning.gradient.press": "@GRA:$glo.deg.ltr,$glo.color.yellow.50,$glo.color.orange.60",
|
||||||
|
"color.warning.gradient.disabled": "$color.surface.nest.disabled",
|
||||||
|
"color.warning.onpic.normal": "$glo.color.orange.50,$glo.transparent.t85",
|
||||||
|
"color.emphasis.normal": "$glo.color.blue.40",
|
||||||
|
"color.emphasis.hover": "$glo.color.blue.30",
|
||||||
|
"color.emphasis.press": "$glo.color.blue.50",
|
||||||
|
"color.emphasis.disabled": "$color.surface.nest.disabled",
|
||||||
|
"color.emphasis.variant.normal": "$glo.color.blue.30",
|
||||||
|
"color.emphasis.variant.hover": "$glo.color.blue.20",
|
||||||
|
"color.emphasis.variant.press": "$glo.color.blue.40",
|
||||||
|
"color.emphasis.variant.disabled": "$glo.color.blue.10,$glo.transparent.t25",
|
||||||
|
"color.emphasis.grandient.normal": "@GRA:$glo.deg.ltr,$glo.color.sky.30,$glo.color.blue.40",
|
||||||
|
"color.emphasis.grandient.hover": "@GRA:$glo.deg.ltr,$glo.color.sky.20,$glo.color.blue.30",
|
||||||
|
"color.emphasis.grandient.press": "@GRA:$glo.deg.ltr,$glo.color.sky.40,$glo.color.blue.50",
|
||||||
|
"color.emphasis.grandient.disabled": "$color.surface.nest.disabled",
|
||||||
|
"color.emphasis.onpic.normal": "$glo.color.blue.40,$glo.transparent.t85",
|
||||||
|
"color.background.default": "$glo.color.grey.100",
|
||||||
|
"color.background.specialmap": "$glo.color.grey.100",
|
||||||
|
"color.background.district": "$glo.color.black,$glo.transparent.t30",
|
||||||
|
"color.surface.base.normal": "$glo.color.grey.80",
|
||||||
|
"color.surface.base.hover": "$glo.color.grey.70",
|
||||||
|
"color.surface.base.press": "$glo.color.grey.90",
|
||||||
|
"color.surface.base.disabled": "$glo.color.grey.90",
|
||||||
|
"color.surface.base.specialmap.normal": "$glo.color.grey.80",
|
||||||
|
"color.surface.base.specialmap.hover": "$glo.color.grey.70",
|
||||||
|
"color.surface.base.specialmap.press": "$glo.color.grey.90,$glo.transparent.t30",
|
||||||
|
"color.surface.base.specialmap.disabled": "$glo.color.white,$glo.transparent.t8",
|
||||||
|
"color.surface.float.normal": "$glo.color.grey.70",
|
||||||
|
"color.surface.float.hover": "$glo.color.grey.60",
|
||||||
|
"color.surface.float.press": "$glo.color.grey.80",
|
||||||
|
"color.surface.float.disabled": "$glo.color.grey.80",
|
||||||
|
"color.surface.top.normal": "$glo.color.black,$glo.transparent.t65",
|
||||||
|
"color.surface.top.hover": "$glo.color.black,$glo.transparent.t45",
|
||||||
|
"color.surface.top.press": "$glo.color.black,$glo.transparent.t85",
|
||||||
|
"color.surface.top.disabled": "$glo.color.black,$glo.transparent.t30",
|
||||||
|
"color.surface.district.normal": "$glo.color.purple.0,$glo.transparent.t4",
|
||||||
|
"color.surface.district.hover": "$glo.color.purple.0,$glo.transparent.t12",
|
||||||
|
"color.surface.district.press": "$glo.color.black,$glo.transparent.t25",
|
||||||
|
"color.surface.district.disabled": "$glo.color.black,$glo.transparent.t25",
|
||||||
|
"color.surface.nest.normal": "$glo.color.purple.0,$glo.transparent.t8",
|
||||||
|
"color.surface.nest.hover": "$glo.color.purple.0,$glo.transparent.t12",
|
||||||
|
"color.surface.nest.press": "$glo.color.purple.0,$glo.transparent.t4",
|
||||||
|
"color.surface.nest.disabled": "$glo.color.purple.0,$glo.transparent.t4",
|
||||||
|
"color.surface.element.normal": "$color.surface.nest.normal",
|
||||||
|
"color.surface.element.hover": "$color.surface.nest.hover",
|
||||||
|
"color.surface.element.press": "$color.surface.nest.press",
|
||||||
|
"color.surface.element.disabled": "$color.surface.nest.disabled",
|
||||||
|
"color.surface.element.dark.normal": "$glo.color.black,$glo.transparent.t65",
|
||||||
|
"color.surface.element.dark.hover": "$glo.color.black,$glo.transparent.t45",
|
||||||
|
"color.surface.element.dark.press": "$glo.color.black,$glo.transparent.t85",
|
||||||
|
"color.surface.element.dark.disabled": "$glo.color.black,$glo.transparent.t45",
|
||||||
|
"color.surface.element.light.normal": "$glo.color.white,$glo.transparent.t15",
|
||||||
|
"color.surface.element.light.hover": "$glo.color.white,$glo.transparent.t25",
|
||||||
|
"color.surface.element.light.press": "$glo.color.white,$glo.transparent.t8",
|
||||||
|
"color.surface.element.light.disabled": "$glo.color.white,$glo.transparent.t8",
|
||||||
|
"color.surface.white.normal": "$glo.color.white",
|
||||||
|
"color.surface.white.hover": "$glo.color.white,$glo.transparent.t85",
|
||||||
|
"color.surface.white.press": "$glo.color.white,$glo.transparent.t65",
|
||||||
|
"color.surface.white.disabled": "$glo.color.white,$glo.transparent.t45",
|
||||||
|
"color.outline.normal": "$glo.color.purple.0,$glo.transparent.t20",
|
||||||
|
"color.outline.hover": "$glo.color.purple.0,$glo.transparent.t30",
|
||||||
|
"color.outline.press": "$glo.color.purple.0,$glo.transparent.t8",
|
||||||
|
"color.outline.disabled": "$color.surface.element.disabled",
|
||||||
|
"color.overlay.primary": "$glo.color.violet.30,$glo.transparent.t30",
|
||||||
|
"color.overlay.gradient": "@GRA:$glo.deg.ttb,$glo.color.black&$glo.transparent.t0,$glo.color.black&$glo.transparent.t100",
|
||||||
|
"color.overlay.dark": "$glo.color.black,$glo.transparent.t65",
|
||||||
|
"color.overlay.base": "@GRA:$glo.deg.ttb,$glo.color.grey.80&$glo.transparent.t0,$glo.color.grey.80&$glo.transparent.t100",
|
||||||
|
"color.context.subscribe.normal": "@GRA:$glo.deg.ltr,$glo.color.purple.50,$glo.color.violet.50",
|
||||||
|
"color.context.subscribe.hover": "@GRA:$glo.deg.ltr,$glo.color.purple.30,$glo.color.violet.30",
|
||||||
|
"color.context.subscribe.press": "@GRA:$glo.deg.ltr,$glo.color.purple.70,$glo.color.violet.70",
|
||||||
|
"color.context.subscribe.disabled": "$color.surface.nest.disabled",
|
||||||
|
"color.context.legends.normal": "@GRA:$glo.deg.ltr,$glo.color.yellow.20,$glo.color.yellow.60",
|
||||||
|
"color.context.legends.hover": "@GRA:$glo.deg.ltr,$glo.color.yellow.10,$glo.color.yellow.40",
|
||||||
|
"color.context.legends.press": "@GRA:$glo.deg.ltr,$glo.color.yellow.60,$glo.color.yellow.90",
|
||||||
|
"color.context.legends.disabled": "$color.surface.nest.disabled",
|
||||||
|
"color.context.legends.variant.normal": "$glo.color.yellow.20",
|
||||||
|
"color.context.legends.variant.hover": "$glo.color.yellow.10",
|
||||||
|
"color.context.legends.variant.press": "$glo.color.yellow.40",
|
||||||
|
"color.context.legends.variant.disabled": "$color.surface.nest.disabled",
|
||||||
|
"color.txt.primary.normal": "$glo.color.white",
|
||||||
|
"color.txt.primary.hover": "$glo.color.magenta.30",
|
||||||
|
"color.txt.primary.press": "$glo.color.magenta.40",
|
||||||
|
"color.txt.primary.disabled": "$color.txt.disabled",
|
||||||
|
"color.txt.primary.specialmap.normal": "$glo.color.white",
|
||||||
|
"color.txt.primary.specialmap.hover": "$glo.color.white,$glo.transparent.t85",
|
||||||
|
"color.txt.primary.specialmap.press": "$glo.color.white,$glo.transparent.t65",
|
||||||
|
"color.txt.primary.specialmap.disabled": "$glo.color.white,$glo.transparent.t45",
|
||||||
|
"color.txt.secondary.normal": "$glo.color.grey.30",
|
||||||
|
"color.txt.secondary.hover": "$glo.color.magenta.30",
|
||||||
|
"color.txt.secondary.press": "$glo.color.magenta.40",
|
||||||
|
"color.txt.secondary.disabled": "$glo.color.grey.50",
|
||||||
|
"color.txt.tertiary.normal": "$glo.color.grey.40",
|
||||||
|
"color.txt.tertiary.hover": "$glo.color.grey.30",
|
||||||
|
"color.txt.tertiary": "$glo.color.grey.50",
|
||||||
|
"color.txt.tertiary.disabled": "$color.txt.disabled",
|
||||||
|
"color.txt.grass": "$glo.color.grass.40",
|
||||||
|
"color.txt.disabled": "$glo.color.grey.50",
|
||||||
|
"txt.display.xl": "$glo.font.family.display,$glo.font.size.64,$glo.font.weight.regular,$glo.font.lineheight.size64",
|
||||||
|
"txt.display.l": "$glo.font.family.display,$glo.font.size.36,$glo.font.weight.regular,$glo.font.lineheight.size36",
|
||||||
|
"txt.display.m": "$glo.font.family.display,$glo.font.size.24,$glo.font.weight.regular,$glo.font.lineheight.size24",
|
||||||
|
"txt.display.s": "$glo.font.family.display,$glo.font.size.16,$glo.font.weight.regular,$glo.font.lineheight.size16",
|
||||||
|
"txt.headline.l": "$glo.font.family.sys,$glo.font.size.36,$glo.font.weight.bold,$glo.font.lineheight.size36",
|
||||||
|
"txt.headline.m": "$glo.font.family.sys,$glo.font.size.24,$glo.font.weight.bold,$glo.font.lineheight.size24",
|
||||||
|
"txt.headline.s": "$glo.font.family.sys,$glo.font.size.20,$glo.font.weight.bold,$glo.font.lineheight.size20",
|
||||||
|
"txt.title.l": "$glo.font.family.sys,$glo.font.size.20,$glo.font.weight.semibold,$glo.font.lineheight.size20",
|
||||||
|
"txt.title.m": "$glo.font.family.sys,$glo.font.size.18,$glo.font.weight.semibold,$glo.font.lineheight.size18",
|
||||||
|
"txt.title.s": "$glo.font.family.sys,$glo.font.size.16,$glo.font.weight.semibold,$glo.font.lineheight.size16",
|
||||||
|
"txt.bodySemibold.l": "$glo.font.family.sys,$glo.font.size.16,$glo.font.weight.semibold,$glo.font.lineheight.size16",
|
||||||
|
"txt.bodySemibold.m": "$glo.font.family.sys,$glo.font.size.14,$glo.font.weight.semibold,$glo.font.lineheight.size14",
|
||||||
|
"txt.bodySemibold.s": "$glo.font.family.sys,$glo.font.size.12,$glo.font.weight.semibold,$glo.font.lineheight.size12",
|
||||||
|
"txt.body.l": "$glo.font.family.sys,$glo.font.size.16,$glo.font.weight.regular,$glo.font.lineheight.size16",
|
||||||
|
"txt.body.m": "$glo.font.family.sys,$glo.font.size.14,$glo.font.weight.regular,$glo.font.lineheight.size14",
|
||||||
|
"txt.body.s": "$glo.font.family.sys,$glo.font.size.12,$glo.font.weight.regular,$glo.font.lineheight.size12",
|
||||||
|
"txt.label.l": "$glo.font.family.sys,$glo.font.size.16,$glo.font.weight.medium,$glo.font.lineheight.size16",
|
||||||
|
"txt.label.m": "$glo.font.family.sys,$glo.font.size.14,$glo.font.weight.medium,$glo.font.lineheight.size14",
|
||||||
|
"txt.label.s": "$glo.font.family.sys,$glo.font.size.12,$glo.font.weight.medium,$glo.font.lineheight.size12",
|
||||||
|
"txt.numDisplay.xl": "$glo.font.family.numDisplay,$glo.font.size.48,$glo.font.weight.bold,$glo.font.lineheight.size48",
|
||||||
|
"txt.numDisplay.l": "$glo.font.family.numDisplay,$glo.font.size.36,$glo.font.weight.bold,$glo.font.lineheight.size36",
|
||||||
|
"txt.numDisplay.m": "$glo.font.family.numDisplay,$glo.font.size.24,$glo.font.weight.bold,$glo.font.lineheight.size24",
|
||||||
|
"txt.numDisplay.s": "$glo.font.family.numDisplay,$glo.font.size.20,$glo.font.weight.bold,$glo.font.lineheight.size20",
|
||||||
|
"txt.numMonotype.xl": "$glo.font.family.sys,$glo.font.size.20,$glo.font.weight.bold,$glo.font.lineheight.size20",
|
||||||
|
"txt.numMonotype.l": "$glo.font.family.sys,$glo.font.size.18,$glo.font.weight.bold,$glo.font.lineheight.size18",
|
||||||
|
"txt.numMonotype.m": "$glo.font.family.sys,$glo.font.size.16,$glo.font.weight.bold,$glo.font.lineheight.size16",
|
||||||
|
"txt.numMonotype.s": "$glo.font.family.sys,$glo.font.size.14,$glo.font.weight.medium,$glo.font.lineheight.size14",
|
||||||
|
"txt.numMonotype.xs": "$glo.font.family.sys,$glo.font.size.12,$glo.font.weight.medium,$glo.font.lineheight.size12",
|
||||||
|
"shadow.s": "@SHA:$glo.color.black,$glo.transparent.t15,0&0,4",
|
||||||
|
"shadow.m": "@SHA:$glo.color.black,$glo.transparent.t15,0&0,8",
|
||||||
|
"shadow.l": "@SHA:$glo.color.black,$glo.transparent.t15,0&0,16",
|
||||||
|
"radius.xs": "$glo.radius.4",
|
||||||
|
"radius.s": "$glo.radius.8",
|
||||||
|
"radius.m": "$glo.radius.12",
|
||||||
|
"radius.l": "$glo.radius.16",
|
||||||
|
"radius.xl": "$glo.radius.20",
|
||||||
|
"radius.xxl": "$glo.radius.24",
|
||||||
|
"radius.40": "$glo.radius.40",
|
||||||
|
"radius.42": "$glo.radius.42",
|
||||||
|
"radius.80": "$glo.radius.80",
|
||||||
|
"radius.round": "$glo.radius.round",
|
||||||
|
"radius.pill": "$glo.radius.round",
|
||||||
|
"border.divider": "$glo.border.half",
|
||||||
|
"border.s": "$glo.border.1",
|
||||||
|
"border.m": "$glo.border.2",
|
||||||
|
"border.l": "$glo.border.4"
|
||||||
|
}
|
||||||
|
|
|
@ -0,0 +1,72 @@
|
||||||
|
package com.remax.visualnovel.api.factory
|
||||||
|
|
||||||
|
import androidx.core.net.toUri
|
||||||
|
import com.remax.visualnovel.BuildConfig
|
||||||
|
import com.remax.visualnovel.api.interceptor.GlobalInterceptor
|
||||||
|
import okhttp3.OkHttpClient
|
||||||
|
import okhttp3.logging.HttpLoggingInterceptor
|
||||||
|
import retrofit2.Retrofit
|
||||||
|
import retrofit2.adapter.rxjava2.RxJava2CallAdapterFactory
|
||||||
|
import retrofit2.converter.gson.GsonConverterFactory
|
||||||
|
import java.util.concurrent.TimeUnit
|
||||||
|
|
||||||
|
object ServiceFactory {
|
||||||
|
/**
|
||||||
|
* 受信任的host列表
|
||||||
|
*/
|
||||||
|
private val hostNameList: List<String?> by lazy {
|
||||||
|
listOf(
|
||||||
|
BuildConfig.API_FROG,
|
||||||
|
BuildConfig.HOST
|
||||||
|
).map {
|
||||||
|
it.parseHost()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun String.parseHost() = this.toUri().host
|
||||||
|
|
||||||
|
private val retrofit: Retrofit by lazy {
|
||||||
|
createRetrofit()
|
||||||
|
}
|
||||||
|
|
||||||
|
private val okHttpClient: OkHttpClient by lazy {
|
||||||
|
createOkHttpClient()
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 构建OkHttpClient对象
|
||||||
|
*/
|
||||||
|
private fun createOkHttpClient() = OkHttpClient.Builder()
|
||||||
|
.addInterceptor(GlobalInterceptor())
|
||||||
|
.callTimeout(30, TimeUnit.SECONDS)
|
||||||
|
.connectTimeout(30, TimeUnit.SECONDS)
|
||||||
|
.readTimeout(30, TimeUnit.SECONDS)
|
||||||
|
.writeTimeout(30, TimeUnit.SECONDS)
|
||||||
|
.hostnameVerifier { hostname, _ ->
|
||||||
|
hostname in hostNameList
|
||||||
|
}
|
||||||
|
.apply {
|
||||||
|
if (BuildConfig.DEBUG) {
|
||||||
|
val interceptor = HttpLoggingInterceptor()
|
||||||
|
interceptor.level = HttpLoggingInterceptor.Level.BODY
|
||||||
|
addNetworkInterceptor(interceptor)
|
||||||
|
}
|
||||||
|
}.build()
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 构建Retrofit对象
|
||||||
|
*/
|
||||||
|
private fun createRetrofit() = Retrofit.Builder()
|
||||||
|
.baseUrl(BuildConfig.API_FROG)
|
||||||
|
.client(okHttpClient)
|
||||||
|
.addCallAdapterFactory(RxJava2CallAdapterFactory.create())
|
||||||
|
.addConverterFactory(GsonConverterFactory.create())
|
||||||
|
.build()
|
||||||
|
|
||||||
|
inline fun <reified T> createService(): T = create(T::class.java)
|
||||||
|
|
||||||
|
fun <T> create(clazz: Class<T>): T {
|
||||||
|
return retrofit.create(clazz)
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,89 @@
|
||||||
|
package com.remax.visualnovel.api.interceptor
|
||||||
|
|
||||||
|
|
||||||
|
import com.remax.visualnovel.BuildConfig
|
||||||
|
import com.remax.visualnovel.constant.AppConstant
|
||||||
|
import com.remax.visualnovel.manager.login.LoginManager
|
||||||
|
import com.remax.visualnovel.utils.AppUtils
|
||||||
|
import okhttp3.Headers
|
||||||
|
import okhttp3.Interceptor
|
||||||
|
import okhttp3.MediaType.Companion.toMediaType
|
||||||
|
import okhttp3.RequestBody
|
||||||
|
import okhttp3.RequestBody.Companion.toRequestBody
|
||||||
|
import okhttp3.Response
|
||||||
|
import okhttp3.ResponseBody
|
||||||
|
import okhttp3.ResponseBody.Companion.asResponseBody
|
||||||
|
import okio.Buffer
|
||||||
|
import timber.log.Timber
|
||||||
|
import java.io.IOException
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Created by HJW on 2022/10/17
|
||||||
|
*/
|
||||||
|
class GlobalInterceptor : Interceptor {
|
||||||
|
|
||||||
|
override fun intercept(chain: Interceptor.Chain): Response {
|
||||||
|
var request = chain.request()
|
||||||
|
val bodyStr = readBody(chain.request().body)
|
||||||
|
val emptyBody = "{}".toRequestBody("application/json;charset=utf-8".toMediaType())
|
||||||
|
val requestBody = if (bodyStr.isNotBlank()) request.body else emptyBody
|
||||||
|
|
||||||
|
// val platform = "ANDROID_${BuildConfig.VERSION_NAME}"
|
||||||
|
// val userAgent = String.format(
|
||||||
|
// "E-Pal/%s (%s; android %s)",
|
||||||
|
// BuildConfig.VERSION_NAME,
|
||||||
|
// Build.MODEL,
|
||||||
|
// Build.VERSION.RELEASE
|
||||||
|
// )
|
||||||
|
|
||||||
|
val headersBuilder = Headers.Builder()
|
||||||
|
.add("AUTH_TK", LoginManager.token ?: "")
|
||||||
|
.add("AUTH_DID", AppUtils.getAndroidID())
|
||||||
|
.add("platform", AppConstant.APP_CLIENT)
|
||||||
|
.add("versionNum", "100")
|
||||||
|
|
||||||
|
val headers = headersBuilder.build()
|
||||||
|
|
||||||
|
request = chain.request().newBuilder()
|
||||||
|
.headers(headers)
|
||||||
|
.post(requestBody ?: emptyBody)
|
||||||
|
.build()
|
||||||
|
|
||||||
|
val response = chain.proceed(request)
|
||||||
|
|
||||||
|
if (BuildConfig.DEBUG) {
|
||||||
|
try {
|
||||||
|
Timber.tag("发起请求")
|
||||||
|
Timber.d(
|
||||||
|
"""
|
||||||
|
———————————————— 我是开始分割线 ——————————————————————————————
|
||||||
|
${request.url}
|
||||||
|
入参:
|
||||||
|
${readBody(requestBody)}
|
||||||
|
响应:
|
||||||
|
${clone(response.body)?.string()}
|
||||||
|
———————————————— 我是结束分割线 ———————————————————————————————
|
||||||
|
""".trimIndent()
|
||||||
|
)
|
||||||
|
} catch (e: Exception) {
|
||||||
|
Timber.e("GlobalInterceptor request.exception : ${e.localizedMessage}}")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return response
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun readBody(body: RequestBody?): String {
|
||||||
|
val buffer = Buffer()
|
||||||
|
body?.writeTo(buffer)
|
||||||
|
return buffer.readUtf8()
|
||||||
|
}
|
||||||
|
|
||||||
|
@Throws(IOException::class)
|
||||||
|
private fun clone(body: ResponseBody?): ResponseBody? {
|
||||||
|
val source = body?.source()
|
||||||
|
if (source?.request(Long.MAX_VALUE) == true) throw IOException("body too long!")
|
||||||
|
val bufferedCopy = source?.buffer?.clone()
|
||||||
|
return bufferedCopy?.asResponseBody(body.contentType(), body.contentLength())
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,76 @@
|
||||||
|
package com.remax.visualnovel.api.interceptor.util;
|
||||||
|
|
||||||
|
import java.nio.charset.StandardCharsets;
|
||||||
|
|
||||||
|
import javax.crypto.Cipher;
|
||||||
|
import javax.crypto.spec.IvParameterSpec;
|
||||||
|
import javax.crypto.spec.SecretKeySpec;
|
||||||
|
|
||||||
|
import timber.log.Timber;
|
||||||
|
|
||||||
|
public class AES {
|
||||||
|
|
||||||
|
private static final String KEY_SUFFIX = "90e8kDQUIWpXg8Jp";
|
||||||
|
|
||||||
|
/**
|
||||||
|
* For a 128-bit AES key you need 16 bytes.
|
||||||
|
* For a 256-bit AES key you need 32 bytes.
|
||||||
|
*/
|
||||||
|
public static String KEY;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* IV length: must be 16 bytes long
|
||||||
|
*/
|
||||||
|
public static final String IV = "sdf4ddfsFD86Vdf2";
|
||||||
|
|
||||||
|
private AES() {
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void genKey(String token) {
|
||||||
|
//token加密最多截取前32位
|
||||||
|
String encodeStr = (token.length() > 32 ? token.substring(0, 32) : token) + KEY_SUFFIX;
|
||||||
|
KEY = Md5.encode(encodeStr).toUpperCase();
|
||||||
|
Timber.d("genKey: %s", KEY);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static String encrypt(String data) {
|
||||||
|
return encrypt(data, KEY, IV);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static String desEncrypt(String data) {
|
||||||
|
return desEncrypt(data, KEY, IV);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static String encrypt(String data, String key, String initVector) {
|
||||||
|
try {
|
||||||
|
IvParameterSpec iv = new IvParameterSpec(initVector.getBytes(StandardCharsets.UTF_8));
|
||||||
|
SecretKeySpec skeySpec = new SecretKeySpec(key.getBytes(StandardCharsets.UTF_8), "AES");
|
||||||
|
|
||||||
|
Cipher cipher = Cipher.getInstance("AES/CBC/PKCS5PADDING");
|
||||||
|
cipher.init(Cipher.ENCRYPT_MODE, skeySpec, iv);
|
||||||
|
|
||||||
|
byte[] encrypted = cipher.doFinal(data.getBytes());
|
||||||
|
return Base64.encode(encrypted);
|
||||||
|
} catch (Exception ex) {
|
||||||
|
ex.printStackTrace();
|
||||||
|
return data;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public static String desEncrypt(String data, String key, String initVector) {
|
||||||
|
try {
|
||||||
|
byte[] encrypted = Base64.decode(data);
|
||||||
|
Cipher cipher = Cipher.getInstance("AES/CBC/PKCS5PADDING");
|
||||||
|
SecretKeySpec keySpec = new SecretKeySpec(key.getBytes(), "AES");
|
||||||
|
IvParameterSpec ivParameterSpec = new IvParameterSpec(initVector.getBytes());
|
||||||
|
cipher.init(Cipher.DECRYPT_MODE, keySpec, ivParameterSpec);
|
||||||
|
|
||||||
|
byte[] original = cipher.doFinal(encrypted);
|
||||||
|
|
||||||
|
return new String(original);
|
||||||
|
} catch (Exception e) {
|
||||||
|
e.printStackTrace();
|
||||||
|
return data;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,102 @@
|
||||||
|
package com.remax.visualnovel.api.interceptor.util;
|
||||||
|
|
||||||
|
import java.io.UnsupportedEncodingException;
|
||||||
|
|
||||||
|
public class Base64 {
|
||||||
|
|
||||||
|
private static final char[] ENCODE_CHARS = new char[]{'A', 'B', 'C', 'D',
|
||||||
|
'E', 'F', 'G', 'H', 'I', 'J', 'K', 'L', 'M', 'N', 'O', 'P', 'Q',
|
||||||
|
'R', 'S', 'T', 'U', 'V', 'W', 'X', 'Y', 'Z', 'a', 'b', 'c', 'd',
|
||||||
|
'e', 'f', 'g', 'h', 'i', 'j', 'k', 'l', 'm', 'n', 'o', 'p', 'q',
|
||||||
|
'r', 's', 't', 'u', 'v', 'w', 'x', 'y', 'z', '0', '1', '2', '3',
|
||||||
|
'4', '5', '6', '7', '8', '9', '+', '/'};
|
||||||
|
|
||||||
|
private static final byte[] DECODE_CHARS = new byte[]{-1, -1, -1, -1, -1,
|
||||||
|
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
|
||||||
|
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
|
||||||
|
-1, -1, -1, -1, 62, -1, -1, -1, 63, 52, 53, 54, 55, 56, 57, 58, 59,
|
||||||
|
60, 61, -1, -1, -1, -1, -1, -1, -1, 0, 1, 2, 3, 4, 5, 6, 7, 8, 9,
|
||||||
|
10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, -1,
|
||||||
|
-1, -1, -1, -1, -1, 26, 27, 28, 29, 30, 31, 32, 33, 34, 35, 36, 37,
|
||||||
|
38, 39, 40, 41, 42, 43, 44, 45, 46, 47, 48, 49, 50, 51, -1, -1, -1,
|
||||||
|
-1, -1};
|
||||||
|
|
||||||
|
public static String encode(byte[] data) {
|
||||||
|
StringBuffer buffer = new StringBuffer();
|
||||||
|
int length = data.length;
|
||||||
|
int indexx = 0;
|
||||||
|
int b1, b2, b3;
|
||||||
|
while (indexx < length) {
|
||||||
|
b1 = data[indexx++] & 0xff;
|
||||||
|
if (indexx == length) {
|
||||||
|
buffer.append(ENCODE_CHARS[b1 >>> 2]);
|
||||||
|
buffer.append(ENCODE_CHARS[(b1 & 0x3) << 4]);
|
||||||
|
buffer.append("==");
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
b2 = data[indexx++] & 0xff;
|
||||||
|
if (indexx == length) {
|
||||||
|
buffer.append(ENCODE_CHARS[b1 >>> 2]);
|
||||||
|
buffer.append(ENCODE_CHARS[((b1 & 0x03) << 4) | ((b2 & 0xf0) >>> 4)]);
|
||||||
|
buffer.append(ENCODE_CHARS[(b2 & 0x0f) << 2]);
|
||||||
|
buffer.append("=");
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
b3 = data[indexx++] & 0xff;
|
||||||
|
buffer.append(ENCODE_CHARS[b1 >>> 2]);
|
||||||
|
buffer.append(ENCODE_CHARS[((b1 & 0x03) << 4) | ((b2 & 0xf0) >>> 4)]);
|
||||||
|
buffer.append(ENCODE_CHARS[((b2 & 0x0f) << 2) | ((b3 & 0xc0) >>> 6)]);
|
||||||
|
buffer.append(ENCODE_CHARS[b3 & 0x3f]);
|
||||||
|
}
|
||||||
|
return buffer.toString();
|
||||||
|
}
|
||||||
|
|
||||||
|
public static byte[] decode(String str) throws UnsupportedEncodingException {
|
||||||
|
StringBuffer sb = new StringBuffer();
|
||||||
|
byte[] data = str.getBytes("US-ASCII");
|
||||||
|
int len = data.length;
|
||||||
|
int i = 0;
|
||||||
|
int b1, b2, b3, b4;
|
||||||
|
while (i < len) {
|
||||||
|
do {
|
||||||
|
b1 = DECODE_CHARS[data[i++]];
|
||||||
|
} while (i < len && b1 == -1);
|
||||||
|
if (b1 == -1) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
do {
|
||||||
|
b2 = DECODE_CHARS[data[i++]];
|
||||||
|
} while (i < len && b2 == -1);
|
||||||
|
if (b2 == -1) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
sb.append((char) ((b1 << 2) | ((b2 & 0x30) >>> 4)));
|
||||||
|
|
||||||
|
do {
|
||||||
|
b3 = data[i++];
|
||||||
|
if (b3 == 61) {
|
||||||
|
return sb.toString().getBytes("iso8859-1");
|
||||||
|
}
|
||||||
|
b3 = DECODE_CHARS[b3];
|
||||||
|
} while (i < len && b3 == -1);
|
||||||
|
if (b3 == -1) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
sb.append((char) (((b2 & 0x0f) << 4) | ((b3 & 0x3c) >>> 2)));
|
||||||
|
|
||||||
|
do {
|
||||||
|
b4 = data[i++];
|
||||||
|
if (b4 == 61) {
|
||||||
|
return sb.toString().getBytes("iso8859-1");
|
||||||
|
}
|
||||||
|
b4 = DECODE_CHARS[b4];
|
||||||
|
} while (i < len && b4 == -1);
|
||||||
|
if (b4 == -1) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
sb.append((char) (((b3 & 0x03) << 6) | b4));
|
||||||
|
}
|
||||||
|
return sb.toString().getBytes("iso8859-1");
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,37 @@
|
||||||
|
package com.remax.visualnovel.api.interceptor.util;
|
||||||
|
|
||||||
|
import java.security.MessageDigest;
|
||||||
|
import java.security.NoSuchAlgorithmException;
|
||||||
|
|
||||||
|
public class Md5 {
|
||||||
|
|
||||||
|
private Md5() {
|
||||||
|
}
|
||||||
|
|
||||||
|
public static String encode(String str) {
|
||||||
|
try {
|
||||||
|
MessageDigest md = MessageDigest.getInstance("MD5");
|
||||||
|
md.update(str.getBytes());
|
||||||
|
byte[] byteDigest = md.digest();
|
||||||
|
StringBuilder buf = new StringBuilder();
|
||||||
|
|
||||||
|
for (int b : byteDigest) {
|
||||||
|
int i = b;
|
||||||
|
if (i < 0) {
|
||||||
|
i += 256;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (i < 16) {
|
||||||
|
buf.append("0");
|
||||||
|
}
|
||||||
|
|
||||||
|
buf.append(Integer.toHexString(i));
|
||||||
|
}
|
||||||
|
|
||||||
|
return buf.toString();
|
||||||
|
} catch (NoSuchAlgorithmException var6) {
|
||||||
|
var6.printStackTrace();
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,222 @@
|
||||||
|
package com.remax.visualnovel.api.service
|
||||||
|
|
||||||
|
import com.remax.visualnovel.BuildConfig
|
||||||
|
import com.remax.visualnovel.entity.request.AIGenerate
|
||||||
|
import com.remax.visualnovel.entity.request.AIGenerateImage
|
||||||
|
import com.remax.visualnovel.entity.request.AIHeadImgRequest
|
||||||
|
import com.remax.visualnovel.entity.request.AIIDRequest
|
||||||
|
import com.remax.visualnovel.entity.request.AlbumCreate
|
||||||
|
import com.remax.visualnovel.entity.request.AlbumDTO
|
||||||
|
import com.remax.visualnovel.entity.request.CardRequest
|
||||||
|
import com.remax.visualnovel.entity.request.ChatAlbum
|
||||||
|
import com.remax.visualnovel.entity.request.ClassificationRequest
|
||||||
|
import com.remax.visualnovel.entity.request.Gift
|
||||||
|
import com.remax.visualnovel.entity.request.QueryAlbumDTO
|
||||||
|
import com.remax.visualnovel.entity.request.SimpleCountDTO
|
||||||
|
import com.remax.visualnovel.entity.response.Album
|
||||||
|
import com.remax.visualnovel.entity.response.AlbumCreateCountOutput
|
||||||
|
import com.remax.visualnovel.entity.response.AppearanceImage
|
||||||
|
import com.remax.visualnovel.entity.response.Character
|
||||||
|
import com.remax.visualnovel.entity.response.ContentRes
|
||||||
|
import com.remax.visualnovel.entity.response.ExploreInfo
|
||||||
|
import com.remax.visualnovel.entity.response.MeetSdOutput
|
||||||
|
import com.remax.visualnovel.entity.response.Pageable
|
||||||
|
import com.remax.visualnovel.entity.response.base.Response
|
||||||
|
import retrofit2.http.Body
|
||||||
|
import retrofit2.http.POST
|
||||||
|
|
||||||
|
interface AIService {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 卡片上报绑定
|
||||||
|
*/
|
||||||
|
@POST("/web/meet/bd")
|
||||||
|
suspend fun cardBind(@Body request: AIIDRequest): Response<Character>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 卡片被喜欢推荐
|
||||||
|
*/
|
||||||
|
@POST("/web/meet/rc")
|
||||||
|
suspend fun cardLiked(): Response<Album>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 卡片上报
|
||||||
|
*/
|
||||||
|
@POST("/web/meet/sd")
|
||||||
|
suspend fun reportCard(@Body request: CardRequest): Response<MeetSdOutput>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取首页卡片列表
|
||||||
|
*/
|
||||||
|
@POST("/web/home/rm-list")
|
||||||
|
suspend fun getHomeCard(@Body request: ClassificationRequest): Response<List<Character>>
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取单个首页卡片
|
||||||
|
*/
|
||||||
|
@POST(" /web/home/meet-detail")
|
||||||
|
suspend fun getHomeCardDetail(@Body request: AIIDRequest): Response<Character>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取分类列表
|
||||||
|
*/
|
||||||
|
@POST("/web/home/classification-list")
|
||||||
|
suspend fun getClassificationList(@Body request: ClassificationRequest): Response<List<Character>>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取榜单
|
||||||
|
*/
|
||||||
|
@POST("/web/rank/heartbeat")
|
||||||
|
suspend fun getHeartbeatRank(): Response<List<Character>>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取榜单
|
||||||
|
*/
|
||||||
|
@POST("/web/rank/gift")
|
||||||
|
suspend fun getGiftRank(): Response<List<Character>>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取榜单
|
||||||
|
*/
|
||||||
|
@POST("/web/rank/chat")
|
||||||
|
suspend fun getChatRank(): Response<List<Character>>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取发现页顶部数据
|
||||||
|
*/
|
||||||
|
@POST("/web/explore/info")
|
||||||
|
suspend fun getExploreInfo(): Response<ExploreInfo>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 解锁加密图片
|
||||||
|
*/
|
||||||
|
@POST("/web/ai-user/unlock-album-img")
|
||||||
|
suspend fun unlockAlbum(@Body dto: ChatAlbum): Response<Album>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 解锁秘密爱慕者
|
||||||
|
*/
|
||||||
|
@POST("/web/meet/unlock")
|
||||||
|
suspend fun unlockSecret(@Body dto: AIIDRequest): Response<Album>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 设置当前图片价格
|
||||||
|
*/
|
||||||
|
@POST("/web/ai-user/set-album-unlock-price")
|
||||||
|
suspend fun setAlbumUnlockPrice(@Body dto: AlbumDTO): Response<Any>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 删除AI角色
|
||||||
|
*/
|
||||||
|
@POST("/web/ai-user/del")
|
||||||
|
suspend fun deleteAICharacter(@Body request: Character): Response<Any>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 设置当前默认图片
|
||||||
|
*/
|
||||||
|
@POST("/web/ai-user/set-default-album")
|
||||||
|
suspend fun setAlbumDefault(@Body dto: AlbumDTO): Response<Any>
|
||||||
|
|
||||||
|
@POST("/web/ai-user/create-edit")
|
||||||
|
suspend fun createOrEditAICharacter(@Body request: Character): Response<Character>
|
||||||
|
|
||||||
|
@POST("/web/ai-user/edit-head-img")
|
||||||
|
suspend fun editAIAvatar(@Body request: AIHeadImgRequest): Response<Any>
|
||||||
|
|
||||||
|
|
||||||
|
@POST(BuildConfig.API_COW + "/web/gen/user-content-v1")
|
||||||
|
suspend fun generateAICharacter(@Body request: AIGenerate): Response<ContentRes>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 编辑时获取我的ai角色信息
|
||||||
|
*/
|
||||||
|
@POST("/web/ai-user/get-my-ai-user/info")
|
||||||
|
suspend fun getAICharacter(@Body request: Character): Response<Character>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 访问AI个人主页时获取信息
|
||||||
|
*/
|
||||||
|
@POST("/web/ai-user-search/base-info")
|
||||||
|
suspend fun getAICharacterProfile(@Body request: Character): Response<Character>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 访问AI的统计信息
|
||||||
|
*/
|
||||||
|
@POST("/web/ai-user/stat")
|
||||||
|
suspend fun getAICharacterStat(@Body request: Character): Response<Character>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 修改点赞状态
|
||||||
|
*/
|
||||||
|
@POST("/web/ai-user/like-or-cancel")
|
||||||
|
suspend fun setAILikeOrCancel(@Body request: AlbumDTO): Response<Any>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 喜欢或取消喜欢相片
|
||||||
|
*/
|
||||||
|
@POST("/web/album/like_or_cancel")
|
||||||
|
suspend fun setLikeOrDislike(@Body dto: AlbumDTO): Response<Any>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 删除相片
|
||||||
|
*/
|
||||||
|
@POST("/web/ai-user/album-del")
|
||||||
|
suspend fun deleteAlbum(@Body dto: AlbumDTO): Response<Any>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 批量添加图片到相册
|
||||||
|
*/
|
||||||
|
@POST("/web/ai-user/batch-add-album")
|
||||||
|
suspend fun addAlbum(@Body dto: AlbumCreate): Response<Any>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取创作次数
|
||||||
|
*/
|
||||||
|
@POST("/web/user/get-user-create-count")
|
||||||
|
suspend fun getAlbumCreateCount(): Response<AlbumCreateCountOutput>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 购买创作次数
|
||||||
|
*/
|
||||||
|
@POST("/web/ai/buy-create-image-count")
|
||||||
|
suspend fun buyAlbumCreateCount(@Body dto: SimpleCountDTO): Response<Any>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 批量添加图片到聊天背景
|
||||||
|
*/
|
||||||
|
@POST("/web/chat-background/batch-add")
|
||||||
|
suspend fun addChatBackground(@Body dto: AlbumCreate): Response<Any>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取相册 分页
|
||||||
|
*/
|
||||||
|
@POST("/web/ai-user/album-list")
|
||||||
|
suspend fun getAlbumList(@Body dto: QueryAlbumDTO): Response<Pageable<Album>>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取用户礼物 分页
|
||||||
|
*/
|
||||||
|
@POST("/web/ai-user-gift/list")
|
||||||
|
suspend fun getUserGiftList(@Body dto: QueryAlbumDTO): Response<Pageable<Gift>>
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* AI一键生成-创建生成人物形象图片任务
|
||||||
|
*/
|
||||||
|
@POST(BuildConfig.API_COW + "/web/gen/image-ct")
|
||||||
|
suspend fun generateImageBatch(@Body request: AIGenerateImage): Response<AIGenerateImage>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* AI一键生成-删除图片生成任务
|
||||||
|
*/
|
||||||
|
@POST(BuildConfig.API_COW + "/web/gen/del")
|
||||||
|
suspend fun generateImageBatchDel(@Body request: AIGenerateImage): Response<Any>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* AI一键生成-轮询查询图片生成结果
|
||||||
|
*/
|
||||||
|
@POST(BuildConfig.API_COW + "/web/gen/image-pl")
|
||||||
|
suspend fun generateImageBatchQuery(@Body request: AIGenerateImage): Response<List<AppearanceImage>>
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,13 @@
|
||||||
|
package com.remax.visualnovel.api.service
|
||||||
|
|
||||||
|
import com.remax.visualnovel.entity.response.Book
|
||||||
|
import com.remax.visualnovel.entity.response.base.Response
|
||||||
|
import retrofit2.http.POST
|
||||||
|
|
||||||
|
|
||||||
|
interface BookService {
|
||||||
|
@POST("/web/si/asi")
|
||||||
|
suspend fun getBooks(): Response<Book>
|
||||||
|
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,159 @@
|
||||||
|
package com.remax.visualnovel.api.service
|
||||||
|
|
||||||
|
import com.remax.visualnovel.BuildConfig
|
||||||
|
import com.remax.visualnovel.entity.request.AIFeedback
|
||||||
|
import com.remax.visualnovel.entity.request.AIIDRequest
|
||||||
|
import com.remax.visualnovel.entity.request.AIIsShowDTO
|
||||||
|
import com.remax.visualnovel.entity.request.ChatAlbum
|
||||||
|
import com.remax.visualnovel.entity.request.ChatSetting
|
||||||
|
import com.remax.visualnovel.entity.request.HeartbeatBuy
|
||||||
|
import com.remax.visualnovel.entity.request.RTCRequest
|
||||||
|
import com.remax.visualnovel.entity.request.SearchPage
|
||||||
|
import com.remax.visualnovel.entity.request.SimpleDataDTO
|
||||||
|
import com.remax.visualnovel.entity.request.VoiceTTS
|
||||||
|
import com.remax.visualnovel.entity.response.Album
|
||||||
|
import com.remax.visualnovel.entity.response.Character
|
||||||
|
import com.remax.visualnovel.entity.response.ChatBackground
|
||||||
|
import com.remax.visualnovel.entity.response.ChatSet
|
||||||
|
import com.remax.visualnovel.entity.response.Friends
|
||||||
|
import com.remax.visualnovel.entity.response.HeartbeatLevelOutput
|
||||||
|
import com.remax.visualnovel.entity.response.Pageable
|
||||||
|
import com.remax.visualnovel.entity.response.Token
|
||||||
|
import com.remax.visualnovel.entity.response.VoiceASR
|
||||||
|
import com.remax.visualnovel.entity.response.base.Response
|
||||||
|
import retrofit2.http.Body
|
||||||
|
import retrofit2.http.POST
|
||||||
|
|
||||||
|
interface ChatService {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 发送开场白消息
|
||||||
|
*/
|
||||||
|
@POST("/web/chat/send-dialogue-prologue-message")
|
||||||
|
suspend fun sendDialogueMsg(@Body request: AIIDRequest): Response<Any>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取IM中AI的基础信息
|
||||||
|
*/
|
||||||
|
@POST("/web/ai-user-search/im-base-info")
|
||||||
|
suspend fun getIMAICharacterProfile(@Body request: Character): Response<Character>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 访问解锁加密图片
|
||||||
|
*/
|
||||||
|
@POST("/web/ai-user/view-unlock-album-img")
|
||||||
|
suspend fun viewAlbumImg(@Body request: ChatAlbum): Response<Album>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 关系列表
|
||||||
|
*/
|
||||||
|
@POST("/web/ai-user/heartbeat-relation-list")
|
||||||
|
suspend fun getMyFriends(@Body request: SearchPage): Response<Pageable<Friends>>
|
||||||
|
|
||||||
|
|
||||||
|
@POST("/web/ai-user/heartbeat-rank")
|
||||||
|
suspend fun getMyFriendRank(): Response<Double>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 生成提示词
|
||||||
|
*/
|
||||||
|
@POST(BuildConfig.API_COW + "/web/gen/sup-content-v2")
|
||||||
|
suspend fun getPrompts(@Body request: AIIDRequest): Response<List<String>>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* AI回话点赞/点踩
|
||||||
|
*/
|
||||||
|
@POST(BuildConfig.API_PIGEON + "/web/fb/v1")
|
||||||
|
suspend fun aiFeedback(@Body request: AIFeedback): Response<Any>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取RTC
|
||||||
|
*/
|
||||||
|
@POST(BuildConfig.API_COW + "/web/voice-chat/gen-rtc-tk")
|
||||||
|
suspend fun getRTCToken(@Body request: RTCRequest): Response<Token>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 操作通话
|
||||||
|
*/
|
||||||
|
@POST(BuildConfig.API_COW + "/web/voice-chat/opt")
|
||||||
|
suspend fun voiceChatOpt(@Body request: RTCRequest): Response<Any>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取聊天背景列表
|
||||||
|
*/
|
||||||
|
@POST("/web/chat-background/list")
|
||||||
|
suspend fun getChatBackgroundList(@Body request: AIIDRequest): Response<List<ChatBackground>>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取聊天设置
|
||||||
|
*/
|
||||||
|
@POST("/web/chat-set/get-my")
|
||||||
|
suspend fun getChatSetting(@Body request: ChatSetting): Response<ChatSet>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 修改聊天设置
|
||||||
|
*/
|
||||||
|
@POST("/web/chat-set/set")
|
||||||
|
suspend fun setChatSetting(@Body request: ChatSet): Response<Any>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 修改聊天气泡
|
||||||
|
*/
|
||||||
|
@POST("/web/chat-set/set-chat-bubble")
|
||||||
|
suspend fun setChatBubble(@Body request: ChatSetting): Response<Any>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 修改聊天模型
|
||||||
|
*/
|
||||||
|
@POST("/web/chat-set/set-chat-model")
|
||||||
|
suspend fun setChatModel(@Body request: ChatSetting): Response<Any>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 修改是否自动播放语音
|
||||||
|
*/
|
||||||
|
@POST("/web/chat-set/auto-play-voice")
|
||||||
|
suspend fun setChatAutoPlay(@Body request: ChatSetting): Response<Any>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 修改聊天背景图
|
||||||
|
*/
|
||||||
|
@POST("/web/chat-background/set-background")
|
||||||
|
suspend fun setChatBackground(@Body request: ChatSetting): Response<Any>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 删除聊天背景图
|
||||||
|
*/
|
||||||
|
@POST("/web/chat-background/del")
|
||||||
|
suspend fun deleteChatBackground(@Body request: ChatSetting): Response<Any>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 展示心动关系开关
|
||||||
|
*/
|
||||||
|
@POST("/web/ai-user/heartbeat-relation-switch")
|
||||||
|
suspend fun relationSwitch(@Body request: AIIsShowDTO): Response<Any>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 语音转文本
|
||||||
|
*/
|
||||||
|
@POST(BuildConfig.API_COW + "/web/voice/asr-v2")
|
||||||
|
suspend fun voiceASR(@Body request: SimpleDataDTO): Response<VoiceASR>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 生成语音
|
||||||
|
*/
|
||||||
|
@POST(BuildConfig.API_COW + "/web/voice/tts-v2")
|
||||||
|
suspend fun voiceTTS(@Body request: VoiceTTS): Response<String>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取心动等级
|
||||||
|
*/
|
||||||
|
@POST("/web/ai-user/heartbeat-level")
|
||||||
|
suspend fun getHeartbeatLevel(@Body request: Character): Response<HeartbeatLevelOutput>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 购买心动值
|
||||||
|
*/
|
||||||
|
@POST("/web/ai-user/buy-heartbeat-val")
|
||||||
|
suspend fun buyHeartbeatVal(@Body request: HeartbeatBuy): Response<Any>
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,40 @@
|
||||||
|
package com.remax.visualnovel.api.service
|
||||||
|
|
||||||
|
|
||||||
|
import com.remax.visualnovel.entity.request.AIIDRequest
|
||||||
|
import com.remax.visualnovel.entity.request.Gift
|
||||||
|
import com.remax.visualnovel.entity.request.PageQuery
|
||||||
|
import com.remax.visualnovel.entity.response.AIDict
|
||||||
|
import com.remax.visualnovel.entity.response.ChatBubble
|
||||||
|
import com.remax.visualnovel.entity.response.ChatModel
|
||||||
|
import com.remax.visualnovel.entity.response.Pageable
|
||||||
|
import com.remax.visualnovel.entity.response.base.Response
|
||||||
|
import retrofit2.http.Body
|
||||||
|
import retrofit2.http.POST
|
||||||
|
|
||||||
|
interface DictService {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取聊天气泡字典
|
||||||
|
*/
|
||||||
|
@POST("/web/chat-set/get-chat-bubble-list")
|
||||||
|
suspend fun getChatBubbleList(@Body request: AIIDRequest): Response<List<ChatBubble>>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* AI标签
|
||||||
|
*/
|
||||||
|
@POST("/web/get-ai-dict")
|
||||||
|
suspend fun getAIDict(): Response<AIDict>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 礼物字典
|
||||||
|
*/
|
||||||
|
@POST("/web/gift/dict-list")
|
||||||
|
suspend fun getGiftDict(@Body pageQuery: PageQuery = PageQuery(1).apply { page.ps = 100 }): Response<Pageable<Gift>>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* chat模型
|
||||||
|
*/
|
||||||
|
@POST("/web/chat-model/dict-list")
|
||||||
|
suspend fun getAIChatModel(): Response<List<ChatModel>>
|
||||||
|
}
|
|
@ -0,0 +1,30 @@
|
||||||
|
package com.remax.visualnovel.api.service
|
||||||
|
|
||||||
|
import com.remax.visualnovel.entity.request.CompleteUserInfoInput
|
||||||
|
import com.remax.visualnovel.entity.request.PlatformAccountVerifyDTO
|
||||||
|
import com.remax.visualnovel.entity.response.PlatformAccountVerify
|
||||||
|
import com.remax.visualnovel.entity.response.base.Response
|
||||||
|
import retrofit2.http.Body
|
||||||
|
import retrofit2.http.POST
|
||||||
|
|
||||||
|
interface LoginService {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 重复昵称验证
|
||||||
|
*/
|
||||||
|
@POST("/web/user/nickname-check")
|
||||||
|
suspend fun checkUserNickname(@Body request: CompleteUserInfoInput): Response<Boolean>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 三方账号验证
|
||||||
|
*/
|
||||||
|
@POST("/web/third/login")
|
||||||
|
suspend fun platformThirdVerify(@Body request: PlatformAccountVerifyDTO): Response<PlatformAccountVerify>
|
||||||
|
|
||||||
|
@POST("/web/user/logout")
|
||||||
|
suspend fun logout(): Response<Any>
|
||||||
|
|
||||||
|
@POST("/web/user/complete-user-info")
|
||||||
|
suspend fun register(@Body request: CompleteUserInfoInput): Response<Any>
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,40 @@
|
||||||
|
package com.remax.visualnovel.api.service
|
||||||
|
|
||||||
|
import com.remax.visualnovel.BuildConfig
|
||||||
|
import com.remax.visualnovel.entity.request.AIListRequest
|
||||||
|
import com.remax.visualnovel.entity.request.PageQuery
|
||||||
|
import com.remax.visualnovel.entity.request.SendGift
|
||||||
|
import com.remax.visualnovel.entity.response.MessageListOutput
|
||||||
|
import com.remax.visualnovel.entity.response.MessageStatOutput
|
||||||
|
import com.remax.visualnovel.entity.response.Pageable
|
||||||
|
import com.remax.visualnovel.entity.response.base.Response
|
||||||
|
import retrofit2.http.Body
|
||||||
|
import retrofit2.http.POST
|
||||||
|
|
||||||
|
interface MessageService {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 删除会话
|
||||||
|
*/
|
||||||
|
@POST(BuildConfig.API_COW + "/web/ai-message/del")
|
||||||
|
suspend fun deleteConversation(@Body request: AIListRequest): Response<Any>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 送礼物
|
||||||
|
*/
|
||||||
|
@POST("/web/ai-user-gift/send")
|
||||||
|
suspend fun sendGift(@Body dto: SendGift): Response<Any>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 未读消息统计
|
||||||
|
*/
|
||||||
|
@POST(BuildConfig.API_PIGEON + "/web/message/stat")
|
||||||
|
suspend fun getMessageStat(): Response<MessageStatOutput>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 系统通知列表
|
||||||
|
*/
|
||||||
|
@POST(BuildConfig.API_PIGEON + "/web/message/list")
|
||||||
|
suspend fun getMessageList(@Body dto: PageQuery): Response<Pageable<MessageListOutput>>
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,39 @@
|
||||||
|
package com.remax.visualnovel.api.service
|
||||||
|
|
||||||
|
import com.remax.visualnovel.BuildConfig
|
||||||
|
import com.remax.visualnovel.entity.request.ImgCheckDTO
|
||||||
|
import com.remax.visualnovel.entity.request.S3TypeDTO
|
||||||
|
import com.remax.visualnovel.entity.request.SimpleContentDTO
|
||||||
|
import com.remax.visualnovel.entity.response.BucketBean
|
||||||
|
import com.remax.visualnovel.entity.response.base.Response
|
||||||
|
import retrofit2.http.Body
|
||||||
|
import retrofit2.http.POST
|
||||||
|
|
||||||
|
/**
|
||||||
|
* OSS文件上传
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
interface OssService {
|
||||||
|
/**
|
||||||
|
* 获取aws s3 bucket信息
|
||||||
|
*/
|
||||||
|
@POST(BuildConfig.API_SHARK + "/web/file/sts-tk")
|
||||||
|
suspend fun getS3Bucket(@Body dto: S3TypeDTO): Response<BucketBean>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 图片鉴黄
|
||||||
|
*/
|
||||||
|
@POST(BuildConfig.API_SHARK + "/web/file/check")
|
||||||
|
suspend fun checkS3Img(
|
||||||
|
@Body imgCheckDTO: ImgCheckDTO
|
||||||
|
): Response<Any>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 关键字校验
|
||||||
|
*/
|
||||||
|
@POST("/web/check_text")
|
||||||
|
suspend fun checkText(
|
||||||
|
@Body simpleContentDTO: SimpleContentDTO
|
||||||
|
): Response<Any>
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,87 @@
|
||||||
|
package com.remax.visualnovel.api.service
|
||||||
|
|
||||||
|
import com.remax.visualnovel.BuildConfig
|
||||||
|
import com.remax.visualnovel.entity.request.ChargeOrderDTO
|
||||||
|
import com.remax.visualnovel.entity.request.ChargeProductDTO
|
||||||
|
import com.remax.visualnovel.entity.request.ChargeProductInfo
|
||||||
|
import com.remax.visualnovel.entity.request.SearchPage
|
||||||
|
import com.remax.visualnovel.entity.request.SubPriceDTO
|
||||||
|
import com.remax.visualnovel.entity.request.ValidateTransactionDTO
|
||||||
|
import com.remax.visualnovel.entity.response.ChargeOrder
|
||||||
|
import com.remax.visualnovel.entity.response.Membership
|
||||||
|
import com.remax.visualnovel.entity.response.SubPrice
|
||||||
|
import com.remax.visualnovel.entity.response.Transaction
|
||||||
|
import com.remax.visualnovel.entity.response.UserSubInfo
|
||||||
|
import com.remax.visualnovel.entity.response.Wallet
|
||||||
|
import com.remax.visualnovel.entity.response.base.Response
|
||||||
|
import retrofit2.http.Body
|
||||||
|
import retrofit2.http.POST
|
||||||
|
|
||||||
|
interface PayService {
|
||||||
|
/**
|
||||||
|
* 获取我的流水
|
||||||
|
*/
|
||||||
|
@POST(BuildConfig.API_LION + "/web/pay/account/bill-list")
|
||||||
|
suspend fun getTransactionList(@Body request: SearchPage): Response<Transaction>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取我的钱包
|
||||||
|
*/
|
||||||
|
@POST(BuildConfig.API_LION + "/web/pay/account/wallet")
|
||||||
|
suspend fun getMyWallet(): Response<Wallet>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取充值产品
|
||||||
|
*/
|
||||||
|
@POST(BuildConfig.API_LION + "/web/pay/config/charge-product-list")
|
||||||
|
suspend fun getChargeProducts(
|
||||||
|
@Body dto: ChargeProductDTO = ChargeProductDTO()
|
||||||
|
): Response<ChargeProductInfo>
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取vip订阅价格列表
|
||||||
|
*/
|
||||||
|
@POST(BuildConfig.API_LION + "/web/pay/config/sub-product-list")
|
||||||
|
suspend fun getSubPriceList(
|
||||||
|
@Body subPriceDTO: SubPriceDTO = SubPriceDTO()
|
||||||
|
): Response<List<SubPrice>>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 会员特权列表
|
||||||
|
*/
|
||||||
|
@POST(BuildConfig.API_LION + "/web/member/detail")
|
||||||
|
suspend fun getVipPrivilegeList(): Response<Membership>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 创建一个订单
|
||||||
|
*/
|
||||||
|
@POST(BuildConfig.API_LION +"/web/pay/trade/pre-charge-google")
|
||||||
|
suspend fun createOrder(
|
||||||
|
@Body dto: ChargeOrderDTO
|
||||||
|
): Response<ChargeOrder>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 验证支付是否成功
|
||||||
|
*/
|
||||||
|
@POST(BuildConfig.API_LION +"/web/pay/webhooks/google/v2")
|
||||||
|
suspend fun validateTranslation(
|
||||||
|
@Body dto: ValidateTransactionDTO
|
||||||
|
): Response<Any>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 验证订阅是否成功
|
||||||
|
*/
|
||||||
|
@POST(BuildConfig.API_LION +"/web/pay/subscribe/upload-google-receipt")
|
||||||
|
suspend fun uploadGoogleReceipt(
|
||||||
|
@Body dto: ValidateTransactionDTO
|
||||||
|
): Response<String>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 订阅/升级VIP前查询订阅信息
|
||||||
|
*/
|
||||||
|
@POST(BuildConfig.API_LION +"/web/pay/appStore/getUserSubscription")
|
||||||
|
suspend fun checkSubInfo(): Response<UserSubInfo>
|
||||||
|
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,52 @@
|
||||||
|
package com.remax.visualnovel.api.service
|
||||||
|
|
||||||
|
import com.remax.visualnovel.BuildConfig
|
||||||
|
import com.remax.visualnovel.entity.request.CompleteUserInfoInput
|
||||||
|
import com.remax.visualnovel.entity.response.Character
|
||||||
|
import com.remax.visualnovel.entity.response.User
|
||||||
|
import com.remax.visualnovel.entity.response.base.Response
|
||||||
|
import retrofit2.http.Body
|
||||||
|
import retrofit2.http.POST
|
||||||
|
|
||||||
|
interface UserService{
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 签到
|
||||||
|
*/
|
||||||
|
@POST("/web/si/asi")
|
||||||
|
suspend fun signToday(): Response<Boolean>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取签到周期数据
|
||||||
|
*/
|
||||||
|
/*@POST("/web/si/list")
|
||||||
|
suspend fun getSignList(): Response<SignInRoundOutput>*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取登录用户基础信息
|
||||||
|
*/
|
||||||
|
@POST("/web/user/base-info")
|
||||||
|
suspend fun getMyBaseInfo(): Response<User>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取me页面的ai列表
|
||||||
|
*/
|
||||||
|
@POST("/web/ai-user-search/base-list")
|
||||||
|
suspend fun getMyCharactersList(): Response<List<Character>>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 删除账号
|
||||||
|
*/
|
||||||
|
@POST("/web/user/del")
|
||||||
|
suspend fun deleteAccount(): Response<Any>
|
||||||
|
|
||||||
|
@POST("/web/user/edit-user-info")
|
||||||
|
suspend fun updateUserInfo(@Body request: CompleteUserInfoInput): Response<Any>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取云信appKey account token
|
||||||
|
*/
|
||||||
|
/*@POST(BuildConfig.API_PIGEON + "/web/im-user/get-account")
|
||||||
|
suspend fun getNimInfo(): Response<NimBean>*/
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,17 @@
|
||||||
|
package com.remax.visualnovel.app
|
||||||
|
|
||||||
|
|
||||||
|
import androidx.lifecycle.LifecycleOwner
|
||||||
|
|
||||||
|
|
||||||
|
interface AbsView : LifecycleOwner{
|
||||||
|
|
||||||
|
fun showLoading()
|
||||||
|
|
||||||
|
fun hideLoading()
|
||||||
|
|
||||||
|
fun showToast(text: String?)
|
||||||
|
|
||||||
|
fun showToast(resId: Int)
|
||||||
|
}
|
||||||
|
|
|
@ -0,0 +1,47 @@
|
||||||
|
package com.remax.visualnovel.app
|
||||||
|
|
||||||
|
import android.view.LayoutInflater
|
||||||
|
import android.view.View
|
||||||
|
import android.view.ViewGroup
|
||||||
|
import androidx.viewbinding.ViewBinding
|
||||||
|
import com.chad.library.adapter.base.BaseQuickAdapter
|
||||||
|
import com.chad.library.adapter.base.viewholder.BaseViewHolder
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Created by HJW on 2021/10/19
|
||||||
|
*/
|
||||||
|
abstract class BaseBindingQuickAdapter<T, out VB : ViewBinding>(
|
||||||
|
private val inflate: (LayoutInflater, ViewGroup, Boolean) -> VB,
|
||||||
|
layoutResId: Int = -1,
|
||||||
|
data: MutableList<T>? = null
|
||||||
|
) :
|
||||||
|
BaseQuickAdapter<T, BaseBindingQuickAdapter.BaseBindingHolder>(layoutResId, data) {
|
||||||
|
|
||||||
|
override fun onCreateDefViewHolder(parent: ViewGroup, viewType: Int) =
|
||||||
|
BaseBindingHolder(inflate(LayoutInflater.from(parent.context), parent, false))
|
||||||
|
|
||||||
|
class BaseBindingHolder(private val binding: ViewBinding) : BaseViewHolder(binding.root) {
|
||||||
|
constructor(itemView: View) : this(ViewBinding { itemView })
|
||||||
|
|
||||||
|
@Suppress("UNCHECKED_CAST")
|
||||||
|
fun <VB : ViewBinding> getViewBinding() = binding as VB
|
||||||
|
|
||||||
|
var extraObj: Any? = null
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 带左右滑动的item收起
|
||||||
|
*/
|
||||||
|
// fun resetDeleteView(position: Int) {
|
||||||
|
// try {
|
||||||
|
// (getViewByPosition(
|
||||||
|
// position + headerLayoutCount,
|
||||||
|
// R.id.easySwipeMenuLayout
|
||||||
|
// ) as? EasySwipeMenuLayout)?.resetStatus()
|
||||||
|
// } catch (e: Exception) {
|
||||||
|
// Timber.d("resetDeleteView position:$position Exception:${e.localizedMessage}")
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,45 @@
|
||||||
|
package com.remax.visualnovel.app
|
||||||
|
|
||||||
|
import androidx.lifecycle.DefaultLifecycleObserver
|
||||||
|
import androidx.lifecycle.LifecycleOwner
|
||||||
|
import com.remax.visualnovel.app.initializer.utils.FirebaseHelper
|
||||||
|
import com.remax.visualnovel.app.viewmodel.AppGlobalViewModel
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Created by HJW on 2022/9/21
|
||||||
|
* 监听整个app生命周期
|
||||||
|
*/
|
||||||
|
object ProcessLifecycleObserver : DefaultLifecycleObserver {
|
||||||
|
|
||||||
|
var isOnResume = false
|
||||||
|
private var refreshFirebaseTokenTime = 0L
|
||||||
|
|
||||||
|
private var appGlobalViewModel: AppGlobalViewModel? = null
|
||||||
|
|
||||||
|
fun setAppGlobalViewModel(appGlobalViewModel: AppGlobalViewModel) {
|
||||||
|
ProcessLifecycleObserver.appGlobalViewModel = appGlobalViewModel
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* APP在前台回调
|
||||||
|
*/
|
||||||
|
override fun onResume(owner: LifecycleOwner) {
|
||||||
|
isOnResume = true
|
||||||
|
val currTime = System.currentTimeMillis()
|
||||||
|
if (refreshFirebaseTokenTime != 0L && currTime - refreshFirebaseTokenTime > 60 * 1000) {
|
||||||
|
refreshFirebaseTokenTime = currTime
|
||||||
|
FirebaseHelper.getToken {
|
||||||
|
appGlobalViewModel?.updateTerminal(it)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* APP进入后台回调
|
||||||
|
*/
|
||||||
|
override fun onPause(owner: LifecycleOwner) {
|
||||||
|
isOnResume = false
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,230 @@
|
||||||
|
package com.remax.visualnovel.app.activityresultapi
|
||||||
|
|
||||||
|
import android.content.Intent
|
||||||
|
import android.text.TextUtils
|
||||||
|
import android.widget.Toast
|
||||||
|
import androidx.activity.result.ActivityResult
|
||||||
|
import androidx.activity.result.ActivityResultCallback
|
||||||
|
import androidx.fragment.app.Fragment
|
||||||
|
import androidx.fragment.app.FragmentActivity
|
||||||
|
import com.alibaba.android.arouter.core.LogisticsCenter
|
||||||
|
import com.alibaba.android.arouter.exception.NoRouteFoundException
|
||||||
|
import com.alibaba.android.arouter.facade.Postcard
|
||||||
|
import com.alibaba.android.arouter.facade.callback.InterceptorCallback
|
||||||
|
import com.alibaba.android.arouter.facade.callback.NavigationCallback
|
||||||
|
import com.alibaba.android.arouter.facade.enums.RouteType
|
||||||
|
import com.alibaba.android.arouter.facade.service.DegradeService
|
||||||
|
import com.alibaba.android.arouter.facade.service.InterceptorService
|
||||||
|
import com.alibaba.android.arouter.facade.service.PretreatmentService
|
||||||
|
import com.alibaba.android.arouter.launcher.ARouter
|
||||||
|
import com.remax.visualnovel.app.initializer.impl.ActivityLifecycleInitializer
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取activityResultLauncher
|
||||||
|
*/
|
||||||
|
fun FragmentActivity.activityResultLauncher(): XActivityResultContract<Intent, ActivityResult>? {
|
||||||
|
val activityKey = intent.getStringExtra(ActivityLifecycleInitializer.KEY_ACTIVITY_RESULT_API)
|
||||||
|
return if (TextUtils.isEmpty(activityKey)) null else ActivityLifecycleInitializer.resultLauncherMap[activityKey]
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 在Activity使用registerForActivityResult
|
||||||
|
*/
|
||||||
|
@JvmOverloads
|
||||||
|
fun FragmentActivity.registerForActivityResult(
|
||||||
|
intent: Intent,
|
||||||
|
activityResultCallback: ActivityResultCallback<ActivityResult>? = null
|
||||||
|
) {
|
||||||
|
activityResultLauncher()?.launch(intent, activityResultCallback)
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 在Activity使用registerForActivityResult
|
||||||
|
*/
|
||||||
|
@JvmOverloads
|
||||||
|
inline fun <reified T : FragmentActivity> FragmentActivity.registerForActivityResult(
|
||||||
|
intentExtra: (intent: Intent) -> Unit = {},
|
||||||
|
activityResultCallback: ActivityResultCallback<ActivityResult>? = null
|
||||||
|
) {
|
||||||
|
val intent = Intent(this, T::class.java)
|
||||||
|
intentExtra(intent)
|
||||||
|
registerForActivityResult(intent, activityResultCallback)
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 在Fragment使用registerForActivityResult
|
||||||
|
*/
|
||||||
|
@JvmOverloads
|
||||||
|
fun Fragment.registerForActivityResult(
|
||||||
|
intent: Intent,
|
||||||
|
activityResultCallback: ActivityResultCallback<ActivityResult>? = null
|
||||||
|
) {
|
||||||
|
requireActivity().activityResultLauncher()?.launch(intent, activityResultCallback)
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 在Fragment使用registerForActivityResult
|
||||||
|
*/
|
||||||
|
@JvmOverloads
|
||||||
|
inline fun <reified T : FragmentActivity> Fragment.registerForActivityResult(
|
||||||
|
intentExtra: (intent: Intent) -> Unit = {},
|
||||||
|
activityResultCallback: ActivityResultCallback<ActivityResult>? = null
|
||||||
|
) {
|
||||||
|
val intent = Intent(this.requireActivity(), T::class.java)
|
||||||
|
intentExtra(intent)
|
||||||
|
registerForActivityResult(intent, activityResultCallback)
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Activity中ARouter导航
|
||||||
|
* @param [activity] activity
|
||||||
|
* @param [activityResultCallback] 返回数据回调
|
||||||
|
*/
|
||||||
|
fun Postcard.navigation(
|
||||||
|
activity: FragmentActivity?,
|
||||||
|
activityResultCallback: ActivityResultCallback<ActivityResult>
|
||||||
|
): Any? {
|
||||||
|
return navigation(activity, null, activityResultCallback)
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Fragment中ARouter导航
|
||||||
|
* @param [fragment] Fragment
|
||||||
|
* @param [activityResultCallback] 返回数据回调
|
||||||
|
*/
|
||||||
|
fun Postcard.navigation(
|
||||||
|
fragment: Fragment?,
|
||||||
|
activityResultCallback: ActivityResultCallback<ActivityResult>
|
||||||
|
): Any? {
|
||||||
|
return navigation(fragment?.requireActivity(), null, activityResultCallback)
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Fragment中ARouter导航
|
||||||
|
* @param [fragment] Fragment
|
||||||
|
* @param [callback] 回调
|
||||||
|
* @param [activityResultCallback] 返回数据回调
|
||||||
|
*/
|
||||||
|
fun Postcard.navigation(
|
||||||
|
fragment: Fragment?,
|
||||||
|
callback: NavigationCallback?,
|
||||||
|
activityResultCallback: ActivityResultCallback<ActivityResult>
|
||||||
|
): Any? {
|
||||||
|
return navigation(fragment?.requireActivity(), callback, activityResultCallback)
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Activity中ARouter导航
|
||||||
|
* @param [activity] Fragment
|
||||||
|
* @param [callback] 回调
|
||||||
|
* @param [activityResultCallback] 返回数据回调
|
||||||
|
*/
|
||||||
|
fun Postcard.navigation(
|
||||||
|
activity: FragmentActivity?,
|
||||||
|
callback: NavigationCallback?,
|
||||||
|
activityResultCallback: ActivityResultCallback<ActivityResult>
|
||||||
|
): Any? {
|
||||||
|
if (activity == null) {
|
||||||
|
return null
|
||||||
|
}
|
||||||
|
val _postcard = this
|
||||||
|
val pretreatmentService = ARouter.getInstance().navigation(PretreatmentService::class.java)
|
||||||
|
if (null != pretreatmentService && !pretreatmentService.onPretreatment(activity, this)) {
|
||||||
|
return null
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
LogisticsCenter.completion(_postcard)
|
||||||
|
} catch (ex: NoRouteFoundException) {
|
||||||
|
debugLog(activity, path, group)
|
||||||
|
if (null != callback) {
|
||||||
|
callback.onLost(_postcard)
|
||||||
|
} else {
|
||||||
|
val degradeService = ARouter.getInstance().navigation(DegradeService::class.java)
|
||||||
|
degradeService?.onLost(activity, _postcard)
|
||||||
|
}
|
||||||
|
|
||||||
|
return null
|
||||||
|
}
|
||||||
|
callback?.onFound(_postcard)
|
||||||
|
val interceptorService = ARouter.getInstance().navigation(InterceptorService::class.java)
|
||||||
|
if (!isGreenChannel && interceptorService != null) {
|
||||||
|
interceptorService.doInterceptions(_postcard, object : InterceptorCallback {
|
||||||
|
|
||||||
|
override fun onContinue(postcard: Postcard?) {
|
||||||
|
_navigation(activity, _postcard, activityResultCallback)
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun onInterrupt(exception: Throwable?) {
|
||||||
|
callback?.onInterrupt(_postcard)
|
||||||
|
}
|
||||||
|
|
||||||
|
})
|
||||||
|
} else {
|
||||||
|
return _navigation(activity, this, activityResultCallback)
|
||||||
|
}
|
||||||
|
return null
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Debug模式下日志打印
|
||||||
|
*/
|
||||||
|
private fun debugLog(activity: FragmentActivity, path: String?, group: String?) {
|
||||||
|
if (ARouter.debuggable()) {
|
||||||
|
// Show friendly tips for user.
|
||||||
|
activity.runOnUiThread {
|
||||||
|
Toast.makeText(
|
||||||
|
activity,
|
||||||
|
"There's no route matched!Path = [${path}]Group = [${group}]",
|
||||||
|
Toast.LENGTH_LONG
|
||||||
|
).show()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun _navigation(
|
||||||
|
activity: FragmentActivity,
|
||||||
|
postcard: Postcard,
|
||||||
|
activityResultCallback: ActivityResultCallback<ActivityResult>,
|
||||||
|
): Any? {
|
||||||
|
|
||||||
|
return when (postcard.type) {
|
||||||
|
RouteType.ACTIVITY -> {
|
||||||
|
|
||||||
|
val intent = Intent(activity, postcard.destination)
|
||||||
|
postcard.extras?.let { intent.putExtras(it) }
|
||||||
|
if (postcard.flags != -1) {
|
||||||
|
intent.flags = postcard.flags
|
||||||
|
}
|
||||||
|
|
||||||
|
postcard.action?.let { intent.action = postcard.action }
|
||||||
|
activity.runOnUiThread {
|
||||||
|
//适配动画
|
||||||
|
if ((postcard.enterAnim != -1 && postcard.exitAnim != -1)) {
|
||||||
|
activity.overridePendingTransition(postcard.enterAnim, postcard.exitAnim)
|
||||||
|
}
|
||||||
|
activity.registerForActivityResult(intent, activityResultCallback)
|
||||||
|
}
|
||||||
|
null
|
||||||
|
}
|
||||||
|
RouteType.PROVIDER -> {
|
||||||
|
postcard.provider
|
||||||
|
}
|
||||||
|
RouteType.FRAGMENT -> {
|
||||||
|
val fragmentMeta = postcard.destination
|
||||||
|
try {
|
||||||
|
val instance = fragmentMeta.getConstructor().newInstance()
|
||||||
|
if (instance is Fragment) {
|
||||||
|
instance.arguments = postcard.extras
|
||||||
|
}
|
||||||
|
instance
|
||||||
|
} catch (ex: Exception) {
|
||||||
|
ex.printStackTrace()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else -> {
|
||||||
|
null
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,37 @@
|
||||||
|
package com.remax.visualnovel.app.activityresultapi
|
||||||
|
|
||||||
|
import androidx.activity.result.ActivityResultCallback
|
||||||
|
import androidx.activity.result.ActivityResultCaller
|
||||||
|
import androidx.activity.result.ActivityResultLauncher
|
||||||
|
import androidx.activity.result.contract.ActivityResultContract
|
||||||
|
|
||||||
|
class XActivityResultContract<I, O>(
|
||||||
|
activityResultCaller: ActivityResultCaller,
|
||||||
|
activityResultContract: ActivityResultContract<I, O>
|
||||||
|
) {
|
||||||
|
|
||||||
|
private var activityResultCallback: ActivityResultCallback<O>? = null
|
||||||
|
|
||||||
|
|
||||||
|
private val launcher: ActivityResultLauncher<I> =
|
||||||
|
activityResultCaller.registerForActivityResult(activityResultContract) {
|
||||||
|
activityResultCallback?.onActivityResult(it)
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 启动
|
||||||
|
*/
|
||||||
|
fun launch(input: I, activityResultCallback: ActivityResultCallback<O>?) {
|
||||||
|
this.activityResultCallback = activityResultCallback
|
||||||
|
launcher.launch(input)
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 注销
|
||||||
|
*/
|
||||||
|
fun unregister() {
|
||||||
|
launcher.unregister()
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,144 @@
|
||||||
|
package com.remax.visualnovel.app.base
|
||||||
|
|
||||||
|
import android.content.res.Resources
|
||||||
|
import android.os.Bundle
|
||||||
|
import android.view.MotionEvent
|
||||||
|
import android.view.View
|
||||||
|
import androidx.appcompat.app.AppCompatActivity
|
||||||
|
import androidx.viewbinding.ViewBinding
|
||||||
|
import com.remax.visualnovel.app.AbsView
|
||||||
|
import com.remax.visualnovel.app.widget.LoadingDialog
|
||||||
|
import com.remax.visualnovel.extension.fixedFontSize
|
||||||
|
import com.remax.visualnovel.extension.getBgColor
|
||||||
|
import com.remax.visualnovel.extension.isShouldHideKeyboard
|
||||||
|
import com.remax.visualnovel.extension.setStatusBarColor
|
||||||
|
import com.remax.visualnovel.extension.toast
|
||||||
|
import com.remax.visualnovel.extension.transitionFromAlpha
|
||||||
|
import com.remax.visualnovel.extension.transitionFromBottom
|
||||||
|
import com.remax.visualnovel.extension.withTransitionFromAlpha
|
||||||
|
import com.remax.visualnovel.extension.withTransitionFromBottom
|
||||||
|
import com.remax.visualnovel.utils.KeyboardUtils
|
||||||
|
import com.remax.visualnovel.utils.StatusBarUtils
|
||||||
|
import com.dylanc.loadingstateview.ActivityTransitionType
|
||||||
|
import com.dylanc.loadingstateview.BgColorType
|
||||||
|
import com.dylanc.loadingstateview.Decorative
|
||||||
|
import com.dylanc.loadingstateview.LoadingState
|
||||||
|
import com.dylanc.loadingstateview.LoadingStateDelegate
|
||||||
|
import com.dylanc.loadingstateview.OnReloadListener
|
||||||
|
import com.dylanc.viewbinding.base.ActivityBinding
|
||||||
|
import com.dylanc.viewbinding.base.ActivityBindingDelegate
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Activity基类
|
||||||
|
*/
|
||||||
|
abstract class BaseBindingActivity<out VB : ViewBinding> : AppCompatActivity(), AbsView,
|
||||||
|
LoadingState by LoadingStateDelegate(), OnReloadListener, Decorative,
|
||||||
|
ActivityBinding<VB> by ActivityBindingDelegate() {
|
||||||
|
|
||||||
|
private val loadingDialog by lazy {
|
||||||
|
createLoadingDialog()
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun createLoadingDialog(): LoadingDialog {
|
||||||
|
val dialog = LoadingDialog()
|
||||||
|
dialog.build(this)
|
||||||
|
return dialog
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun getResources(): Resources {
|
||||||
|
return super.getResources().fixedFontSize(this)
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun onCreate(savedInstanceState: Bundle?) {
|
||||||
|
super.onCreate(savedInstanceState)
|
||||||
|
setContentViewWithBinding()
|
||||||
|
binding.root.decorate(this, this)
|
||||||
|
StatusBarUtils.setStatusBarAndNavBarIsLight(this, false)
|
||||||
|
setStatusBarColor(backgroundColorType())
|
||||||
|
binding.root.setBackgroundColor(getBgColor(backgroundColorType()))
|
||||||
|
when (transitionType()) {
|
||||||
|
ActivityTransitionType.BOTTOM -> withTransitionFromBottom()
|
||||||
|
ActivityTransitionType.ALPHA -> withTransitionFromAlpha()
|
||||||
|
else -> {}
|
||||||
|
}
|
||||||
|
initView()
|
||||||
|
initData()
|
||||||
|
subscribeUi()
|
||||||
|
}
|
||||||
|
|
||||||
|
protected open fun backgroundColorType() = BgColorType.SPECIAL_MAP
|
||||||
|
|
||||||
|
protected abstract fun initView()
|
||||||
|
protected open fun subscribeUi() {}
|
||||||
|
protected open fun initData() {}
|
||||||
|
|
||||||
|
override fun onResume() {
|
||||||
|
super.onResume()
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun onPause() {
|
||||||
|
super.onPause()
|
||||||
|
if (KeyboardUtils.isSoftInputVisible(this)) {
|
||||||
|
KeyboardUtils.hideSoftInput(this)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun finish() {
|
||||||
|
super.finish()
|
||||||
|
when (transitionType()) {
|
||||||
|
ActivityTransitionType.BOTTOM -> transitionFromBottom()
|
||||||
|
ActivityTransitionType.ALPHA -> transitionFromAlpha()
|
||||||
|
else -> {}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
protected open var touchEventFun: ((MotionEvent) -> Unit)? = null
|
||||||
|
protected open fun touchHideKeyboardViewList(): List<View>? = null
|
||||||
|
|
||||||
|
override fun dispatchTouchEvent(ev: MotionEvent?): Boolean {
|
||||||
|
touchEventFun?.let {
|
||||||
|
runCatching { ev?.run(it) }
|
||||||
|
}
|
||||||
|
if (touchHideKeyboardViewList() != null) {
|
||||||
|
if (ev?.action == MotionEvent.ACTION_DOWN && KeyboardUtils.isSoftInputVisible(this) && touchHideKeyboardViewList()?.any {
|
||||||
|
it.isShouldHideKeyboard(
|
||||||
|
ev
|
||||||
|
)
|
||||||
|
} == true) {
|
||||||
|
KeyboardUtils.hideSoftInput(this)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return super.dispatchTouchEvent(ev)
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 页面开关方向
|
||||||
|
*/
|
||||||
|
protected open fun transitionType() = ActivityTransitionType.DEFAULT
|
||||||
|
|
||||||
|
override fun showLoading() {
|
||||||
|
if (!this.isDestroyed) {
|
||||||
|
runOnUiThread {
|
||||||
|
loadingDialog.show()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun hideLoading() {
|
||||||
|
if (!this.isDestroyed) {
|
||||||
|
runOnUiThread {
|
||||||
|
loadingDialog.dismiss()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun showToast(text: String?) {
|
||||||
|
toast(text)
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun showToast(resId: Int) {
|
||||||
|
toast(resId)
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,112 @@
|
||||||
|
package com.remax.visualnovel.app.base
|
||||||
|
|
||||||
|
import android.os.Bundle
|
||||||
|
import android.view.LayoutInflater
|
||||||
|
import android.view.View
|
||||||
|
import android.view.ViewGroup
|
||||||
|
import androidx.fragment.app.Fragment
|
||||||
|
import androidx.viewbinding.ViewBinding
|
||||||
|
import com.dylanc.loadingstateview.BgColorType
|
||||||
|
import com.dylanc.loadingstateview.Decorative
|
||||||
|
import com.dylanc.loadingstateview.LoadingState
|
||||||
|
import com.dylanc.loadingstateview.LoadingStateDelegate
|
||||||
|
import com.dylanc.loadingstateview.OnReloadListener
|
||||||
|
import com.dylanc.viewbinding.base.FragmentBinding
|
||||||
|
import com.dylanc.viewbinding.base.FragmentBindingDelegate
|
||||||
|
import com.remax.visualnovel.app.AbsView
|
||||||
|
import com.remax.visualnovel.app.widget.LoadingDialog
|
||||||
|
import com.remax.visualnovel.extension.getBgColor
|
||||||
|
import com.remax.visualnovel.extension.toast
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 基类Fragment
|
||||||
|
*/
|
||||||
|
abstract class BaseBindingFragment<VB : ViewBinding> : Fragment(), AbsView,
|
||||||
|
LoadingState by LoadingStateDelegate(), OnReloadListener, Decorative,
|
||||||
|
FragmentBinding<VB> by FragmentBindingDelegate() {
|
||||||
|
|
||||||
|
override fun onCreateView(
|
||||||
|
inflater: LayoutInflater, container: ViewGroup?, savedInstanceState: Bundle?
|
||||||
|
): View? {
|
||||||
|
val view = createViewWithBinding(inflater, container).decorate(this, this)
|
||||||
|
view.setBackgroundColor(requireContext().getBgColor(backgroundColorType()))
|
||||||
|
return view
|
||||||
|
}
|
||||||
|
|
||||||
|
open fun backgroundColorType() = BgColorType.SPECIAL_MAP
|
||||||
|
|
||||||
|
private val loadingDialog by lazy {
|
||||||
|
createLoadingDialog()
|
||||||
|
}
|
||||||
|
|
||||||
|
private var isLoaded = false
|
||||||
|
|
||||||
|
private fun createLoadingDialog(): LoadingDialog {
|
||||||
|
val dialog = LoadingDialog()
|
||||||
|
dialog.build(requireActivity())
|
||||||
|
return dialog
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun onViewCreated(view: View, savedInstanceState: Bundle?) {
|
||||||
|
super.onViewCreated(view, savedInstanceState)
|
||||||
|
onCreated(arguments)
|
||||||
|
subscribeUi()
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun onResume() {
|
||||||
|
super.onResume()
|
||||||
|
if (!isLoaded && !isHidden) {
|
||||||
|
lazyInit()
|
||||||
|
isLoaded = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
abstract fun onCreated(bundle: Bundle?)
|
||||||
|
|
||||||
|
open fun lazyInit() {}
|
||||||
|
|
||||||
|
open fun subscribeUi() {}
|
||||||
|
|
||||||
|
override fun showLoading() {
|
||||||
|
if (activity?.isDestroyed == false) {
|
||||||
|
activity?.runOnUiThread {
|
||||||
|
loadingDialog.show()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun hideLoading() {
|
||||||
|
if (activity?.isDestroyed == false) {
|
||||||
|
activity?.runOnUiThread {
|
||||||
|
loadingDialog.dismiss()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun showToast(text: String?) {
|
||||||
|
if (!isDetached) {
|
||||||
|
activity?.let {
|
||||||
|
if (!it.isDestroyed) {
|
||||||
|
it.toast(text)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun showToast(resId: Int) {
|
||||||
|
if (!isDetached) {
|
||||||
|
activity?.let {
|
||||||
|
if (!it.isDestroyed) {
|
||||||
|
it.toast(resId)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun onDetach() {
|
||||||
|
if (loadingDialog.getDialog().isShowing) {
|
||||||
|
loadingDialog.dismiss()
|
||||||
|
}
|
||||||
|
super.onDetach()
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,13 @@
|
||||||
|
package com.remax.visualnovel.app.base
|
||||||
|
|
||||||
|
import androidx.viewpager.widget.ViewPager
|
||||||
|
import androidx.viewpager2.widget.ViewPager2
|
||||||
|
import net.lucode.hackware.magicindicator.buildins.commonnavigator.abs.CommonNavigatorAdapter
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Created by HJW on 2023/7/18
|
||||||
|
*/
|
||||||
|
abstract class BaseCommonNavigatorAdapter(
|
||||||
|
open val viewPager2: ViewPager2?,
|
||||||
|
open val viewPager: ViewPager? = null
|
||||||
|
) : CommonNavigatorAdapter()
|
|
@ -0,0 +1,30 @@
|
||||||
|
package com.remax.visualnovel.app.base.app
|
||||||
|
|
||||||
|
import android.app.Application
|
||||||
|
import androidx.lifecycle.ViewModel
|
||||||
|
import androidx.lifecycle.ViewModelProvider
|
||||||
|
import androidx.lifecycle.viewmodel.CreationExtras
|
||||||
|
import com.remax.visualnovel.app.viewmodel.AppGlobalViewModel
|
||||||
|
import com.remax.visualnovel.app.viewmodel.AppIMViewModel
|
||||||
|
import com.remax.visualnovel.repository.api.MessageRepository
|
||||||
|
import com.remax.visualnovel.repository.api.UserRepository
|
||||||
|
import javax.inject.Inject
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 用于创建[AppIMViewModel, AppGlobalViewModel]等实例
|
||||||
|
*/
|
||||||
|
class AppViewModelFactory @Inject constructor(
|
||||||
|
private val application: Application,
|
||||||
|
private val userRepository: UserRepository,
|
||||||
|
private val messageRepository: MessageRepository
|
||||||
|
) : ViewModelProvider.Factory {
|
||||||
|
|
||||||
|
@Suppress("UNCHECKED_CAST")
|
||||||
|
override fun <T : ViewModel> create(modelClass: Class<T>, extras: CreationExtras): T {
|
||||||
|
return when (modelClass) {
|
||||||
|
AppIMViewModel::class.java -> AppIMViewModel(application)
|
||||||
|
AppGlobalViewModel::class.java -> AppGlobalViewModel(application, userRepository, messageRepository)
|
||||||
|
else -> throw IllegalArgumentException("Unknown class $modelClass")
|
||||||
|
} as T
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,11 @@
|
||||||
|
package com.remax.visualnovel.app.base.app
|
||||||
|
|
||||||
|
import android.app.Application
|
||||||
|
|
||||||
|
interface ApplicationProxy {
|
||||||
|
|
||||||
|
fun onCreate(application: Application)
|
||||||
|
|
||||||
|
fun onTerminate()
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,22 @@
|
||||||
|
package com.remax.visualnovel.app.base.app
|
||||||
|
|
||||||
|
import android.app.Application
|
||||||
|
import androidx.lifecycle.ViewModelStore
|
||||||
|
import androidx.lifecycle.ViewModelStoreOwner
|
||||||
|
|
||||||
|
object CommonApplicationProxy : ApplicationProxy, ViewModelStoreOwner {
|
||||||
|
|
||||||
|
lateinit var application: Application
|
||||||
|
private set
|
||||||
|
|
||||||
|
override val viewModelStore = ViewModelStore()
|
||||||
|
|
||||||
|
override fun onCreate(application: Application) {
|
||||||
|
CommonApplicationProxy.application = application
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun onTerminate() {
|
||||||
|
viewModelStore.clear()
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,264 @@
|
||||||
|
package com.remax.visualnovel.app.delegate
|
||||||
|
|
||||||
|
import android.app.Activity
|
||||||
|
import android.content.Context
|
||||||
|
import android.view.LayoutInflater
|
||||||
|
import android.view.View
|
||||||
|
import android.view.ViewGroup
|
||||||
|
import androidx.core.view.isInvisible
|
||||||
|
import androidx.core.view.isVisible
|
||||||
|
import com.remax.visualnovel.R
|
||||||
|
import com.remax.visualnovel.databinding.LayoutToolbarBinding
|
||||||
|
import com.remax.visualnovel.extension.findActivityContext
|
||||||
|
import com.remax.visualnovel.extension.getBgColor
|
||||||
|
import com.remax.visualnovel.extension.getNavHeight
|
||||||
|
import com.remax.visualnovel.extension.setOnClick
|
||||||
|
import com.remax.visualnovel.extension.setSize
|
||||||
|
import com.remax.visualnovel.utils.StatusBarUtils
|
||||||
|
import com.remax.visualnovel.utils.spannablex.utils.dp
|
||||||
|
import com.remax.visualnovel.widget.ui.buttons.IconButtonView
|
||||||
|
import com.remax.visualnovel.widget.uitoken.changeTextColor
|
||||||
|
import com.remax.visualnovel.widget.uitoken.expend.dsl.expand
|
||||||
|
import com.remax.visualnovel.widget.uitoken.handleUIToken
|
||||||
|
import com.dylanc.loadingstateview.BaseToolbarViewDelegate
|
||||||
|
import com.dylanc.loadingstateview.NavBtnType
|
||||||
|
import com.dylanc.loadingstateview.ToolbarConfig
|
||||||
|
import com.dylanc.loadingstateview.toolbarExtras
|
||||||
|
|
||||||
|
var ToolbarConfig.titleTextColorToken: Int? by toolbarExtras()
|
||||||
|
var ToolbarConfig.titleTextColor: Int? by toolbarExtras()
|
||||||
|
var ToolbarConfig.titleTextAlpha: Float? by toolbarExtras()
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 整个导航栏相关
|
||||||
|
*/
|
||||||
|
var ToolbarConfig.navBgColorToken: Int? by toolbarExtras()
|
||||||
|
var ToolbarConfig.navBgColor: Int? by toolbarExtras()
|
||||||
|
var ToolbarConfig.navBgAlpha: Float? by toolbarExtras() //一般和isFull = true 配合使用,因为只有当navbar和content布局重叠时,才需要滑动渐隐导航栏
|
||||||
|
var ToolbarConfig.navIsShow: Boolean? by toolbarExtras()
|
||||||
|
|
||||||
|
var ToolbarConfig.confirmEnabled: Boolean? by toolbarExtras()
|
||||||
|
var ToolbarConfig.isFull: Boolean? by toolbarExtras()
|
||||||
|
var ToolbarConfig.confirmContent: String? by toolbarExtras()
|
||||||
|
var ToolbarConfig.contractUSEnabled: Boolean? by toolbarExtras()
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 返回键相关
|
||||||
|
*/
|
||||||
|
var ToolbarConfig.navBackColorToken: Int? by toolbarExtras()
|
||||||
|
var ToolbarConfig.navBackColor: Int? by toolbarExtras()
|
||||||
|
var ToolbarConfig.onNavBackClick: (() -> Unit)? by toolbarExtras()
|
||||||
|
var ToolbarConfig.onNavBackIconToken: Int? by toolbarExtras()
|
||||||
|
|
||||||
|
var ToolbarConfig.iconButtonType: IconButtonType? by toolbarExtras()
|
||||||
|
|
||||||
|
|
||||||
|
enum class IconButtonType {
|
||||||
|
ON_PIC, NORMAL
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 公共导航栏
|
||||||
|
*/
|
||||||
|
class ToolbarViewDelegate : BaseToolbarViewDelegate() {
|
||||||
|
private lateinit var binding: LayoutToolbarBinding
|
||||||
|
private var context: Context? = null
|
||||||
|
|
||||||
|
override fun onCreateToolbar(inflater: LayoutInflater, parent: ViewGroup): View {
|
||||||
|
context = parent.context
|
||||||
|
binding = LayoutToolbarBinding.inflate(inflater, parent, false)
|
||||||
|
return binding.root
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun onBindToolbar(config: ToolbarConfig) {
|
||||||
|
binding.apply {
|
||||||
|
val expendSize = 4.dp
|
||||||
|
tvTitle.text = config.title?.ifEmpty { " " } ?: " "
|
||||||
|
context?.run {
|
||||||
|
if (config.isFull == true) {
|
||||||
|
root.tag = "isFull"
|
||||||
|
(context?.findActivityContext() as? Activity)?.let {
|
||||||
|
StatusBarUtils.setTransparent(it)
|
||||||
|
navBg.setSize(height = it.getNavHeight())
|
||||||
|
}
|
||||||
|
navBg.alpha = 0f
|
||||||
|
tvTitle.alpha = 0f
|
||||||
|
}
|
||||||
|
//设置导航栏是否显示
|
||||||
|
config.navIsShow?.let { root.isVisible = it }
|
||||||
|
navBg.setBackgroundColor(getBgColor(config.colorType))
|
||||||
|
//设置导航栏颜色
|
||||||
|
config.navBgColorToken?.let { navBg.setBackgroundColor(handleUIToken(it)?.color ?: 0) }
|
||||||
|
config.navBgColor?.let { navBg.setBackgroundColor(it) }
|
||||||
|
//设置导航栏透明度
|
||||||
|
config.navBgAlpha?.let { navBg.alpha = it }
|
||||||
|
|
||||||
|
//设置标题颜色
|
||||||
|
config.titleTextColorToken?.let { tvTitle.setTextColor(handleUIToken(it)?.color ?: 0) }
|
||||||
|
config.titleTextColor?.let { tvTitle.setTextColor(it) }
|
||||||
|
//设置标题透明度
|
||||||
|
config.titleTextAlpha?.let { tvTitle.alpha = it }
|
||||||
|
|
||||||
|
//设置返回按钮颜色
|
||||||
|
config.navBackColorToken?.let { navBack.changeTextColor { textUIColorToken = getString(it) } }
|
||||||
|
config.navBackColor?.let { navBack.setTextColor(it) }
|
||||||
|
|
||||||
|
//设置confirm按钮是否可点
|
||||||
|
config.confirmEnabled?.let { rightConfirmBtn.isEnabled = it }
|
||||||
|
//设置confirm按钮的文字
|
||||||
|
config.confirmContent?.let { rightConfirmBtn.text = it }
|
||||||
|
//设置contract us按钮是否可点
|
||||||
|
config.contractUSEnabled?.let { contractUsBtn.isEnabled = it }
|
||||||
|
|
||||||
|
//设置iconfont按钮样式
|
||||||
|
when (config.iconButtonType) {
|
||||||
|
IconButtonType.ON_PIC -> {
|
||||||
|
navBack.setButtonStyle(buttonName = IconButtonView.NavButton_OnPic)
|
||||||
|
rightIconBtn1.setButtonStyle(buttonName = IconButtonView.NavButton_OnPic)
|
||||||
|
rightIconBtn2.setButtonStyle(buttonName = IconButtonView.NavButton_OnPic)
|
||||||
|
}
|
||||||
|
|
||||||
|
else -> {
|
||||||
|
navBack.setButtonStyle(buttonName = IconButtonView.NavButton)
|
||||||
|
rightIconBtn1.setButtonStyle(buttonName = IconButtonView.NavButton)
|
||||||
|
rightIconBtn2.setButtonStyle(buttonName = IconButtonView.NavButton)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
navBack.expand(expendSize, expendSize)
|
||||||
|
}
|
||||||
|
|
||||||
|
val setBackClick = {
|
||||||
|
setOnClick(navBack) {
|
||||||
|
if (config.onNavBackClick != null) {
|
||||||
|
config.onNavBackClick?.invoke()
|
||||||
|
} else {
|
||||||
|
(context.findActivityContext() as? Activity)?.onBackPressed()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
when (config.navBtnType) {
|
||||||
|
//navBack是返回按钮
|
||||||
|
NavBtnType.BACK -> {
|
||||||
|
navBack.setText(R.string.icon_arrow_left_border)
|
||||||
|
setBackClick.invoke()
|
||||||
|
navBack.isVisible = true
|
||||||
|
}
|
||||||
|
//navBack是关闭按钮
|
||||||
|
NavBtnType.ClOSE -> {
|
||||||
|
navBack.setText(R.string.icon_close)
|
||||||
|
setBackClick.invoke()
|
||||||
|
navBack.isVisible = true
|
||||||
|
}
|
||||||
|
|
||||||
|
//navBack是向下关闭按钮
|
||||||
|
NavBtnType.DOWN -> {
|
||||||
|
navBack.setText(R.string.icon_arrow_down_border)
|
||||||
|
setBackClick.invoke()
|
||||||
|
navBack.isVisible = true
|
||||||
|
}
|
||||||
|
|
||||||
|
//navBack是自定义按钮
|
||||||
|
NavBtnType.CUSTOM -> {
|
||||||
|
if (config.onNavBackIconToken != null) {
|
||||||
|
navBack.setText(config.onNavBackIconToken!!)
|
||||||
|
navBack.isVisible = true
|
||||||
|
setBackClick.invoke()
|
||||||
|
} else {
|
||||||
|
navBack.isInvisible = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
NavBtnType.INVISIBLE -> {
|
||||||
|
navBack.isInvisible = true
|
||||||
|
}
|
||||||
|
|
||||||
|
NavBtnType.NONE -> {
|
||||||
|
navBack.isVisible = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
with(contractUsBtn) {
|
||||||
|
if (config.contractUsBtnText != null) {
|
||||||
|
isVisible = true
|
||||||
|
val expandX = 24.dp
|
||||||
|
val expandY = 10.dp
|
||||||
|
expand(expandX, expandY)
|
||||||
|
if (config.contractUsBtnText != null) {
|
||||||
|
setText(config.contractUsBtnText ?: 0)
|
||||||
|
}
|
||||||
|
setOnClick(this) {
|
||||||
|
config.onContractUsBtnBlock?.invoke(this)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
isVisible = false
|
||||||
|
setOnClickListener(null)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
with(rightConfirmBtn) {
|
||||||
|
if (config.showConfirmBtn == true) {
|
||||||
|
isVisible = true
|
||||||
|
setOnClick(this) {
|
||||||
|
config.onConfirmBtnBlock?.invoke(this)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
isVisible = false
|
||||||
|
setOnClickListener(null)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
with(rightIconBtn1) {
|
||||||
|
if (config.showRightIconBtn1 != null) {
|
||||||
|
isVisible = true
|
||||||
|
setText(config.showRightIconBtn1!!)
|
||||||
|
expand(expendSize, expendSize)
|
||||||
|
setOnClick(this) {
|
||||||
|
config.onRightIconBtnBlock1?.invoke(this)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
isVisible = false
|
||||||
|
setOnClickListener(null)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
with(rightIconBtn2) {
|
||||||
|
if (config.showRightIconBtn2 != null) {
|
||||||
|
isVisible = true
|
||||||
|
setText(config.showRightIconBtn2!!)
|
||||||
|
changeTextColor {
|
||||||
|
textUIColorToken = if (config.rightIconColorBtn2 != null) {
|
||||||
|
context.getString(config.rightIconColorBtn2!!)
|
||||||
|
} else {
|
||||||
|
context.getString(R.string.color_txt_primary_normal)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
expand(expendSize, expendSize)
|
||||||
|
setOnClick(this) {
|
||||||
|
config.onRightIconBtnBlock2?.invoke(this)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
isVisible = false
|
||||||
|
setOnClickListener(null)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
with(rightBtn) {
|
||||||
|
if (config.showRightBtn != null) {
|
||||||
|
isVisible = true
|
||||||
|
setText(config.showRightBtn!!)
|
||||||
|
setOnClick(this) {
|
||||||
|
config.onRightBtnBlock?.invoke(this)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
isVisible = false
|
||||||
|
setOnClickListener(null)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,67 @@
|
||||||
|
package com.remax.visualnovel.app.di
|
||||||
|
|
||||||
|
|
||||||
|
import com.remax.visualnovel.api.factory.ServiceFactory
|
||||||
|
import com.remax.visualnovel.api.service.AIService
|
||||||
|
import com.remax.visualnovel.api.service.BookService
|
||||||
|
import com.remax.visualnovel.api.service.ChatService
|
||||||
|
import com.remax.visualnovel.api.service.DictService
|
||||||
|
import com.remax.visualnovel.api.service.LoginService
|
||||||
|
import com.remax.visualnovel.api.service.MessageService
|
||||||
|
import com.remax.visualnovel.api.service.OssService
|
||||||
|
import com.remax.visualnovel.api.service.PayService
|
||||||
|
import com.remax.visualnovel.api.service.UserService
|
||||||
|
import dagger.Module
|
||||||
|
import dagger.Provides
|
||||||
|
import dagger.hilt.InstallIn
|
||||||
|
import dagger.hilt.components.SingletonComponent
|
||||||
|
import javax.inject.Singleton
|
||||||
|
|
||||||
|
/**
|
||||||
|
* hilt注入service请求
|
||||||
|
*/
|
||||||
|
@Module
|
||||||
|
@InstallIn(SingletonComponent::class)
|
||||||
|
object ApiServiceModule {
|
||||||
|
|
||||||
|
@Singleton
|
||||||
|
@Provides
|
||||||
|
fun userService() = create<UserService>()
|
||||||
|
|
||||||
|
@Singleton
|
||||||
|
@Provides
|
||||||
|
fun loginService() = create<LoginService>()
|
||||||
|
|
||||||
|
@Singleton
|
||||||
|
@Provides
|
||||||
|
fun messageService() = create<MessageService>()
|
||||||
|
|
||||||
|
@Singleton
|
||||||
|
@Provides
|
||||||
|
fun dictService() = create<DictService>()
|
||||||
|
|
||||||
|
@Singleton
|
||||||
|
@Provides
|
||||||
|
fun bookService() = create<BookService>()
|
||||||
|
|
||||||
|
@Singleton
|
||||||
|
@Provides
|
||||||
|
fun aiService() = create<AIService>()
|
||||||
|
|
||||||
|
@Singleton
|
||||||
|
@Provides
|
||||||
|
fun ossService() = create<OssService>()
|
||||||
|
|
||||||
|
@Singleton
|
||||||
|
@Provides
|
||||||
|
fun payService() = create<PayService>()
|
||||||
|
|
||||||
|
@Singleton
|
||||||
|
@Provides
|
||||||
|
fun chatService() = create<ChatService>()
|
||||||
|
|
||||||
|
|
||||||
|
private inline fun <reified T> create(): T {
|
||||||
|
return ServiceFactory.createService()
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,31 @@
|
||||||
|
package com.remax.visualnovel.app.di
|
||||||
|
|
||||||
|
import androidx.lifecycle.ViewModelProvider
|
||||||
|
import com.remax.visualnovel.app.base.app.AppViewModelFactory
|
||||||
|
import com.remax.visualnovel.app.base.app.CommonApplicationProxy
|
||||||
|
import com.remax.visualnovel.app.viewmodel.AppGlobalViewModel
|
||||||
|
import com.remax.visualnovel.app.viewmodel.AppIMViewModel
|
||||||
|
import dagger.Module
|
||||||
|
import dagger.Provides
|
||||||
|
import dagger.hilt.InstallIn
|
||||||
|
import dagger.hilt.components.SingletonComponent
|
||||||
|
import javax.inject.Singleton
|
||||||
|
|
||||||
|
/**
|
||||||
|
* [AppIMViewModel] 提供者
|
||||||
|
*/
|
||||||
|
@InstallIn(SingletonComponent::class)
|
||||||
|
@Module
|
||||||
|
object ViewModelModule {
|
||||||
|
|
||||||
|
@Singleton
|
||||||
|
@Provides
|
||||||
|
fun provideAppIMViewModel(factory: AppViewModelFactory) =
|
||||||
|
ViewModelProvider(CommonApplicationProxy.viewModelStore, factory)[AppIMViewModel::class.java]
|
||||||
|
|
||||||
|
@Singleton
|
||||||
|
@Provides
|
||||||
|
fun provideAppGlobalViewModel(factory: AppViewModelFactory) =
|
||||||
|
ViewModelProvider(CommonApplicationProxy.viewModelStore, factory)[AppGlobalViewModel::class.java]
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,18 @@
|
||||||
|
package com.remax.visualnovel.app.initializer
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Created by HJW on 2023/5/11
|
||||||
|
*
|
||||||
|
* 启动类型
|
||||||
|
*/
|
||||||
|
enum class AppInitializerStartType {
|
||||||
|
/**
|
||||||
|
* 串行执行
|
||||||
|
*/
|
||||||
|
TYPE_SERIES,
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 并发执行
|
||||||
|
*/
|
||||||
|
TYPE_PARALLEL,
|
||||||
|
}
|
|
@ -0,0 +1,25 @@
|
||||||
|
package com.remax.visualnovel.app.initializer
|
||||||
|
|
||||||
|
import com.remax.visualnovel.app.initializer.di.AppInitializerPriority
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Created by HJW on 2023/5/11
|
||||||
|
* 后续都需要依赖此框架
|
||||||
|
*/
|
||||||
|
interface AppInitializers {
|
||||||
|
/**
|
||||||
|
* 初始化代码
|
||||||
|
*/
|
||||||
|
fun init()
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @return 初始化类型
|
||||||
|
*/
|
||||||
|
fun getStartType(): AppInitializerStartType = AppInitializerStartType.TYPE_SERIES
|
||||||
|
|
||||||
|
/**
|
||||||
|
* TYPE_SERIES 类型时,权重越大,越先执行
|
||||||
|
*/
|
||||||
|
fun widget(): Int = AppInitializerPriority.SERIES_1.priority
|
||||||
|
}
|
|
@ -0,0 +1,41 @@
|
||||||
|
package com.remax.visualnovel.app.initializer
|
||||||
|
|
||||||
|
import android.app.Application
|
||||||
|
import android.util.Log
|
||||||
|
import com.remax.visualnovel.app.initializer.di.AppInitializerEntryPoint
|
||||||
|
import dagger.hilt.android.EntryPointAccessors
|
||||||
|
import kotlinx.coroutines.MainScope
|
||||||
|
import kotlinx.coroutines.launch
|
||||||
|
import javax.inject.Inject
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Created by HJW on 2023/5/11
|
||||||
|
*/
|
||||||
|
class AppInitializersProvider @Inject constructor(private val application: Application) {
|
||||||
|
|
||||||
|
private val TAG = "AppInitializersProvider"
|
||||||
|
|
||||||
|
private val initializers: Set<AppInitializers> by lazy {
|
||||||
|
EntryPointAccessors.fromApplication(application, AppInitializerEntryPoint::class.java)
|
||||||
|
.getAppInitializers()
|
||||||
|
}
|
||||||
|
|
||||||
|
fun startInit() {
|
||||||
|
val seriesList = initializers.filter {
|
||||||
|
it.getStartType() == AppInitializerStartType.TYPE_SERIES
|
||||||
|
}.sortedByDescending { it.widget() }
|
||||||
|
val parallelList =
|
||||||
|
initializers.filter { it.getStartType() == AppInitializerStartType.TYPE_PARALLEL }
|
||||||
|
Log.d(TAG, "AppInitializersProvider 开始执行 并行")
|
||||||
|
parallelList.parallelStream().forEach {
|
||||||
|
MainScope().launch {
|
||||||
|
Log.d(TAG, "AppInitializersProvider $it 初始化模块协程开始执行: ${this.coroutineContext}")
|
||||||
|
it.init()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Log.d(TAG, "AppInitializersProvider 结束执行 并行")
|
||||||
|
seriesList.forEach {
|
||||||
|
it.init()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,18 @@
|
||||||
|
package com.remax.visualnovel.app.initializer.di
|
||||||
|
|
||||||
|
|
||||||
|
import com.remax.visualnovel.app.initializer.AppInitializers
|
||||||
|
import dagger.hilt.EntryPoint
|
||||||
|
import dagger.hilt.InstallIn
|
||||||
|
import dagger.hilt.components.SingletonComponent
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Created by HJW on 2023/5/11
|
||||||
|
* 启动框架容器注解
|
||||||
|
*/
|
||||||
|
|
||||||
|
@EntryPoint
|
||||||
|
@InstallIn(SingletonComponent::class)
|
||||||
|
interface AppInitializerEntryPoint {
|
||||||
|
fun getAppInitializers(): Set<AppInitializers>
|
||||||
|
}
|
|
@ -0,0 +1,19 @@
|
||||||
|
package com.remax.visualnovel.app.initializer.di
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Created by HJW on 2023/6/12
|
||||||
|
*/
|
||||||
|
enum class AppInitializerPriority(val priority: Int) {
|
||||||
|
/**
|
||||||
|
* 并发,无优先级
|
||||||
|
*/
|
||||||
|
PARALLEL(1),
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 串行,数字越大,优先级越高
|
||||||
|
*/
|
||||||
|
SERIES_1(1),
|
||||||
|
SERIES_2(2),
|
||||||
|
SERIES_3(3),
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,42 @@
|
||||||
|
package com.remax.visualnovel.app.initializer.di
|
||||||
|
|
||||||
|
import android.app.Application
|
||||||
|
import com.remax.visualnovel.app.initializer.AppInitializers
|
||||||
|
import com.remax.visualnovel.app.initializer.impl.ActivityLifecycleInitializer
|
||||||
|
import com.remax.visualnovel.app.initializer.impl.LocalDataInitializer
|
||||||
|
import com.remax.visualnovel.app.initializer.impl.RouterInitializer
|
||||||
|
import com.remax.visualnovel.app.initializer.impl.SystemInitializer
|
||||||
|
import com.remax.visualnovel.app.initializer.impl.ThirdInitializer
|
||||||
|
import com.remax.visualnovel.app.initializer.impl.UserInitializer
|
||||||
|
import com.remax.visualnovel.app.viewmodel.AppGlobalViewModel
|
||||||
|
import com.remax.visualnovel.app.viewmodel.AppIMViewModel
|
||||||
|
import dagger.Module
|
||||||
|
import dagger.Provides
|
||||||
|
import dagger.hilt.InstallIn
|
||||||
|
import dagger.hilt.components.SingletonComponent
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Created by HJW on 2023/5/11
|
||||||
|
*
|
||||||
|
* 添加容器注入
|
||||||
|
*/
|
||||||
|
@Module
|
||||||
|
@InstallIn(SingletonComponent::class)
|
||||||
|
object AppInitializersModule {
|
||||||
|
|
||||||
|
@Provides
|
||||||
|
fun provideAppInitializers(application: Application,
|
||||||
|
appIMViewModel: AppIMViewModel,
|
||||||
|
appGlobalViewModel: AppGlobalViewModel
|
||||||
|
): Set<AppInitializers> = setOf(
|
||||||
|
// FirebaseInitializer(application), TODO- add firebase support later
|
||||||
|
UserInitializer(application),
|
||||||
|
//JsInitializer(application),
|
||||||
|
LocalDataInitializer(application),
|
||||||
|
RouterInitializer(application),
|
||||||
|
ThirdInitializer(application),
|
||||||
|
ActivityLifecycleInitializer(application, appIMViewModel),
|
||||||
|
SystemInitializer(application, appGlobalViewModel)
|
||||||
|
)
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,101 @@
|
||||||
|
package com.remax.visualnovel.app.initializer.impl
|
||||||
|
|
||||||
|
import android.app.Activity
|
||||||
|
import android.app.Application
|
||||||
|
import android.content.Intent
|
||||||
|
import android.os.Bundle
|
||||||
|
import android.text.TextUtils
|
||||||
|
import androidx.activity.result.ActivityResult
|
||||||
|
import androidx.activity.result.ActivityResultCaller
|
||||||
|
import androidx.activity.result.contract.ActivityResultContracts
|
||||||
|
import androidx.fragment.app.FragmentActivity
|
||||||
|
import com.remax.visualnovel.app.activityresultapi.XActivityResultContract
|
||||||
|
import com.remax.visualnovel.app.initializer.AppInitializers
|
||||||
|
import com.remax.visualnovel.app.viewmodel.AppIMViewModel
|
||||||
|
import com.remax.visualnovel.configs.NovelApplication
|
||||||
|
import com.remax.visualnovel.utils.StatusBarUtils
|
||||||
|
import timber.log.Timber
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Created by HJW on 2023/5/11
|
||||||
|
*/
|
||||||
|
class ActivityLifecycleInitializer(val application: Application, private val appIMViewModel: AppIMViewModel) : AppInitializers {
|
||||||
|
|
||||||
|
override fun init() {
|
||||||
|
application.registerActivityLifecycleCallbacks(AppLifecycleCallbacks(appIMViewModel))
|
||||||
|
}
|
||||||
|
|
||||||
|
inner class AppLifecycleCallbacks constructor(private val appIMViewModel: AppIMViewModel) : Application.ActivityLifecycleCallbacks {
|
||||||
|
|
||||||
|
private var activityCount = 0
|
||||||
|
|
||||||
|
override fun onActivityCreated(activity: Activity, savedInstanceState: Bundle?) {
|
||||||
|
if (activity is ActivityResultCaller) {
|
||||||
|
//生成一个Key
|
||||||
|
val activityKey = activity.javaClass.simpleName + System.currentTimeMillis()
|
||||||
|
//添加一个默认ActivityResultLauncher
|
||||||
|
val resultLauncher =
|
||||||
|
XActivityResultContract(activity, ActivityResultContracts.StartActivityForResult())
|
||||||
|
//把生成的Key放到intent中,作为每一个Activity的唯一标识
|
||||||
|
activity.intent.putExtra(KEY_ACTIVITY_RESULT_API, activityKey)
|
||||||
|
//存放到Map中
|
||||||
|
resultLauncherMap[activityKey] = resultLauncher
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun onActivityStarted(activity: Activity) {
|
||||||
|
activityCount++
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun onActivityPaused(activity: Activity) {
|
||||||
|
NovelApplication.setCurrentActivity(null)
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun onActivityResumed(activity: Activity) {
|
||||||
|
Timber.d("currentActivity:${activity::class.java.name}")
|
||||||
|
NovelApplication.setCurrentActivity(activity)
|
||||||
|
if (StatusBarUtils.statusBarHeight == 0) {
|
||||||
|
StatusBarUtils.getStatusBarHeight(activity)
|
||||||
|
}
|
||||||
|
|
||||||
|
//TODO - check pay info later
|
||||||
|
/*if (activity !is WelcomeActivity) {
|
||||||
|
GooglePayManager.checkProductDetails()
|
||||||
|
}*/
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun onActivityStopped(activity: Activity) {
|
||||||
|
activityCount--
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun onActivityDestroyed(activity: Activity) {
|
||||||
|
if (activity is FragmentActivity) {
|
||||||
|
val activityKey = activity.intent.getStringExtra(KEY_ACTIVITY_RESULT_API)
|
||||||
|
if (!TextUtils.isEmpty(activityKey)) {
|
||||||
|
resultLauncherMap[activityKey]?.unregister()
|
||||||
|
//移除activity的resultLauncher
|
||||||
|
resultLauncherMap.remove(activityKey)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun onActivitySaveInstanceState(activity: Activity, outState: Bundle) {
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
companion object {
|
||||||
|
/**
|
||||||
|
* 保存ActivityResultApi的Key
|
||||||
|
*/
|
||||||
|
const val KEY_ACTIVITY_RESULT_API = "activityResultApi"
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 保存activity和Fragment的resultLauncher
|
||||||
|
*/
|
||||||
|
val resultLauncherMap: MutableMap<String, XActivityResultContract<Intent, ActivityResult>> =
|
||||||
|
mutableMapOf()
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,27 @@
|
||||||
|
package com.remax.visualnovel.app.initializer.impl
|
||||||
|
|
||||||
|
import android.app.Application
|
||||||
|
import com.remax.visualnovel.app.initializer.AppInitializers
|
||||||
|
import com.remax.visualnovel.app.initializer.di.AppInitializerPriority
|
||||||
|
import com.remax.visualnovel.utils.datastore.IDataStoreOwner
|
||||||
|
import com.tencent.mmkv.MMKV
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Created by HJW on 2023/5/11
|
||||||
|
* 本地数据保存相关初始化
|
||||||
|
*/
|
||||||
|
class LocalDataInitializer(val application: Application) : AppInitializers {
|
||||||
|
|
||||||
|
override fun init() {
|
||||||
|
//初始化mmkv
|
||||||
|
val dir = application.filesDir.absolutePath + "/mmkv_epal"
|
||||||
|
MMKV.initialize(application, dir)
|
||||||
|
//初始化datastore
|
||||||
|
IDataStoreOwner.application = application
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun widget(): Int {
|
||||||
|
return AppInitializerPriority.SERIES_3.priority
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,26 @@
|
||||||
|
package com.remax.visualnovel.app.initializer.impl
|
||||||
|
|
||||||
|
import android.app.Application
|
||||||
|
import com.alibaba.android.arouter.launcher.ARouter
|
||||||
|
import com.remax.visualnovel.BuildConfig
|
||||||
|
import com.remax.visualnovel.app.initializer.AppInitializers
|
||||||
|
import com.remax.visualnovel.app.initializer.di.AppInitializerPriority
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Created by HJW on 2023/5/11
|
||||||
|
* 路由相关初始化
|
||||||
|
*/
|
||||||
|
class RouterInitializer(val application: Application) : AppInitializers {
|
||||||
|
|
||||||
|
override fun init() {
|
||||||
|
if (BuildConfig.DEBUG) {
|
||||||
|
ARouter.openLog()
|
||||||
|
ARouter.openDebug()
|
||||||
|
}
|
||||||
|
ARouter.init(application)
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun widget(): Int {
|
||||||
|
return AppInitializerPriority.SERIES_3.priority
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,47 @@
|
||||||
|
package com.remax.visualnovel.app.initializer.impl
|
||||||
|
|
||||||
|
import android.app.ActivityManager
|
||||||
|
import android.app.Application
|
||||||
|
import android.content.Context
|
||||||
|
import android.os.Build
|
||||||
|
import android.webkit.WebView
|
||||||
|
import androidx.appcompat.app.AppCompatDelegate
|
||||||
|
import androidx.lifecycle.ProcessLifecycleOwner
|
||||||
|
import com.remax.visualnovel.app.ProcessLifecycleObserver
|
||||||
|
import com.remax.visualnovel.app.initializer.AppInitializers
|
||||||
|
import com.remax.visualnovel.app.initializer.di.AppInitializerPriority
|
||||||
|
import com.remax.visualnovel.app.viewmodel.AppGlobalViewModel
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Created by HJW on 2023/5/11
|
||||||
|
* 系统相关初始化
|
||||||
|
*/
|
||||||
|
class SystemInitializer(val application: Application, val appGlobalViewModel: AppGlobalViewModel) : AppInitializers {
|
||||||
|
|
||||||
|
override fun init() {
|
||||||
|
//监听application生命周期
|
||||||
|
ProcessLifecycleOwner.get().lifecycle.addObserver(ProcessLifecycleObserver)
|
||||||
|
ProcessLifecycleObserver.setAppGlobalViewModel(appGlobalViewModel)
|
||||||
|
AppCompatDelegate.setDefaultNightMode(AppCompatDelegate.MODE_NIGHT_YES)
|
||||||
|
//启用矢量图兼容
|
||||||
|
AppCompatDelegate.setCompatVectorFromResourcesEnabled(true)
|
||||||
|
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.P) {
|
||||||
|
var processName = ""
|
||||||
|
(application.getSystemService(Context.ACTIVITY_SERVICE) as? ActivityManager)
|
||||||
|
?.runningAppProcesses
|
||||||
|
?.asSequence()
|
||||||
|
?.forEach { processInfo ->
|
||||||
|
if (processInfo.pid == android.os.Process.myPid()) {
|
||||||
|
processName = processInfo.processName
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (processName != application.packageName) {
|
||||||
|
WebView.setDataDirectorySuffix(processName)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun widget(): Int {
|
||||||
|
return AppInitializerPriority.SERIES_2.priority
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,153 @@
|
||||||
|
package com.remax.visualnovel.app.initializer.impl
|
||||||
|
|
||||||
|
import android.app.Application
|
||||||
|
import android.content.Context
|
||||||
|
import android.net.http.HttpResponseCache
|
||||||
|
import android.os.Environment
|
||||||
|
import android.text.TextUtils
|
||||||
|
import com.chad.library.adapter.base.module.LoadMoreModuleConfig
|
||||||
|
import com.remax.visualnovel.BuildConfig
|
||||||
|
import com.remax.visualnovel.app.delegate.ToolbarViewDelegate
|
||||||
|
import com.remax.visualnovel.app.initializer.AppInitializers
|
||||||
|
import com.remax.visualnovel.app.widget.CustomLoadMoreView
|
||||||
|
import com.remax.visualnovel.app.widget.LoadMoreFooter
|
||||||
|
import com.remax.visualnovel.app.widget.RefreshHeader
|
||||||
|
import com.remax.visualnovel.utils.NotLoggingTree
|
||||||
|
import com.dylanc.loadingstateview.LoadingStateView
|
||||||
|
import com.github.boybeak.skbglobal.SoftKeyboardGlobal
|
||||||
|
import com.github.sahasbhop.apngview.ApngImageLoader
|
||||||
|
import com.lzf.easyfloat.EasyFloat
|
||||||
|
import com.pengxr.modular.eventbus.facade.launcher.IEventListener
|
||||||
|
import com.pengxr.modular.eventbus.facade.launcher.ModularEventBus
|
||||||
|
import com.pengxr.modular.eventbus.facade.template.BaseEvent
|
||||||
|
import com.scwang.smart.refresh.layout.SmartRefreshLayout
|
||||||
|
import timber.log.Timber
|
||||||
|
import java.io.File
|
||||||
|
import java.io.IOException
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Created by HJW on 2023/5/11
|
||||||
|
* 三方库相关初始化
|
||||||
|
*/
|
||||||
|
class ThirdInitializer(val application: Application) : AppInitializers {
|
||||||
|
|
||||||
|
override fun init() {
|
||||||
|
ApngImageLoader.getInstance().init(application)
|
||||||
|
|
||||||
|
if (BuildConfig.DEBUG) {
|
||||||
|
Timber.plant(Timber.DebugTree())
|
||||||
|
} else {
|
||||||
|
Timber.plant(NotLoggingTree())
|
||||||
|
}
|
||||||
|
SoftKeyboardGlobal.install(application, false)
|
||||||
|
|
||||||
|
LoadingStateView.setViewDelegatePool {
|
||||||
|
register(ToolbarViewDelegate())
|
||||||
|
}
|
||||||
|
/**
|
||||||
|
* SVGA http缓存
|
||||||
|
*/
|
||||||
|
val cacheDir = File(application.cacheDir, "http")
|
||||||
|
HttpResponseCache.install(cacheDir, 1024 * 1024 * 200)
|
||||||
|
EasyFloat.init(application, BuildConfig.DEBUG)
|
||||||
|
|
||||||
|
LoadMoreModuleConfig.defLoadMoreView = CustomLoadMoreView()
|
||||||
|
//设置全局默认配置(优先级最低,会被其他设置覆盖)
|
||||||
|
SmartRefreshLayout.setDefaultRefreshInitializer { _, layout -> //全局设置(优先级最低)
|
||||||
|
layout.setEnableLoadMore(false)
|
||||||
|
layout.setEnableAutoLoadMore(false)
|
||||||
|
layout.setEnableOverScrollDrag(true)
|
||||||
|
layout.setEnableOverScrollBounce(true)
|
||||||
|
layout.setEnableLoadMoreWhenContentNotFull(false)
|
||||||
|
layout.setEnableScrollContentWhenRefreshed(true)
|
||||||
|
layout.setHeaderMaxDragRate(3.0f)
|
||||||
|
}
|
||||||
|
SmartRefreshLayout.setDefaultRefreshHeaderCreator { context, layout -> //全局设置header
|
||||||
|
layout.setEnableHeaderTranslationContent(true)
|
||||||
|
RefreshHeader(context)
|
||||||
|
}
|
||||||
|
SmartRefreshLayout.setDefaultRefreshFooterCreator { context, layout -> //全局设置footer
|
||||||
|
layout.setEnableFooterTranslationContent(true)
|
||||||
|
LoadMoreFooter(context)
|
||||||
|
}
|
||||||
|
|
||||||
|
ModularEventBus.debug(BuildConfig.DEBUG)
|
||||||
|
.throwNullEventException(false)
|
||||||
|
.setEventListener(object : IEventListener {
|
||||||
|
override fun <T> onEventPost(eventName: String, event: BaseEvent<T>, data: T?) {
|
||||||
|
Timber.d("ModularEventBus发送事件 eventName: $eventName, event=$event data=$data")
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
//NIMClient.initV2(application, getSDKOptions(application))
|
||||||
|
initLocalNimString()
|
||||||
|
}
|
||||||
|
|
||||||
|
/*private fun getSDKOptions(context: Context): SDKOptions {
|
||||||
|
val options = SDKOptions()
|
||||||
|
with(options) {
|
||||||
|
this.appKey = "2d6abc076f434fc52320c7118de5fead"
|
||||||
|
enableV2CloudConversation = true
|
||||||
|
//在线多端同步未读数
|
||||||
|
sessionReadAck = true
|
||||||
|
// 采用异步加载SDK
|
||||||
|
asyncInitSDK = true
|
||||||
|
//禁止后台进程唤醒UI进程
|
||||||
|
disableAwake = true
|
||||||
|
useXLog = true
|
||||||
|
enableBackOffReconnectStrategy = true
|
||||||
|
checkManifestConfig = BuildConfig.DEBUG
|
||||||
|
loginCustomTag = ClientType.Android.toString()
|
||||||
|
notifyStickTopSession = true
|
||||||
|
mixPushConfig = MixPushConfig().apply {
|
||||||
|
fcmCertificateName = "VisualNovel"
|
||||||
|
}
|
||||||
|
if (BuildConfig.DEBUG) {
|
||||||
|
sdkStorageRootPath = getAppCacheDir(context) + "/nim"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return options
|
||||||
|
}*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 配置 APP 保存图片/语音/文件/log等数据的目录
|
||||||
|
* 这里示例用SD卡的应用扩展存储目录
|
||||||
|
*/
|
||||||
|
private fun getAppCacheDir(context: Context): String? {
|
||||||
|
var storageRootPath: String? = null
|
||||||
|
try {
|
||||||
|
// SD卡应用扩展存储区(APP卸载后,该目录下被清除,用户也可以在设置界面中手动清除),请根据APP对数据缓存的重要性及生命周期来决定是否采用此缓存目录.
|
||||||
|
// 该存储区在API 19以上不需要写权限,即可配置 <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" android:maxSdkVersion="18"/>
|
||||||
|
if (context.externalCacheDir != null) {
|
||||||
|
storageRootPath = context.externalCacheDir!!.canonicalPath
|
||||||
|
}
|
||||||
|
} catch (e: IOException) {
|
||||||
|
e.printStackTrace()
|
||||||
|
}
|
||||||
|
if (TextUtils.isEmpty(storageRootPath)) {
|
||||||
|
// SD卡应用公共存储区(APP卸载后,该目录不会被清除,下载安装APP后,缓存数据依然可以被加载。SDK默认使用此目录),该存储区域需要写权限!
|
||||||
|
storageRootPath = Environment.getExternalStorageDirectory().toString() + "/" + context.packageName
|
||||||
|
}
|
||||||
|
return storageRootPath
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun initLocalNimString() {
|
||||||
|
/*NimStrings.DEFAULT.apply {
|
||||||
|
status_bar_multi_messages_incoming = application.getString(R.string.status_bar_multi_messages_incoming)
|
||||||
|
status_bar_ticker_text = application.getString(R.string.status_bar_ticker_text)
|
||||||
|
status_bar_image_message = application.getString(R.string.status_bar_image_message)
|
||||||
|
status_bar_audio_message = application.getString(R.string.status_bar_audio_message)
|
||||||
|
status_bar_video_message = application.getString(R.string.status_bar_video_message)
|
||||||
|
status_bar_file_message = application.getString(R.string.status_bar_file_message)
|
||||||
|
status_bar_location_message = application.getString(R.string.status_bar_location_message)
|
||||||
|
status_bar_notification_message = application.getString(R.string.status_bar_notification_message)
|
||||||
|
status_bar_avchat_message = application.getString(R.string.status_bar_avchat_message)
|
||||||
|
status_bar_tip_message = application.getString(R.string.status_bar_tip_message)
|
||||||
|
status_bar_custom_message = application.getString(R.string.status_bar_custom_message)
|
||||||
|
status_bar_unsupported_message = application.getString(R.string.status_bar_unsupported_message)
|
||||||
|
status_bar_hidden_message_content = application.getString(R.string.status_bar_hidden_message_content)
|
||||||
|
status_bar_hidden_message_title = application.getString(R.string.status_bar_hidden_message_title)
|
||||||
|
}*/
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,20 @@
|
||||||
|
package com.remax.visualnovel.app.initializer.impl
|
||||||
|
|
||||||
|
import android.app.Application
|
||||||
|
import com.remax.visualnovel.manager.login.LoginManager
|
||||||
|
import com.remax.visualnovel.app.initializer.AppInitializers
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Created by HJW on 2023/5/11
|
||||||
|
* user相关初始化
|
||||||
|
*/
|
||||||
|
class UserInitializer(val application: Application) : AppInitializers {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 初始化必须放在本地数据存储框架初始化之后才行
|
||||||
|
*/
|
||||||
|
override fun init() {
|
||||||
|
LoginManager.init()
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,29 @@
|
||||||
|
package com.remax.visualnovel.app.initializer.utils
|
||||||
|
|
||||||
|
import com.google.android.gms.tasks.Task
|
||||||
|
import com.google.firebase.messaging.FirebaseMessaging
|
||||||
|
import timber.log.Timber
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Created by HJW on 2022/10/18
|
||||||
|
*/
|
||||||
|
object FirebaseHelper {
|
||||||
|
|
||||||
|
fun getToken(tokenCallback: ((String?) -> Unit)? = null) {
|
||||||
|
FirebaseMessaging.getInstance().token.addOnCompleteListener { task: Task<String?> ->
|
||||||
|
try {
|
||||||
|
Timber.d("firebase CompleteListener isSuccessful : %s", task.isSuccessful)
|
||||||
|
if (task.isSuccessful && task.result != null) {
|
||||||
|
val token = task.result
|
||||||
|
Timber.d("firebase token : %s", token)
|
||||||
|
tokenCallback?.invoke(token)
|
||||||
|
} else {
|
||||||
|
tokenCallback?.invoke(null)
|
||||||
|
}
|
||||||
|
} catch (e: Exception) {
|
||||||
|
Timber.d("firebase token exception: %s", e.localizedMessage)
|
||||||
|
tokenCallback?.invoke(null)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,37 @@
|
||||||
|
package com.remax.visualnovel.app.viewmodel
|
||||||
|
|
||||||
|
import android.app.Application
|
||||||
|
import androidx.lifecycle.AndroidViewModel
|
||||||
|
import androidx.lifecycle.viewModelScope
|
||||||
|
import com.remax.visualnovel.manager.login.LoginManager
|
||||||
|
import com.remax.visualnovel.repository.api.MessageRepository
|
||||||
|
import com.remax.visualnovel.repository.api.UserRepository
|
||||||
|
|
||||||
|
import dagger.hilt.android.lifecycle.HiltViewModel
|
||||||
|
import kotlinx.coroutines.launch
|
||||||
|
import javax.inject.Inject
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Created by HJW on 2022/11/1
|
||||||
|
*/
|
||||||
|
@HiltViewModel
|
||||||
|
class AppGlobalViewModel @Inject constructor(
|
||||||
|
application: Application,
|
||||||
|
private val userRepository: UserRepository,
|
||||||
|
private val messageRepository: MessageRepository
|
||||||
|
) : AndroidViewModel(application) {
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 更新firebase设备码到后端,推送用
|
||||||
|
*/
|
||||||
|
fun updateTerminal(terminalCode: String?) {
|
||||||
|
if (LoginManager.isLogin) {
|
||||||
|
viewModelScope.launch {
|
||||||
|
// userRepository.updateTerminal(terminalCode)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,32 @@
|
||||||
|
package com.remax.visualnovel.app.viewmodel
|
||||||
|
|
||||||
|
import android.app.Application
|
||||||
|
import androidx.lifecycle.AndroidViewModel
|
||||||
|
import androidx.lifecycle.DefaultLifecycleObserver
|
||||||
|
import androidx.lifecycle.LifecycleOwner
|
||||||
|
import com.google.gson.Gson
|
||||||
|
import com.remax.visualnovel.utils.TimeUtils
|
||||||
|
|
||||||
|
/*import com.netease.nimlib.sdk.NIMClient
|
||||||
|
import com.netease.nimlib.sdk.Observer
|
||||||
|
import com.netease.nimlib.sdk.StatusCode
|
||||||
|
import com.netease.nimlib.sdk.msg.MsgServiceObserve
|
||||||
|
import com.netease.nimlib.sdk.msg.constant.MsgTypeEnum
|
||||||
|
import com.netease.nimlib.sdk.msg.model.IMMessage*/
|
||||||
|
import dagger.hilt.android.lifecycle.HiltViewModel
|
||||||
|
import kotlinx.coroutines.flow.MutableSharedFlow
|
||||||
|
import timber.log.Timber
|
||||||
|
import javax.inject.Inject
|
||||||
|
import kotlin.math.abs
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Created by HJW on 2022/10/18
|
||||||
|
* [Application]生命周期内的[AndroidViewModel]
|
||||||
|
*/
|
||||||
|
@HiltViewModel
|
||||||
|
class AppIMViewModel @Inject constructor(application: Application) : AndroidViewModel(application),
|
||||||
|
DefaultLifecycleObserver {
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,16 @@
|
||||||
|
package com.remax.visualnovel.app.viewmodel.base
|
||||||
|
|
||||||
|
import com.remax.visualnovel.app.viewmodel.base.UserViewModel
|
||||||
|
import dagger.hilt.android.lifecycle.HiltViewModel
|
||||||
|
import javax.inject.Inject
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Created by HJW on 2022/10/28
|
||||||
|
*
|
||||||
|
* 应用相关viewmodel的父类
|
||||||
|
*/
|
||||||
|
@HiltViewModel
|
||||||
|
open class AppViewModel @Inject constructor() : UserViewModel() {
|
||||||
|
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,16 @@
|
||||||
|
package com.remax.visualnovel.app.viewmodel.base
|
||||||
|
|
||||||
|
import androidx.lifecycle.ViewModel
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Created by HJW on 2022/10/27
|
||||||
|
*/
|
||||||
|
open class BaseViewModel : ViewModel() {
|
||||||
|
|
||||||
|
open fun onStart() {
|
||||||
|
}
|
||||||
|
|
||||||
|
open fun onDestroy() {
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -0,0 +1,271 @@
|
||||||
|
package com.remax.visualnovel.app.viewmodel.base
|
||||||
|
|
||||||
|
import android.graphics.BitmapFactory
|
||||||
|
import com.amazonaws.auth.BasicSessionCredentials
|
||||||
|
import com.amazonaws.mobileconnectors.s3.transferutility.TransferListener
|
||||||
|
import com.amazonaws.mobileconnectors.s3.transferutility.TransferState
|
||||||
|
import com.amazonaws.mobileconnectors.s3.transferutility.TransferUtility
|
||||||
|
import com.amazonaws.services.s3.AmazonS3Client
|
||||||
|
import com.amazonaws.services.s3.S3ClientOptions
|
||||||
|
import com.amazonaws.services.s3.model.ObjectMetadata
|
||||||
|
import com.remax.visualnovel.R
|
||||||
|
import com.remax.visualnovel.app.base.app.CommonApplicationProxy
|
||||||
|
import com.remax.visualnovel.constant.StatusCode
|
||||||
|
import com.remax.visualnovel.entity.request.ImgCheckDTO
|
||||||
|
import com.remax.visualnovel.entity.response.BucketBean
|
||||||
|
import com.remax.visualnovel.entity.response.base.ApiFailedResponse
|
||||||
|
import com.remax.visualnovel.entity.response.base.ApiSuccessResponse
|
||||||
|
import com.remax.visualnovel.entity.response.base.Response
|
||||||
|
import com.remax.visualnovel.extension.resumeWithActive
|
||||||
|
import com.remax.visualnovel.repository.api.OssRepository
|
||||||
|
import dagger.hilt.android.lifecycle.HiltViewModel
|
||||||
|
import kotlinx.coroutines.suspendCancellableCoroutine
|
||||||
|
import timber.log.Timber
|
||||||
|
import java.io.File
|
||||||
|
import javax.inject.Inject
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Created by HJW on 2022/11/9
|
||||||
|
*
|
||||||
|
* oss上传相关
|
||||||
|
*/
|
||||||
|
@HiltViewModel
|
||||||
|
open class OssViewModel @Inject constructor() : UserViewModel() {
|
||||||
|
|
||||||
|
@Inject
|
||||||
|
lateinit var ossRepository: OssRepository
|
||||||
|
|
||||||
|
data class LoadFileData(
|
||||||
|
var isSuccess: Boolean,
|
||||||
|
val isViolation: Boolean = false,
|
||||||
|
val errorMsg: String = "",
|
||||||
|
var urlPath: String = "",
|
||||||
|
var filePath: String = "",
|
||||||
|
var width: Int = 0,
|
||||||
|
var height: Int = 0
|
||||||
|
)
|
||||||
|
|
||||||
|
data class FileUpLoadRes(
|
||||||
|
val loadFileData: LoadFileData,
|
||||||
|
val fileOption: FileOption? = null,
|
||||||
|
)
|
||||||
|
|
||||||
|
data class FileOption(
|
||||||
|
val path: String,
|
||||||
|
val urlPath: String,
|
||||||
|
val ossType: String,
|
||||||
|
val width: Int,
|
||||||
|
val height: Int
|
||||||
|
)
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 请求oss的token
|
||||||
|
*/
|
||||||
|
suspend fun getBucketToken(postfix: String, ossType: String): Response<BucketBean> {
|
||||||
|
return ossRepository.getS3Bucket(ossType, postfix)
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 挂起函数上传图片
|
||||||
|
* @param filePath String
|
||||||
|
* @param ossType String
|
||||||
|
* @param isImg Boolean
|
||||||
|
* @param checkNSFW Boolean 是否鉴黄
|
||||||
|
* @param checkRealPerson Boolean 是否鉴定真人
|
||||||
|
* @param checkKid Boolean 是否鉴定儿童
|
||||||
|
* @return Response<LoadFileData> 封装成服务器返回一致类型处理
|
||||||
|
*/
|
||||||
|
suspend fun ossUploadFile(
|
||||||
|
filePath: String,
|
||||||
|
ossType: String,
|
||||||
|
isImg: Boolean = true,
|
||||||
|
checkNSFW: Boolean = true,
|
||||||
|
checkRealPerson: Boolean = false,
|
||||||
|
checkKid: Boolean = false,
|
||||||
|
token: BucketBean? = null
|
||||||
|
): Response<LoadFileData> {
|
||||||
|
/**
|
||||||
|
* 获取S3 STS Token对象
|
||||||
|
*/
|
||||||
|
var s3BucketRes = token
|
||||||
|
if (s3BucketRes == null) {
|
||||||
|
val postfix = if (filePath.isNotEmpty()) filePath.substring(filePath.lastIndexOf(".") + 1) else "png"
|
||||||
|
val getTokenRes = getBucketToken(postfix, ossType)
|
||||||
|
//请求s3 token失败
|
||||||
|
if (!getTokenRes.isApiSuccess) {
|
||||||
|
return ApiFailedResponse(
|
||||||
|
errorMsg = getTokenRes.errorMsg,
|
||||||
|
errorData = createNormalErrorFileData(filePath).loadFileData
|
||||||
|
)
|
||||||
|
} else {
|
||||||
|
s3BucketRes = getTokenRes.data!!
|
||||||
|
}
|
||||||
|
}
|
||||||
|
val uploadRes = uploadFile(s3BucketRes, isImg, filePath, ossType)
|
||||||
|
//上传图片失败
|
||||||
|
if (!uploadRes.loadFileData.isSuccess) {
|
||||||
|
return ApiFailedResponse(errorMsg = uploadRes.loadFileData.errorMsg, errorData = uploadRes.loadFileData)
|
||||||
|
}
|
||||||
|
//如果不是图片 或者 不需要鉴黄、鉴定真人、鉴定儿童,直接返回成功结果
|
||||||
|
if (!isImg || (!checkNSFW && !checkRealPerson && !checkKid)) {
|
||||||
|
return ApiSuccessResponse(uploadRes.loadFileData)
|
||||||
|
}
|
||||||
|
val fileOption = uploadRes.fileOption!!
|
||||||
|
val checkDTO = ImgCheckDTO(fileOption.ossType, fileOption.path)
|
||||||
|
//鉴黄
|
||||||
|
val checkNSFWRes = if (checkNSFW) ossRepository.checkS3Img(checkDTO) else ApiSuccessResponse()
|
||||||
|
return when {
|
||||||
|
checkNSFWRes.isApiSuccess -> {
|
||||||
|
ApiSuccessResponse(uploadRes.loadFileData)
|
||||||
|
}
|
||||||
|
|
||||||
|
else -> {
|
||||||
|
ApiFailedResponse(StatusCode.UPLOAD_FILE_VIOLATION.code, checkNSFWRes.errorMsg, uploadRes.loadFileData)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 包装上传失败的实体
|
||||||
|
* @param filePath String 本地图片地址
|
||||||
|
* @return FileUpLoadRes
|
||||||
|
*/
|
||||||
|
private fun createNormalErrorFileData(filePath: String) = FileUpLoadRes(
|
||||||
|
LoadFileData(
|
||||||
|
isSuccess = false,
|
||||||
|
errorMsg = CommonApplicationProxy.application.getString(R.string.upload_error),
|
||||||
|
filePath = filePath
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 协程处理亚马逊上传图片
|
||||||
|
*
|
||||||
|
* 使用带取消回调的协程,当上传回调时,需要判断协程isActive以防报错崩溃
|
||||||
|
* @param stsToken BucketBean 授权信息
|
||||||
|
* @param isImg Boolean 是否是图片
|
||||||
|
* @param filePath String 本地地址
|
||||||
|
* @param ossType String 上传类型
|
||||||
|
* @return FileUpLoadRes 返回结果封装
|
||||||
|
*/
|
||||||
|
private suspend fun uploadFile(
|
||||||
|
stsToken: BucketBean,
|
||||||
|
isImg: Boolean,
|
||||||
|
filePath: String,
|
||||||
|
ossType: String,
|
||||||
|
) = suspendCancellableCoroutine {
|
||||||
|
it.invokeOnCancellation { _ ->
|
||||||
|
it.resumeWithActive(createNormalErrorFileData(filePath))
|
||||||
|
}
|
||||||
|
|
||||||
|
val awsCreds = BasicSessionCredentials(
|
||||||
|
stsToken.accessKeyId,
|
||||||
|
stsToken.accessKeySecret,
|
||||||
|
stsToken.securityToken
|
||||||
|
)
|
||||||
|
val uploadClient = AmazonS3Client(
|
||||||
|
awsCreds,
|
||||||
|
com.amazonaws.regions.Region.getRegion(stsToken.region)
|
||||||
|
).apply {
|
||||||
|
setS3ClientOptions(
|
||||||
|
S3ClientOptions.builder()
|
||||||
|
.setAccelerateModeEnabled(false)
|
||||||
|
.build()
|
||||||
|
)
|
||||||
|
}
|
||||||
|
val transferUtility = TransferUtility.builder()
|
||||||
|
.s3Client(uploadClient)
|
||||||
|
.context(CommonApplicationProxy.application)
|
||||||
|
.build()
|
||||||
|
|
||||||
|
val fileName = filePath.substring(filePath.lastIndexOf("/") + 1)
|
||||||
|
val path = if (stsToken.path.endsWith("*")) {
|
||||||
|
stsToken.path.replace("*", fileName)
|
||||||
|
} else {
|
||||||
|
stsToken.path
|
||||||
|
}
|
||||||
|
val urlPath = if (stsToken.urlPath.endsWith("*")) {
|
||||||
|
stsToken.urlPath.replace("*", fileName)
|
||||||
|
} else {
|
||||||
|
stsToken.urlPath
|
||||||
|
}
|
||||||
|
Timber.d("oss上传 - AmazonS3 Token path:$path urlPath:$urlPath")
|
||||||
|
|
||||||
|
val obj = ObjectMetadata()
|
||||||
|
obj.addUserMetadata("x-amz-tagging", "temp=1")
|
||||||
|
|
||||||
|
val transferListener = object :
|
||||||
|
TransferListener {
|
||||||
|
override fun onStateChanged(id: Int, state: TransferState?) {
|
||||||
|
Timber.d("oss上传 - AmazonS3 onStateChanged:$state")
|
||||||
|
Timber.d("oss上传 - 协程状态 isActive: ${it.isActive} isCancelled: ${it.isCancelled} isCompleted: ${it.isCompleted}")
|
||||||
|
when (state) {
|
||||||
|
TransferState.COMPLETED -> {
|
||||||
|
//此方法是上传图片完成后再打标签
|
||||||
|
// uploadClient.setObjectTagging(SetObjectTaggingRequest(stsToken.bucket, stsToken.path, ObjectTagging(listOf(Tag("temp", "1")))))
|
||||||
|
if (it.isActive) {
|
||||||
|
if (isImg) {
|
||||||
|
val options = BitmapFactory.Options()
|
||||||
|
options.inJustDecodeBounds = true
|
||||||
|
BitmapFactory.decodeFile(filePath, options)
|
||||||
|
val wid = options.outWidth
|
||||||
|
val hei = options.outHeight
|
||||||
|
val res = FileUpLoadRes(
|
||||||
|
LoadFileData(
|
||||||
|
isSuccess = true,
|
||||||
|
urlPath = urlPath,
|
||||||
|
width = wid,
|
||||||
|
height = hei,
|
||||||
|
filePath = filePath
|
||||||
|
),
|
||||||
|
FileOption(path, urlPath, ossType, wid, hei)
|
||||||
|
)
|
||||||
|
it.resumeWithActive(res)
|
||||||
|
} else {
|
||||||
|
val res = FileUpLoadRes(
|
||||||
|
LoadFileData(
|
||||||
|
isSuccess = true,
|
||||||
|
urlPath = urlPath,
|
||||||
|
filePath = filePath
|
||||||
|
)
|
||||||
|
)
|
||||||
|
it.resumeWithActive(res)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
TransferState.FAILED, TransferState.CANCELED -> {
|
||||||
|
it.resumeWithActive(createNormalErrorFileData(filePath))
|
||||||
|
}
|
||||||
|
|
||||||
|
else -> {
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun onProgressChanged(id: Int, bytesCurrent: Long, bytesTotal: Long) {
|
||||||
|
Timber.d("oss上传 - AmazonS3 onProgressChanged - bytesTotal:${bytesTotal} - bytesCurrent:${bytesCurrent}")
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun onError(id: Int, ex: Exception?) {
|
||||||
|
Timber.d("oss上传 - AmazonS3 onError:${ex?.localizedMessage} - id:$id")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
when {
|
||||||
|
filePath.isNotEmpty() -> {
|
||||||
|
val file = File(filePath)
|
||||||
|
if (!file.exists()) {
|
||||||
|
it.cancel()
|
||||||
|
}
|
||||||
|
Timber.d("oss上传 - 上传文件大小 ${file.length() / 1024}")
|
||||||
|
transferUtility.upload(stsToken.bucket, path, file, obj)
|
||||||
|
.setTransferListener(transferListener)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
suspend fun checkText(content: String): Response<Any> = ossRepository.checkText(content)
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,28 @@
|
||||||
|
package com.remax.visualnovel.app.viewmodel.base
|
||||||
|
|
||||||
|
import dagger.hilt.android.lifecycle.HiltViewModel
|
||||||
|
import javax.inject.Inject
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Created by HJW on 2022/10/28
|
||||||
|
*
|
||||||
|
* 钱包。支付相关父类
|
||||||
|
*/
|
||||||
|
@HiltViewModel
|
||||||
|
open class PayViewModel @Inject constructor() : BaseViewModel() {
|
||||||
|
|
||||||
|
// @Inject
|
||||||
|
// lateinit var walletRepository: WalletRepository
|
||||||
|
//
|
||||||
|
// @Inject
|
||||||
|
// lateinit var payRepository: PayRepository
|
||||||
|
//
|
||||||
|
// private val _walletFlow = MutableSharedFlow<Response<Wallet>>()
|
||||||
|
// val walletFlow = _walletFlow.asSharedFlow()
|
||||||
|
//
|
||||||
|
// suspend fun getMyWallet(): Response<Wallet> {
|
||||||
|
// return walletRepository.getMyWallet().apply { _walletFlow.emit(this) }
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
// suspend fun checkOut(tradeNo: String) = payRepository.checkOut(tradeNo)
|
||||||
|
}
|
|
@ -0,0 +1,72 @@
|
||||||
|
package com.remax.visualnovel.app.viewmodel.base
|
||||||
|
|
||||||
|
|
||||||
|
import com.remax.visualnovel.R
|
||||||
|
import com.remax.visualnovel.app.base.app.CommonApplicationProxy
|
||||||
|
import com.remax.visualnovel.entity.response.User
|
||||||
|
import com.remax.visualnovel.entity.response.base.ApiFailedResponse
|
||||||
|
import com.remax.visualnovel.entity.response.base.ApiSuccessResponse
|
||||||
|
import com.remax.visualnovel.entity.response.base.Response
|
||||||
|
import com.remax.visualnovel.repository.api.LoginRepository
|
||||||
|
import com.remax.visualnovel.repository.api.UserRepository
|
||||||
|
import dagger.hilt.android.lifecycle.HiltViewModel
|
||||||
|
import kotlinx.coroutines.flow.MutableStateFlow
|
||||||
|
import kotlinx.coroutines.flow.StateFlow
|
||||||
|
import kotlinx.coroutines.flow.asStateFlow
|
||||||
|
import javax.inject.Inject
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Created by HJW on 2022/10/28
|
||||||
|
*
|
||||||
|
* User相关viewmodel的父类
|
||||||
|
*/
|
||||||
|
@HiltViewModel
|
||||||
|
open class UserViewModel @Inject constructor() : BaseViewModel() {
|
||||||
|
|
||||||
|
@Inject
|
||||||
|
lateinit var userRepository: UserRepository
|
||||||
|
|
||||||
|
private val _userInfoFlow = MutableStateFlow<Response<User>>(Response())
|
||||||
|
val userInfoFlow: StateFlow<Response<User>> = _userInfoFlow.asStateFlow()
|
||||||
|
|
||||||
|
suspend fun getMyBaseInfo(): Response<User> {
|
||||||
|
return userRepository.getMyBaseInfo().also { _userInfoFlow.value = it }
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 公共返回,很多接口操作后需要再次请求用户数据
|
||||||
|
*/
|
||||||
|
protected suspend fun returnUserResponse(response: Response<*>): Response<User> {
|
||||||
|
return if (response.isApiSuccess) {
|
||||||
|
getMyBaseInfo()
|
||||||
|
} else {
|
||||||
|
ApiFailedResponse(response.errorCode, response.errorMsg)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//suspend fun getNimInfo() = userRepository.getNimInfo()
|
||||||
|
|
||||||
|
@Inject
|
||||||
|
lateinit var loginRepository: LoginRepository
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 公共返回,检查nickname是否存在
|
||||||
|
*/
|
||||||
|
suspend fun <T> checkNickname(
|
||||||
|
nickName: String?,
|
||||||
|
exUserId: String? = null,
|
||||||
|
apiCall: (suspend () -> Response<T>)? = null
|
||||||
|
): Response<T> {
|
||||||
|
val checkRes = loginRepository.checkUserNickname(nickName, exUserId)
|
||||||
|
return if (checkRes.isApiSuccess) {
|
||||||
|
if (checkRes.data == true) {
|
||||||
|
ApiFailedResponse("", CommonApplicationProxy.application.getString(R.string.nickname_exist_error))
|
||||||
|
} else {
|
||||||
|
apiCall?.invoke() ?: ApiSuccessResponse()
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
ApiFailedResponse(checkRes.errorCode, checkRes.errorMsg)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,40 @@
|
||||||
|
package com.remax.visualnovel.app.widget
|
||||||
|
|
||||||
|
import android.view.LayoutInflater
|
||||||
|
import android.view.View
|
||||||
|
import android.view.ViewGroup
|
||||||
|
import com.chad.library.adapter.base.loadmore.BaseLoadMoreView
|
||||||
|
import com.chad.library.adapter.base.viewholder.BaseViewHolder
|
||||||
|
import com.remax.visualnovel.R
|
||||||
|
import com.remax.visualnovel.extension.setSize
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Created by HJW on 2020/9/18
|
||||||
|
*/
|
||||||
|
class CustomLoadMoreView(private val showLoading: Boolean = true) : BaseLoadMoreView() {
|
||||||
|
|
||||||
|
override fun getLoadComplete(holder: BaseViewHolder): View {
|
||||||
|
return holder.getView(R.id.load_more_load_complete_view)
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun getLoadEndView(holder: BaseViewHolder): View {
|
||||||
|
return holder.getView(R.id.load_more_load_end_view)
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun getLoadFailView(holder: BaseViewHolder): View {
|
||||||
|
return holder.getView(R.id.load_more_load_fail_view)
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun getLoadingView(holder: BaseViewHolder): View {
|
||||||
|
val loadingView = holder.getView<View>(R.id.load_more_loading_view)
|
||||||
|
if (!showLoading) {
|
||||||
|
loadingView.setSize(height = 0)
|
||||||
|
}
|
||||||
|
return loadingView
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun getRootView(parent: ViewGroup): View {
|
||||||
|
// 整个 LoadMore 布局
|
||||||
|
return LayoutInflater.from(parent.context).inflate(R.layout.view_load_more_common, parent, false)
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,113 @@
|
||||||
|
package com.remax.visualnovel.app.widget
|
||||||
|
|
||||||
|
import android.content.Context
|
||||||
|
import android.view.LayoutInflater
|
||||||
|
import android.view.View
|
||||||
|
import androidx.annotation.DrawableRes
|
||||||
|
import androidx.annotation.StringRes
|
||||||
|
import androidx.core.view.isVisible
|
||||||
|
import com.chad.library.adapter.base.BaseQuickAdapter
|
||||||
|
import com.drake.brv.PageRefreshLayout
|
||||||
|
import com.remax.visualnovel.R
|
||||||
|
import com.remax.visualnovel.configs.NovelApplication
|
||||||
|
import com.remax.visualnovel.databinding.LayoutEmptyBinding
|
||||||
|
import com.remax.visualnovel.extension.setOnClick
|
||||||
|
import com.remax.visualnovel.utils.spannablex.utils.dp
|
||||||
|
|
||||||
|
|
||||||
|
fun PageRefreshLayout.setEmptyText(
|
||||||
|
@StringRes emptyTextResId: Int = 0,
|
||||||
|
@DrawableRes emptyIcon: Int = R.mipmap.icon_new_empty,
|
||||||
|
topMargin: Int? = null,
|
||||||
|
) {
|
||||||
|
emptyLayout = R.layout.layout_empty
|
||||||
|
stateEnabled = true
|
||||||
|
onEmpty {
|
||||||
|
LayoutEmptyBinding.bind(this).apply {
|
||||||
|
if (emptyIcon != 0) {
|
||||||
|
ivEmpty.setImageResource(emptyIcon)
|
||||||
|
ivEmpty.isVisible = true
|
||||||
|
}
|
||||||
|
if (emptyTextResId != 0) {
|
||||||
|
tvEmpty.setText(emptyTextResId)
|
||||||
|
tvEmpty.isVisible = true
|
||||||
|
}
|
||||||
|
topMargin?.let {
|
||||||
|
root.run {
|
||||||
|
setPaddingRelative(paddingStart, topMargin.dp, paddingEnd, paddingBottom)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fun BaseQuickAdapter<*, *>.setMyEmptyView(
|
||||||
|
@StringRes emptyTextResId: Int = 0,
|
||||||
|
@DrawableRes emptyIcon: Int = R.mipmap.icon_new_empty,
|
||||||
|
topMargin: Int? = null,
|
||||||
|
@StringRes btnText: Int = 0,
|
||||||
|
btnInvoke: (() -> Unit)? = null
|
||||||
|
) {
|
||||||
|
NovelApplication.getCurrentActivity()?.let { activity ->
|
||||||
|
setEmptyView(
|
||||||
|
EmptyView(activity).createEmptyView(emptyTextResId, emptyIcon, topMargin, btnText, btnInvoke)
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class EmptyView(private val context: Context) {
|
||||||
|
|
||||||
|
private val viewBind: LayoutEmptyBinding = LayoutEmptyBinding.inflate(LayoutInflater.from(context))
|
||||||
|
|
||||||
|
fun createEmptyView(
|
||||||
|
@StringRes emptyTextResId: Int = 0,
|
||||||
|
@DrawableRes emptyIcon: Int = R.mipmap.icon_new_empty,
|
||||||
|
topMargin: Int? = null,
|
||||||
|
@StringRes btnText: Int = 0,
|
||||||
|
btnInvoke: (() -> Unit)? = null
|
||||||
|
): View {
|
||||||
|
setEmptyView(emptyIcon, emptyTextResId)
|
||||||
|
topMargin?.let {
|
||||||
|
setMarginTop(it.dp)
|
||||||
|
}
|
||||||
|
if (btnText != 0) {
|
||||||
|
setBtn(btnText, btnInvoke)
|
||||||
|
}
|
||||||
|
return viewBind.root
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun setEmptyView(@DrawableRes emptyIcon: Int = R.mipmap.icon_new_empty, @StringRes emptyTextResId: Int = 0): EmptyView {
|
||||||
|
with(viewBind) {
|
||||||
|
ivEmpty.isVisible = false
|
||||||
|
emptyIcon.takeIf { it != 0 }?.let {
|
||||||
|
ivEmpty.setImageResource(it)
|
||||||
|
ivEmpty.isVisible = true
|
||||||
|
}
|
||||||
|
tvEmpty.isVisible = false
|
||||||
|
emptyTextResId.takeIf { it != 0 }?.let {
|
||||||
|
tvEmpty.text = context.getString(emptyTextResId)
|
||||||
|
tvEmpty.isVisible = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return this
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun setBtn(@StringRes btnText: Int, btnInvoke: (() -> Unit)? = null): EmptyView {
|
||||||
|
with(viewBind.btnEmptyMessage) {
|
||||||
|
isVisible = true
|
||||||
|
text = context.getString(btnText)
|
||||||
|
setOnClick(this) {
|
||||||
|
btnInvoke?.invoke()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return this
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun setMarginTop(topMargin: Int): EmptyView {
|
||||||
|
viewBind.root.run {
|
||||||
|
setPaddingRelative(paddingStart, topMargin, paddingEnd, paddingBottom)
|
||||||
|
}
|
||||||
|
return this
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,92 @@
|
||||||
|
package com.remax.visualnovel.app.widget;
|
||||||
|
|
||||||
|
import android.content.Context;
|
||||||
|
import android.util.AttributeSet;
|
||||||
|
import android.view.View;
|
||||||
|
import android.widget.LinearLayout;
|
||||||
|
|
||||||
|
import androidx.annotation.NonNull;
|
||||||
|
import androidx.annotation.Nullable;
|
||||||
|
|
||||||
|
import com.remax.visualnovel.R;
|
||||||
|
import com.scwang.smart.refresh.layout.api.RefreshKernel;
|
||||||
|
import com.scwang.smart.refresh.layout.api.RefreshLayout;
|
||||||
|
import com.scwang.smart.refresh.layout.constant.RefreshState;
|
||||||
|
import com.scwang.smart.refresh.layout.constant.SpinnerStyle;
|
||||||
|
import com.scwang.smart.refresh.layout.util.SmartUtil;
|
||||||
|
|
||||||
|
public class LoadMoreFooter extends LinearLayout implements com.scwang.smart.refresh.layout.api.RefreshFooter {
|
||||||
|
|
||||||
|
public LoadMoreFooter(Context context) {
|
||||||
|
this(context, null);
|
||||||
|
}
|
||||||
|
|
||||||
|
public LoadMoreFooter(Context context, @Nullable AttributeSet attrs) {
|
||||||
|
super(context, attrs, 0);
|
||||||
|
View.inflate(context, R.layout.load_more_loading_view, this);
|
||||||
|
setMinimumHeight(SmartUtil.dp2px(50));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean setNoMoreData(boolean noMoreData) {
|
||||||
|
this.setVisibility(noMoreData ? View.GONE : View.VISIBLE);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
@NonNull
|
||||||
|
@Override
|
||||||
|
public View getView() {
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
@NonNull
|
||||||
|
@Override
|
||||||
|
public SpinnerStyle getSpinnerStyle() {
|
||||||
|
return SpinnerStyle.Translate;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void setPrimaryColors(int... colors) {
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onInitialized(@NonNull RefreshKernel kernel, int height, int maxDragHeight) {
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onMoving(boolean isDragging, float percent, int offset, int height, int maxDragHeight) {
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onReleased(@NonNull RefreshLayout refreshLayout, int height, int maxDragHeight) {
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onStartAnimator(@NonNull RefreshLayout refreshLayout, int height, int maxDragHeight) {
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int onFinish(@NonNull RefreshLayout refreshLayout, boolean success) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onHorizontalDrag(float percentX, int offsetX, int offsetMax) {
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean isSupportHorizontalDrag() {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onStateChanged(@NonNull RefreshLayout refreshLayout, @NonNull RefreshState oldState, @NonNull RefreshState newState) {
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,40 @@
|
||||||
|
package com.remax.visualnovel.app.widget
|
||||||
|
|
||||||
|
import android.content.Context
|
||||||
|
import com.remax.visualnovel.databinding.DialogLoadingBinding
|
||||||
|
import com.remax.visualnovel.widget.dialoglib.LBindingDialog
|
||||||
|
|
||||||
|
class LoadingDialog {
|
||||||
|
|
||||||
|
private lateinit var dialog: LBindingDialog<DialogLoadingBinding>
|
||||||
|
private var context: Context? = null
|
||||||
|
|
||||||
|
fun build(context: Context): LoadingDialog {
|
||||||
|
this.context = context
|
||||||
|
dialog = LBindingDialog(context,DialogLoadingBinding::inflate)
|
||||||
|
dialog.with()
|
||||||
|
.setCenter()
|
||||||
|
.setBgRadius(4)
|
||||||
|
.setWidth(80)
|
||||||
|
.setHeight(80)
|
||||||
|
dialog.setCancelable(false)
|
||||||
|
dialog.setCanceledOnTouchOutside(false)
|
||||||
|
return this
|
||||||
|
}
|
||||||
|
|
||||||
|
fun getDialog(): LBindingDialog<DialogLoadingBinding> {
|
||||||
|
return dialog
|
||||||
|
}
|
||||||
|
|
||||||
|
fun show() {
|
||||||
|
if (this::dialog.isInitialized) {
|
||||||
|
dialog.show()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fun dismiss() {
|
||||||
|
if (this::dialog.isInitialized) {
|
||||||
|
dialog.dismiss()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,109 @@
|
||||||
|
package com.remax.visualnovel.app.widget;
|
||||||
|
|
||||||
|
import android.content.Context;
|
||||||
|
import android.util.AttributeSet;
|
||||||
|
import android.view.Gravity;
|
||||||
|
import android.view.View;
|
||||||
|
import android.widget.LinearLayout;
|
||||||
|
|
||||||
|
import androidx.annotation.NonNull;
|
||||||
|
import androidx.annotation.Nullable;
|
||||||
|
|
||||||
|
import com.airbnb.lottie.LottieAnimationView;
|
||||||
|
import com.remax.visualnovel.R;
|
||||||
|
import com.scwang.smart.refresh.layout.api.RefreshKernel;
|
||||||
|
import com.scwang.smart.refresh.layout.api.RefreshLayout;
|
||||||
|
import com.scwang.smart.refresh.layout.constant.RefreshState;
|
||||||
|
import com.scwang.smart.refresh.layout.constant.SpinnerStyle;
|
||||||
|
import com.scwang.smart.refresh.layout.util.SmartUtil;
|
||||||
|
|
||||||
|
public class RefreshHeader extends LinearLayout implements com.scwang.smart.refresh.layout.api.RefreshHeader {
|
||||||
|
private final LottieAnimationView mProgressView;//刷新动画视图
|
||||||
|
|
||||||
|
public RefreshHeader(Context context) {
|
||||||
|
this(context, null);
|
||||||
|
}
|
||||||
|
|
||||||
|
public RefreshHeader(Context context, @Nullable AttributeSet attrs) {
|
||||||
|
super(context, attrs, 0);
|
||||||
|
setGravity(Gravity.CENTER);
|
||||||
|
mProgressView = new LottieAnimationView(context);
|
||||||
|
mProgressView.setAnimation(R.raw.single_ring);
|
||||||
|
mProgressView.setSafeMode(true);
|
||||||
|
mProgressView.setRepeatCount(-1);
|
||||||
|
addView(mProgressView, SmartUtil.dp2px(26), SmartUtil.dp2px(26));
|
||||||
|
setMinimumHeight(SmartUtil.dp2px(30));
|
||||||
|
}
|
||||||
|
|
||||||
|
@NonNull
|
||||||
|
@Override
|
||||||
|
public View getView() {
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
@NonNull
|
||||||
|
@Override
|
||||||
|
public SpinnerStyle getSpinnerStyle() {
|
||||||
|
return SpinnerStyle.Translate;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void setPrimaryColors(int... colors) {
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onInitialized(@NonNull RefreshKernel kernel, int height, int maxDragHeight) {
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onMoving(boolean isDragging, float percent, int offset, int height, int maxDragHeight) {
|
||||||
|
if (isDragging){
|
||||||
|
float currPer = percent / 3.0f;
|
||||||
|
if (currPer > 1.0f) currPer = 1.0f;
|
||||||
|
float frameF = mProgressView.getMaxFrame() * currPer;
|
||||||
|
int frame = (int) (frameF);
|
||||||
|
mProgressView.setFrame(frame);
|
||||||
|
mProgressView.invalidate();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onReleased(@NonNull RefreshLayout refreshLayout, int height, int maxDragHeight) {
|
||||||
|
mProgressView.setRepeatCount(-1);
|
||||||
|
mProgressView.setProgress(0f);
|
||||||
|
mProgressView.playAnimation();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onStartAnimator(@NonNull RefreshLayout refreshLayout, int height, int maxDragHeight) {
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int onFinish(@NonNull RefreshLayout refreshLayout, boolean success) {
|
||||||
|
mProgressView.postDelayed(new Runnable() {
|
||||||
|
@Override
|
||||||
|
public void run() {
|
||||||
|
mProgressView.cancelAnimation();
|
||||||
|
}
|
||||||
|
},1100);
|
||||||
|
return 1000;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onHorizontalDrag(float percentX, int offsetX, int offsetMax) {
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean isSupportHorizontalDrag() {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onStateChanged(@NonNull RefreshLayout refreshLayout, @NonNull RefreshState oldState, @NonNull RefreshState newState) {
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,147 @@
|
||||||
|
package com.remax.visualnovel.app.widget.tips
|
||||||
|
|
||||||
|
import android.content.Context
|
||||||
|
import android.graphics.drawable.ColorDrawable
|
||||||
|
import android.view.LayoutInflater
|
||||||
|
import android.view.View
|
||||||
|
import android.widget.PopupWindow
|
||||||
|
import com.remax.visualnovel.R
|
||||||
|
import com.remax.visualnovel.databinding.PopwindowFeedsBackTipsBinding
|
||||||
|
import com.remax.visualnovel.entity.request.AIFeedback
|
||||||
|
import com.remax.visualnovel.extension.setOnClick
|
||||||
|
|
||||||
|
class TipsFeedbackWindow {
|
||||||
|
|
||||||
|
private var popupWindow: PopupWindow? = null
|
||||||
|
|
||||||
|
fun build(
|
||||||
|
context: Context,
|
||||||
|
optType: Int,
|
||||||
|
clickCallback: (Int, optType: Int) -> Unit
|
||||||
|
): TipsFeedbackWindow {
|
||||||
|
val view = LayoutInflater.from(context).inflate(R.layout.popwindow_feeds_back_tips, null)
|
||||||
|
popupWindow = PopupWindow(context).apply {
|
||||||
|
isFocusable = true
|
||||||
|
contentView = view
|
||||||
|
isOutsideTouchable = true
|
||||||
|
setBackgroundDrawable(ColorDrawable())
|
||||||
|
}
|
||||||
|
var currOptType = optType
|
||||||
|
|
||||||
|
PopwindowFeedsBackTipsBinding.bind(view).run {
|
||||||
|
fun changeOptType() {
|
||||||
|
val likeColorToken = context.getString(R.string.color_primary_variant_normal)
|
||||||
|
val normalColorToken = context.getString(R.string.color_txt_primary_normal)
|
||||||
|
val iconSize = 20
|
||||||
|
val iconPadding = 16
|
||||||
|
|
||||||
|
fun updateLikeDislikeIcons(
|
||||||
|
likeIconRes: String,
|
||||||
|
likeColor: String,
|
||||||
|
dislikeIconRes: String,
|
||||||
|
dislikeColor: String,
|
||||||
|
) {
|
||||||
|
like.setIconFontDrawable(
|
||||||
|
likeIconRes,
|
||||||
|
iconColorToken = likeColor,
|
||||||
|
iconSize = iconSize,
|
||||||
|
iconPadding = iconPadding
|
||||||
|
)
|
||||||
|
dislike.setIconFontDrawable(
|
||||||
|
dislikeIconRes,
|
||||||
|
iconColorToken = dislikeColor,
|
||||||
|
iconSize = iconSize,
|
||||||
|
iconPadding = iconPadding
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
when (currOptType) {
|
||||||
|
AIFeedback.LIKE -> updateLikeDislikeIcons(
|
||||||
|
context.getString(R.string.icon_post_recommend_fill),
|
||||||
|
likeColorToken,
|
||||||
|
context.getString(R.string.icon_post_notrecommend),
|
||||||
|
normalColorToken
|
||||||
|
)
|
||||||
|
|
||||||
|
AIFeedback.DISLIKE -> updateLikeDislikeIcons(
|
||||||
|
context.getString(R.string.icon_post_recommend),
|
||||||
|
normalColorToken,
|
||||||
|
context.getString(R.string.icon_post_notrecommend_fill),
|
||||||
|
likeColorToken
|
||||||
|
)
|
||||||
|
|
||||||
|
else -> updateLikeDislikeIcons(
|
||||||
|
context.getString(R.string.icon_post_recommend),
|
||||||
|
normalColorToken,
|
||||||
|
context.getString(R.string.icon_post_notrecommend),
|
||||||
|
normalColorToken
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
when (currOptType) {
|
||||||
|
AIFeedback.LIKE -> {
|
||||||
|
like.setIconFontDrawable(
|
||||||
|
context.getString(R.string.icon_post_recommend_fill),
|
||||||
|
iconColorToken = likeColorToken,
|
||||||
|
iconSize = iconSize,
|
||||||
|
iconPadding = iconPadding
|
||||||
|
)
|
||||||
|
dislike.setIconFontDrawable(
|
||||||
|
context.getString(R.string.icon_post_notrecommend),
|
||||||
|
iconColorToken = normalColorToken,
|
||||||
|
iconSize = iconSize,
|
||||||
|
iconPadding = iconPadding
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
changeOptType()
|
||||||
|
setOnClick(copy, like, dislike) {
|
||||||
|
when (this) {
|
||||||
|
copy -> {
|
||||||
|
clickCallback.invoke(0, currOptType)
|
||||||
|
popupWindow?.dismiss()
|
||||||
|
}
|
||||||
|
|
||||||
|
like -> {
|
||||||
|
currOptType = if (currOptType == AIFeedback.LIKE) AIFeedback.NONE else AIFeedback.LIKE
|
||||||
|
changeOptType()
|
||||||
|
clickCallback.invoke(1, currOptType)
|
||||||
|
}
|
||||||
|
|
||||||
|
dislike -> {
|
||||||
|
currOptType = if (currOptType == AIFeedback.DISLIKE) AIFeedback.NONE else AIFeedback.DISLIKE
|
||||||
|
changeOptType()
|
||||||
|
clickCallback.invoke(2, currOptType)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return this
|
||||||
|
}
|
||||||
|
|
||||||
|
fun showAsDropDown(view: View?) {
|
||||||
|
if (popupWindow!!.isShowing) {
|
||||||
|
popupWindow!!.dismiss()
|
||||||
|
} else {
|
||||||
|
popupWindow!!.showAsDropDown(view)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fun showAsDropDown(view: View?, xoff: Int, yoff: Int) {
|
||||||
|
if (popupWindow!!.isShowing) {
|
||||||
|
popupWindow!!.dismiss()
|
||||||
|
} else {
|
||||||
|
popupWindow!!.showAsDropDown(view, xoff, yoff)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fun showAtLocation(parent: View?, gravity: Int, x: Int, y: Int) {
|
||||||
|
if (popupWindow!!.isShowing) {
|
||||||
|
popupWindow!!.dismiss()
|
||||||
|
} else {
|
||||||
|
popupWindow!!.showAtLocation(parent, gravity, x, y)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,92 @@
|
||||||
|
package com.remax.visualnovel.app.widget.tips
|
||||||
|
|
||||||
|
import android.content.Context
|
||||||
|
import android.graphics.drawable.ColorDrawable
|
||||||
|
import android.view.Gravity
|
||||||
|
import android.view.LayoutInflater
|
||||||
|
import android.view.View
|
||||||
|
import android.view.ViewGroup
|
||||||
|
import android.widget.LinearLayout
|
||||||
|
import android.widget.PopupWindow
|
||||||
|
import androidx.annotation.StringRes
|
||||||
|
import com.remax.visualnovel.R
|
||||||
|
import com.remax.visualnovel.extension.setOnClick
|
||||||
|
import com.remax.visualnovel.extension.setSize
|
||||||
|
import com.remax.visualnovel.utils.spannablex.utils.dp
|
||||||
|
import com.remax.visualnovel.widget.ui.IconFontTextView
|
||||||
|
import com.remax.visualnovel.widget.uitoken.changeTextStyle
|
||||||
|
|
||||||
|
class TipsMoreWindow {
|
||||||
|
data class TipsMoreUIData(
|
||||||
|
@StringRes val titleRes: Int,
|
||||||
|
@StringRes val iconRes: Int,
|
||||||
|
)
|
||||||
|
|
||||||
|
private var popupWindow: PopupWindow? = null
|
||||||
|
|
||||||
|
fun build(
|
||||||
|
context: Context,
|
||||||
|
tipsData: List<TipsMoreUIData>?,
|
||||||
|
clickCallback: (TipsMoreUIData) -> Unit = {}
|
||||||
|
): TipsMoreWindow {
|
||||||
|
val view = LayoutInflater.from(context).inflate(R.layout.popwindow_btn_tips, null)
|
||||||
|
popupWindow = PopupWindow(context).apply {
|
||||||
|
isFocusable = true
|
||||||
|
contentView = view
|
||||||
|
isOutsideTouchable = true
|
||||||
|
setBackgroundDrawable(ColorDrawable())
|
||||||
|
}
|
||||||
|
val group = view.findViewById<LinearLayout>(R.id.group)
|
||||||
|
group.removeAllViews()
|
||||||
|
tipsData?.forEach { item ->
|
||||||
|
val iconView = IconFontTextView(context)
|
||||||
|
group.addView(iconView)
|
||||||
|
iconView.apply {
|
||||||
|
gravity = Gravity.START
|
||||||
|
setSize(ViewGroup.LayoutParams.MATCH_PARENT, ViewGroup.LayoutParams.WRAP_CONTENT)
|
||||||
|
setPadding(8.dp, 12.dp, 8.dp, 12.dp)
|
||||||
|
changeTextStyle {
|
||||||
|
textUITextToken = context.getString(R.string.txt_label_l)
|
||||||
|
textUIColorToken = context.getString(R.string.color_txt_primary_normal)
|
||||||
|
|
||||||
|
}
|
||||||
|
setIconFontDrawable(
|
||||||
|
startIconFont = context.getString(item.iconRes),
|
||||||
|
iconColorToken = context.getString(R.string.color_txt_primary_normal),
|
||||||
|
iconSize = 20,
|
||||||
|
iconPadding = 16
|
||||||
|
)
|
||||||
|
setText(item.titleRes)
|
||||||
|
setOnClick(this) {
|
||||||
|
popupWindow?.dismiss()
|
||||||
|
clickCallback(item)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return this
|
||||||
|
}
|
||||||
|
|
||||||
|
fun showAsDropDown(view: View?) {
|
||||||
|
if (popupWindow!!.isShowing) {
|
||||||
|
popupWindow!!.dismiss()
|
||||||
|
} else {
|
||||||
|
popupWindow!!.showAsDropDown(view)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fun showAsDropDown(view: View?, xoff: Int, yoff: Int) {
|
||||||
|
if (popupWindow!!.isShowing) {
|
||||||
|
popupWindow!!.dismiss()
|
||||||
|
} else {
|
||||||
|
popupWindow!!.showAsDropDown(view, xoff, yoff)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fun showAtLocation(parent: View?, gravity: Int, x: Int, y: Int) {
|
||||||
|
if (popupWindow!!.isShowing) {
|
||||||
|
popupWindow!!.dismiss()
|
||||||
|
} else {
|
||||||
|
popupWindow!!.showAtLocation(parent, gravity, x, y)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,76 @@
|
||||||
|
package com.remax.visualnovel.app.widget.tips;
|
||||||
|
|
||||||
|
import android.content.Context;
|
||||||
|
import android.graphics.drawable.ColorDrawable;
|
||||||
|
import android.view.LayoutInflater;
|
||||||
|
import android.view.View;
|
||||||
|
import android.view.ViewGroup;
|
||||||
|
import android.widget.PopupWindow;
|
||||||
|
import android.widget.TextView;
|
||||||
|
|
||||||
|
import com.remax.visualnovel.R;
|
||||||
|
import com.remax.visualnovel.extension.ViewExtKt;
|
||||||
|
|
||||||
|
|
||||||
|
public class TipsPopWindow {
|
||||||
|
|
||||||
|
private PopupWindow popupWindow;
|
||||||
|
|
||||||
|
public TipsPopWindow build(Context context, String content, int width) {
|
||||||
|
View view = LayoutInflater.from(context).inflate(R.layout.popwindow_tips, null);
|
||||||
|
popupWindow = new PopupWindow(context);
|
||||||
|
popupWindow.setFocusable(true);
|
||||||
|
popupWindow.setContentView(view);
|
||||||
|
if (width != ViewGroup.LayoutParams.MATCH_PARENT) {
|
||||||
|
ViewExtKt.setMargin(view, 0, 0, 0, 0);
|
||||||
|
TextView tv = view.findViewById(R.id.tvContent);
|
||||||
|
textWidth = tv.getPaint().measureText(content);
|
||||||
|
}
|
||||||
|
popupWindow.setWidth(width);
|
||||||
|
popupWindow.setHeight(ViewGroup.LayoutParams.WRAP_CONTENT);
|
||||||
|
popupWindow.setOutsideTouchable(true);
|
||||||
|
popupWindow.setBackgroundDrawable(new ColorDrawable());
|
||||||
|
|
||||||
|
TextView tvContent = view.findViewById(R.id.tvContent);
|
||||||
|
tvContent.setText(content);
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
private float textWidth;
|
||||||
|
|
||||||
|
public float getTextWidth() {
|
||||||
|
return textWidth;
|
||||||
|
}
|
||||||
|
|
||||||
|
public TipsPopWindow build(Context context, String content) {
|
||||||
|
return build(context, content, ViewGroup.LayoutParams.MATCH_PARENT);
|
||||||
|
}
|
||||||
|
|
||||||
|
public TipsPopWindow build(Context context, int resId) {
|
||||||
|
return build(context, context.getString(resId));
|
||||||
|
}
|
||||||
|
|
||||||
|
public void showAsDropDown(View view) {
|
||||||
|
if (popupWindow.isShowing()) {
|
||||||
|
popupWindow.dismiss();
|
||||||
|
} else {
|
||||||
|
popupWindow.showAsDropDown(view);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void showAsDropDown(View view, int xoff, int yoff) {
|
||||||
|
if (popupWindow.isShowing()) {
|
||||||
|
popupWindow.dismiss();
|
||||||
|
} else {
|
||||||
|
popupWindow.showAsDropDown(view, xoff, yoff);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void showAtLocation(View parent, int gravity, int x, int y) {
|
||||||
|
if (popupWindow.isShowing()) {
|
||||||
|
popupWindow.dismiss();
|
||||||
|
} else {
|
||||||
|
popupWindow.showAtLocation(parent, gravity, x, y);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,60 @@
|
||||||
|
package com.remax.visualnovel.app.widget.tips
|
||||||
|
|
||||||
|
import android.content.Context
|
||||||
|
import android.graphics.drawable.ColorDrawable
|
||||||
|
import android.view.LayoutInflater
|
||||||
|
import android.view.View
|
||||||
|
import android.widget.PopupWindow
|
||||||
|
import android.widget.TextView
|
||||||
|
import androidx.annotation.StringRes
|
||||||
|
import com.remax.visualnovel.R
|
||||||
|
import com.remax.visualnovel.widget.ui.SwitchView
|
||||||
|
|
||||||
|
class TipsSwitchWindow {
|
||||||
|
private var popupWindow: PopupWindow? = null
|
||||||
|
|
||||||
|
fun build(
|
||||||
|
context: Context,
|
||||||
|
@StringRes tips: Int,
|
||||||
|
isChecked: Boolean = false,
|
||||||
|
switchCallback: (SwitchView,Boolean) -> Unit
|
||||||
|
): TipsSwitchWindow {
|
||||||
|
val view = LayoutInflater.from(context).inflate(R.layout.popwindow_switch_tips, null)
|
||||||
|
popupWindow = PopupWindow(context).apply {
|
||||||
|
isFocusable = true
|
||||||
|
contentView = view
|
||||||
|
isOutsideTouchable = true
|
||||||
|
setBackgroundDrawable(ColorDrawable())
|
||||||
|
}
|
||||||
|
view.findViewById<TextView>(R.id.tvContent).setText(tips)
|
||||||
|
view.findViewById<SwitchView>(R.id.switchView).run {
|
||||||
|
this.isChecked = isChecked
|
||||||
|
setPressChanged { switchCallback(this,it) }
|
||||||
|
}
|
||||||
|
return this
|
||||||
|
}
|
||||||
|
|
||||||
|
fun showAsDropDown(view: View?) {
|
||||||
|
if (popupWindow!!.isShowing) {
|
||||||
|
popupWindow!!.dismiss()
|
||||||
|
} else {
|
||||||
|
popupWindow!!.showAsDropDown(view)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fun showAsDropDown(view: View?, xoff: Int, yoff: Int) {
|
||||||
|
if (popupWindow!!.isShowing) {
|
||||||
|
popupWindow!!.dismiss()
|
||||||
|
} else {
|
||||||
|
popupWindow!!.showAsDropDown(view, xoff, yoff)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fun showAtLocation(parent: View?, gravity: Int, x: Int, y: Int) {
|
||||||
|
if (popupWindow!!.isShowing) {
|
||||||
|
popupWindow!!.dismiss()
|
||||||
|
} else {
|
||||||
|
popupWindow!!.showAtLocation(parent, gravity, x, y)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,65 @@
|
||||||
|
package com.remax.visualnovel.configs
|
||||||
|
|
||||||
|
import android.app.Activity
|
||||||
|
import android.content.Context
|
||||||
|
import android.content.res.Configuration
|
||||||
|
import androidx.multidex.MultiDex
|
||||||
|
import androidx.multidex.MultiDexApplication
|
||||||
|
import com.remax.visualnovel.app.base.app.ApplicationProxy
|
||||||
|
import com.remax.visualnovel.app.base.app.CommonApplicationProxy
|
||||||
|
import com.remax.visualnovel.app.initializer.AppInitializersProvider
|
||||||
|
import com.remax.visualnovel.utils.StatusBarUtils
|
||||||
|
import com.tencent.mmkv.MMKV
|
||||||
|
import dagger.hilt.android.HiltAndroidApp
|
||||||
|
import timber.log.Timber
|
||||||
|
import java.lang.ref.WeakReference
|
||||||
|
import javax.inject.Inject
|
||||||
|
|
||||||
|
|
||||||
|
@HiltAndroidApp
|
||||||
|
class NovelApplication : MultiDexApplication() {
|
||||||
|
|
||||||
|
companion object {
|
||||||
|
private var currentActivity: WeakReference<Activity>? = null
|
||||||
|
|
||||||
|
fun setCurrentActivity(activity: Activity?) {
|
||||||
|
currentActivity = if (activity == null) null else WeakReference(activity)
|
||||||
|
}
|
||||||
|
|
||||||
|
fun getCurrentActivity(): Activity? {
|
||||||
|
return currentActivity?.get()
|
||||||
|
}
|
||||||
|
|
||||||
|
private lateinit var instance: NovelApplication
|
||||||
|
fun appContext(): Context = instance
|
||||||
|
}
|
||||||
|
|
||||||
|
private val proxies = listOf<ApplicationProxy>(CommonApplicationProxy)
|
||||||
|
|
||||||
|
@Inject
|
||||||
|
lateinit var appInitializersProvider: AppInitializersProvider
|
||||||
|
|
||||||
|
override fun onCreate() {
|
||||||
|
super.onCreate()
|
||||||
|
MultiDex.install(this)
|
||||||
|
instance = this
|
||||||
|
proxies.forEach { it.onCreate(this) }
|
||||||
|
appInitializersProvider.startInit()
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun onTerminate() {
|
||||||
|
MMKV.onExit()
|
||||||
|
super.onTerminate()
|
||||||
|
proxies.forEach {
|
||||||
|
it.onTerminate()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 系统资源配置发生更改回调,例如主题模式,需要重新刷新多语言
|
||||||
|
override fun onConfigurationChanged(newConfig: Configuration) {
|
||||||
|
super.onConfigurationChanged(newConfig)
|
||||||
|
StatusBarUtils.resetNavBarHeight()
|
||||||
|
Timber.d("onConfigurationChanged ${newConfig.locales[0]}")
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,13 @@
|
||||||
|
package com.remax.visualnovel.constant
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Created by HJW on 2021/12/7
|
||||||
|
*/
|
||||||
|
class AppConstant {
|
||||||
|
|
||||||
|
companion object {
|
||||||
|
const val ANDROID = "android"
|
||||||
|
const val APP_CLIENT = "ANDROID"
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,18 @@
|
||||||
|
package com.remax.visualnovel.constant
|
||||||
|
|
||||||
|
import com.remax.visualnovel.BuildConfig
|
||||||
|
|
||||||
|
|
||||||
|
class AppStatus {
|
||||||
|
|
||||||
|
companion object {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 是否是生产环境
|
||||||
|
*/
|
||||||
|
val isProduct
|
||||||
|
get() = BuildConfig.FLAVOR == "product"
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,15 @@
|
||||||
|
package com.remax.visualnovel.constant
|
||||||
|
|
||||||
|
import androidx.annotation.StringRes
|
||||||
|
import com.remax.visualnovel.R
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 性别枚举
|
||||||
|
*/
|
||||||
|
enum class Gender(val value: Int, @StringRes val txtRes: Int) {
|
||||||
|
MALE(0, R.string.male),
|
||||||
|
FEMALE(1, R.string.female),
|
||||||
|
NONCONFORMING(2, R.string.nonconforming),
|
||||||
|
OTHER(2, R.string.other),
|
||||||
|
}
|
||||||
|
|
|
@ -0,0 +1,24 @@
|
||||||
|
package com.remax.visualnovel.constant
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Created by HJW on 2023/8/25
|
||||||
|
*/
|
||||||
|
class LockTypeConstant {
|
||||||
|
companion object {
|
||||||
|
const val PUBLIC = 1
|
||||||
|
const val PRIVATE = 2
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 图片公开
|
||||||
|
*/
|
||||||
|
fun isOpen(pubType: Int?) = pubType != PRIVATE
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 图片是否解锁
|
||||||
|
*/
|
||||||
|
fun isUnLock(lockType: String?) = lockType != LOCK
|
||||||
|
|
||||||
|
const val LOCK = "LOCK"
|
||||||
|
const val UNLOCK = "UNLOCK"
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,27 @@
|
||||||
|
package com.remax.visualnovel.constant
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Created by HJW on 2025/7/21
|
||||||
|
*/
|
||||||
|
enum class StatusCode(val code:String) {
|
||||||
|
/**
|
||||||
|
* 登录过期
|
||||||
|
*/
|
||||||
|
TOKEN_EXPIRED("10050001"),
|
||||||
|
|
||||||
|
NO_ALBUM_PERMISSION("10010011"),
|
||||||
|
|
||||||
|
UNUSED_PURCHASE_TOKEN("1019"), //无效的支付凭据
|
||||||
|
|
||||||
|
AI_USER_NOT_EXIST("10010012"),
|
||||||
|
|
||||||
|
//余额不足
|
||||||
|
INSUFFICIENT_BALANCE("INSUFFICIENT_BALANCE"),
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 前段自定义code
|
||||||
|
*/
|
||||||
|
UPLOAD_FILE_VIOLATION("112233")
|
||||||
|
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,12 @@
|
||||||
|
package com.remax.visualnovel.entity.imbean
|
||||||
|
|
||||||
|
import com.remax.visualnovel.entity.imbean.voice.IMVoice
|
||||||
|
import com.netease.nimlib.sdk.v2.message.V2NIMMessage
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Created by HJW on 2025/8/19
|
||||||
|
*/
|
||||||
|
data class IMAIInMessage(
|
||||||
|
override var message: V2NIMMessage?,
|
||||||
|
val imVoice: IMVoice
|
||||||
|
) : IMMessageWrapper(type = IN_TEXT_TYPE)
|
|
@ -0,0 +1,10 @@
|
||||||
|
package com.remax.visualnovel.entity.imbean
|
||||||
|
|
||||||
|
import com.remax.visualnovel.entity.response.Character
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Created by HJW on 2025/8/19
|
||||||
|
*/
|
||||||
|
data class IMBaseInfoMessage(
|
||||||
|
val character: Character?
|
||||||
|
) : IMMessageWrapper(type = BASE_INFO)
|
|
@ -0,0 +1,12 @@
|
||||||
|
package com.remax.visualnovel.entity.imbean
|
||||||
|
|
||||||
|
import com.remax.visualnovel.entity.imbean.raw.CustomCallData
|
||||||
|
import com.netease.nimlib.sdk.v2.message.V2NIMMessage
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Created by HJW on 2025/8/19
|
||||||
|
*/
|
||||||
|
data class IMCallMessage(
|
||||||
|
override var message: V2NIMMessage?,
|
||||||
|
val call: CustomCallData?
|
||||||
|
) : IMMessageWrapper(type = OUT_CALL_TYPE)
|
|
@ -0,0 +1,12 @@
|
||||||
|
package com.remax.visualnovel.entity.imbean
|
||||||
|
|
||||||
|
import com.remax.visualnovel.entity.imbean.raw.CustomGiftData
|
||||||
|
import com.netease.nimlib.sdk.v2.message.V2NIMMessage
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Created by HJW on 2025/8/19
|
||||||
|
*/
|
||||||
|
data class IMGiftMessage(
|
||||||
|
override var message: V2NIMMessage?,
|
||||||
|
val gift: CustomGiftData?
|
||||||
|
) : IMMessageWrapper(type = OUT_GIFT_TYPE)
|
|
@ -0,0 +1,12 @@
|
||||||
|
package com.remax.visualnovel.entity.imbean
|
||||||
|
|
||||||
|
import com.remax.visualnovel.entity.imbean.raw.CustomAlbumData
|
||||||
|
import com.netease.nimlib.sdk.v2.message.V2NIMMessage
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Created by HJW on 2025/8/19
|
||||||
|
*/
|
||||||
|
data class IMInImageMessage(
|
||||||
|
override var message: V2NIMMessage?,
|
||||||
|
val albumData: CustomAlbumData?
|
||||||
|
) : IMMessageWrapper(type = IN_IMAGE_TYPE)
|
|
@ -0,0 +1,12 @@
|
||||||
|
package com.remax.visualnovel.entity.imbean
|
||||||
|
|
||||||
|
import com.remax.visualnovel.entity.imbean.raw.CustomLevelChangeData
|
||||||
|
import com.netease.nimlib.sdk.v2.message.V2NIMMessage
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Created by HJW on 2025/8/19
|
||||||
|
*/
|
||||||
|
data class IMLevelMessage(
|
||||||
|
override var message: V2NIMMessage?,
|
||||||
|
val level: CustomLevelChangeData?
|
||||||
|
) : IMMessageWrapper(type = HEART_BEAT_CHANGED_TYPE)
|
|
@ -0,0 +1,44 @@
|
||||||
|
package com.remax.visualnovel.entity.imbean
|
||||||
|
|
||||||
|
import com.remax.visualnovel.manager.nim.FetchResult
|
||||||
|
import com.remax.visualnovel.manager.nim.LoadStatus
|
||||||
|
import com.netease.nimlib.sdk.v2.message.V2NIMMessage
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Created by HJW on 2020/9/28
|
||||||
|
*/
|
||||||
|
open class IMMessageWrapper(
|
||||||
|
open var message: V2NIMMessage? = null,
|
||||||
|
var type: Int = OUT_TEXT_TYPE,
|
||||||
|
) {
|
||||||
|
/**
|
||||||
|
* 表示该消息后是否要加一条AI输入中的消息
|
||||||
|
*/
|
||||||
|
var aiIsSending: Boolean = true
|
||||||
|
|
||||||
|
var fetchType: FetchResult.FetchType = FetchResult.FetchType.Init
|
||||||
|
var loadStatus: LoadStatus = LoadStatus.Success
|
||||||
|
|
||||||
|
companion object {
|
||||||
|
const val BASE_INFO = 0
|
||||||
|
|
||||||
|
// ai回复中
|
||||||
|
const val INPUT_ING = 1
|
||||||
|
|
||||||
|
const val OUT_TEXT_TYPE = 2
|
||||||
|
const val IN_TEXT_TYPE = 3
|
||||||
|
|
||||||
|
const val OUT_IMAGE_TYPE = 4
|
||||||
|
const val IN_IMAGE_TYPE = 5
|
||||||
|
|
||||||
|
const val OUT_GIFT_TYPE = 6
|
||||||
|
|
||||||
|
const val OUT_CALL_TYPE = 7
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 心动等级升级/降级
|
||||||
|
*/
|
||||||
|
const val HEART_BEAT_CHANGED_TYPE = 8
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,12 @@
|
||||||
|
package com.remax.visualnovel.entity.imbean
|
||||||
|
|
||||||
|
import com.remax.visualnovel.entity.imbean.raw.CustomRawData
|
||||||
|
import com.netease.nimlib.sdk.v2.message.V2NIMMessage
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Created by HJW on 2025/8/19
|
||||||
|
*/
|
||||||
|
data class IMOutImageMessage(
|
||||||
|
override var message: V2NIMMessage?,
|
||||||
|
val customRawData: CustomRawData?
|
||||||
|
) : IMMessageWrapper(type = OUT_IMAGE_TYPE)
|
|
@ -0,0 +1,19 @@
|
||||||
|
package com.remax.visualnovel.entity.imbean
|
||||||
|
|
||||||
|
import com.netease.nimlib.sdk.v2.conversation.model.V2NIMConversation
|
||||||
|
import com.netease.nimlib.sdk.v2.utils.V2NIMConversationIdUtil
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Created by HJW on 2020/10/9
|
||||||
|
*/
|
||||||
|
data class RecentContactWrapper(
|
||||||
|
var recentContact: V2NIMConversation
|
||||||
|
) {
|
||||||
|
val aiId: String
|
||||||
|
get() {
|
||||||
|
val targetId = V2NIMConversationIdUtil.conversationTargetId(recentContact.conversationId)
|
||||||
|
|
||||||
|
return targetId.substring(0, targetId.indexOf("@"))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
Reference in New Issue