diff --git a/.github/workflows/firebase_distribution_builder.yml b/.github/workflows/firebase_distribution_builder.yml new file mode 100644 index 00000000..1422730d --- /dev/null +++ b/.github/workflows/firebase_distribution_builder.yml @@ -0,0 +1,81 @@ +name: Genti Firebase App Distribution Builder + +on: + push: + branches: [ production ] + +defaults: + run: + shell: bash + working-directory: . + +jobs: + build: + name: Firebase App Distribution Builder + runs-on: ubuntu-latest + steps: + - name: Checkout + uses: actions/checkout@v3 + + - name: Gradle cache + uses: actions/cache@v3 + with: + path: | + ~/.gradle/caches + ~/.gradle/wrapper + key: ${{ runner.os }}-gradle-${{ hashFiles('**/*.gradle*') }} + restore-keys: | + ${{ runner.os }}-gradle- + - name: set up JDK 17 + uses: actions/setup-java@v3 + with: + distribution: 'temurin' + java-version: 17 + + - name: Change gradlew permissions + run: chmod +x ./gradlew + + - name: Create Local Properties + run: touch local.properties + + - name: Access Local Properties + env: + base_url: ${{ secrets.BASE_URL }} + test_base_url: ${{ secrets.TEST_BASE_URL }} + native_app_key: ${{ secrets.NATIVE_APP_KEY }} + amplitude_api_key: ${{ secrets.AMPLITUDE_API_KEY }} + amplitude_test_key: ${{ secrets.AMPLITUDE_TEST_KEY }} + run: | + echo "base.url=\"$base_url\"" >> local.properties + echo "test.base.url=\"$test_base_url\"" >> local.properties + echo "native.app.key=\"$native_app_key\"" >> local.properties + echo "nativeAppKey=\"$native_app_key\"" >> local.properties + echo "amplitude.api.key=\"$amplitude_api_key\"" >> local.properties + echo "amplitude.test.key=\"$amplitude_test_key\"" >> local.properties + + - name: Access Keystore Properties + env: + keystore_file: ${{ secrets.KEYSTORE_FILE }} + store_password: ${{ secrets.STORE_PASSWORD }} + key_password: ${{ secrets.KEY_PASSWORD }} + key_alias: ${{ secrets.KEY_ALIAS }} + run: | + echo "storeFile=keystore.jks" > keystore.properties + echo "storePassword=\"$store_password\"" >> keystore.properties + echo "keyAlias=\"$key_alias\"" >> keystore.properties + echo "keyPassword=\"$key_password\"" >> keystore.properties + echo "$keystore_file" | base64 -d > keystore.jks + + - name: Access Firebase Service + run: echo '${{ secrets.GOOGLE_SERVICES_JSON }}' > ./app/google-services.json + + - name: Build release APK + run: ./gradlew assembleRelease --stacktrace + + - name: Upload to Firebase App Distribution + uses: wzieba/Firebase-Distribution-Github-Action@v1 + with: + appId: ${{ secrets.FIREBASE_APP_ID }} + serviceCredentialsFileContent: ${{ secrets.FIREBASE_APP_DISTRIBUTION_KEY }} + groups: tester + file: app/build/outputs/apk/release/app-release.apk \ No newline at end of file diff --git a/.github/workflows/pr_checker.yml b/.github/workflows/pr_checker.yml index eda66bdd..96b66cb1 100644 --- a/.github/workflows/pr_checker.yml +++ b/.github/workflows/pr_checker.yml @@ -41,5 +41,33 @@ jobs: - name: Access Local Properties env: base_url: ${{ secrets.BASE_URL }} + test_base_url: ${{ secrets.TEST_BASE_URL }} + native_app_key: ${{ secrets.NATIVE_APP_KEY }} + amplitude_api_key: ${{ secrets.AMPLITUDE_API_KEY }} + amplitude_test_key: ${{ secrets.AMPLITUDE_TEST_KEY }} run: | - echo base.url=\"$base_url\" >> local.properties \ No newline at end of file + echo "base.url=\"$base_url\"" >> local.properties + echo "test.base.url=\"$test_base_url\"" >> local.properties + echo "native.app.key=\"$native_app_key\"" >> local.properties + echo "nativeAppKey=\"$native_app_key\"" >> local.properties + echo "amplitude.api.key=\"$amplitude_api_key\"" >> local.properties + echo "amplitude.test.key=\"$amplitude_test_key\"" >> local.properties + + - name: Access Keystore Properties + env: + keystore_file: ${{ secrets.KEYSTORE_FILE }} + store_password: ${{ secrets.STORE_PASSWORD }} + key_password: ${{ secrets.KEY_PASSWORD }} + key_alias: ${{ secrets.KEY_ALIAS }} + run: | + echo "storeFile=keystore.jks" > keystore.properties + echo "storePassword=\"$store_password\"" >> keystore.properties + echo "keyAlias=\"$key_alias\"" >> keystore.properties + echo "keyPassword=\"$key_password\"" >> keystore.properties + echo "$keystore_file" | base64 -d > keystore.jks + + - name: Access Firebase Service + run: echo '${{ secrets.GOOGLE_SERVICES_JSON }}' > ./app/google-services.json + + - name: Build debug APK + run: ./gradlew assembleDebug --stacktrace \ No newline at end of file diff --git a/.gitignore b/.gitignore index ffe9ebf8..4214c82c 100644 --- a/.gitignore +++ b/.gitignore @@ -115,6 +115,9 @@ captures/ # Google Services (e.g. APIs or Firebase) google-services.json +# Local configuration file (sdk path, etc) +keystore.properties + # Android Patch gen-external-apklibs diff --git a/README.md b/README.md index dffe1678..49b1d6da 100644 --- a/README.md +++ b/README.md @@ -7,19 +7,36 @@
-## TEAM _ 안드로이드 단독 개발 -![2024-08-01_00-28-32](https://github.com/user-attachments/assets/5179534b-09d6-428d-aa67-92afd1163cc7) +## TEAM +![2024-08-01_00-28-32](https://github.com/user-attachments/assets/64bb2fa3-cd67-430a-ae53-ff20c76d53b7) + +### Android Contributor + +[![contributors](https://contrib.rocks/image?repo=Genti2024/Genti-Android)](https://github.com/Genti2024/Genti-Android/contributors) + +
+ +## PRODUCT +![Group 8245](https://github.com/user-attachments/assets/6616126f-8be8-4011-afc8-d312145cd5e9)
## SOLUTIONS -![2024-08-01_00-34-24](https://github.com/user-attachments/assets/649fb66d-6788-4d6c-a6de-f4b673f9623b) -![2024-08-01_00-34-43](https://github.com/user-attachments/assets/897c4340-d6d9-4f1a-b9bd-ff4c69675864) +![Group 8250](https://github.com/user-attachments/assets/c96de744-a45f-4512-9896-cd4c6b278ecb) +![Group 8251](https://github.com/user-attachments/assets/45c89b6e-6c74-498d-b498-aaa1bad0484f) +![Group 8252](https://github.com/user-attachments/assets/27210a73-d53c-479c-be24-bc9fe4c5a060) + +
+ +## PROGRESS +![Group 8249](https://github.com/user-attachments/assets/30ab4785-68f0-4490-8bbe-a2b0a9a269e6) +![Group 8248](https://github.com/user-attachments/assets/a78918f8-14e5-4853-b0f2-806e880d086a) +![Group 8247](https://github.com/user-attachments/assets/cf32392b-c7f1-4966-9439-ffe150bdc4d0)
## ACHIEVEMENT -- 2024 정주영 창업경진대회 (아산나눔재단) 사업실행팀 선발 +- 2024 정주영 창업경진대회 (아산나눔재단) 사업실행팀 선발 및 본상 수상 - 2024 고려대 KU 창업동아리 아이디어 트랙 선정 - 2024 동국대 아이디어 사업화 지원사업 선발 - 2024 KUCT 딥테크 스타트업 프론티어 선발전 도전상 수상 diff --git a/app/build.gradle.kts b/app/build.gradle.kts index 7bfec101..091d41e4 100644 --- a/app/build.gradle.kts +++ b/app/build.gradle.kts @@ -1,28 +1,14 @@ import com.android.build.gradle.internal.cxx.configure.gradleLocalProperties +import java.util.Properties plugins { - id("com.android.application") - kotlin("android") - kotlin("kapt") - id("kotlin-parcelize") - id("dagger.hilt.android.plugin") + id("kr.genti.androidApplication") id("com.google.gms.google-services") id("com.google.firebase.crashlytics") } android { - namespace = Constants.packageName - compileSdk = Constants.compileSdk - defaultConfig { - applicationId = Constants.packageName - minSdk = Constants.minSdk - targetSdk = Constants.targetSdk - versionCode = Constants.versionCode - versionName = Constants.versionName - - testInstrumentationRunner = "androidx.test.runner.AndroidJUnitRunner" - buildConfigField( "String", "NATIVE_APP_KEY", @@ -31,6 +17,19 @@ android { manifestPlaceholders["NATIVE_APP_KEY"] = gradleLocalProperties(rootDir).getProperty("nativeAppKey") + + val keystorePropertiesFile = rootProject.file("keystore.properties") + val keystoreProperties = Properties() + keystoreProperties.load(keystorePropertiesFile.inputStream()) + + signingConfigs { + create("release") { + storeFile = file(keystoreProperties["storeFile"] as String) + storePassword = keystoreProperties["storePassword"] as String + keyAlias = keystoreProperties["keyAlias"] as String + keyPassword = keystoreProperties["keyPassword"] as String + } + } } buildTypes { @@ -57,72 +56,17 @@ android { "AMPLITUDE_KEY", gradleLocalProperties(rootDir).getProperty("amplitude.api.key"), ) - - isMinifyEnabled = false - proguardFiles( - getDefaultProguardFile("proguard-android-optimize.txt"), - "proguard-rules.pro", - ) } } - - compileOptions { - sourceCompatibility = Versions.javaVersion - targetCompatibility = Versions.javaVersion - } - - kotlinOptions { - jvmTarget = Versions.jvmVersion - } - - buildFeatures { - buildConfig = true - dataBinding = true - viewBinding = true - } } dependencies { - implementation(project(":core")) - implementation(project(":data")) - implementation(project(":domain")) - implementation(project(":presentation")) - - KotlinDependencies.run { - implementation(kotlin) - implementation(coroutines) - implementation(jsonSerialization) - } - - AndroidXDependencies.run { - implementation(coreKtx) - implementation(appCompat) - implementation(hilt) - } - - KaptDependencies.run { - kapt(hiltCompiler) - } - - TestDependencies.run { - testImplementation(jUnit) - androidTestImplementation(androidTest) - androidTestImplementation(espresso) - } - - RetrofitDependencies.run { - implementation(platform(okHttpBom)) - implementation(okHttp) - implementation(okHttpLoggingInterceptor) - implementation(retrofit) - implementation(retrofitJsonConverter) - } - - ThirdPartyDependencies.run { - implementation(timber) - } - - KakaoDependencies.run { - implementation(user) - } + implementation(projects.core) + implementation(projects.data) + implementation(projects.domain) + implementation(projects.presentation) + + implementation(platform(libs.okhttp.bom)) + implementation(libs.bundles.networking) + implementation(libs.kakao) } diff --git a/app/src/main/AndroidManifest.xml b/app/src/main/AndroidManifest.xml index 5cdd7497..d92d8e9c 100644 --- a/app/src/main/AndroidManifest.xml +++ b/app/src/main/AndroidManifest.xml @@ -54,17 +54,18 @@ android:screenOrientation="portrait" /> + android:screenOrientation="portrait" + android:windowSoftInputMode="adjustResize" /> diff --git a/app/src/main/java/kr/genti/android/di/AuthInterceptor.kt b/app/src/main/java/kr/genti/android/di/AuthInterceptor.kt index ee7ff2dd..fa784976 100644 --- a/app/src/main/java/kr/genti/android/di/AuthInterceptor.kt +++ b/app/src/main/java/kr/genti/android/di/AuthInterceptor.kt @@ -18,69 +18,70 @@ import timber.log.Timber import javax.inject.Inject class AuthInterceptor - @Inject - constructor( - private val authRepository: AuthRepository, - private val userRepository: UserRepository, - @ApplicationContext private val context: Context, - ) : Interceptor { - override fun intercept(chain: Interceptor.Chain): Response { - val originalRequest = chain.request() +@Inject +constructor( + private val authRepository: AuthRepository, + private val userRepository: UserRepository, + @ApplicationContext private val context: Context, +) : Interceptor { + override fun intercept(chain: Interceptor.Chain): Response { + val originalRequest = chain.request() - Timber.tag("okhttp").d("ACCESS TOKEN : ${userRepository.getAccessToken()}") + Timber.tag("okhttp").d("ACCESS TOKEN : ${userRepository.getAccessToken()}") - val authRequest = - if (userRepository.getAccessToken().isNotBlank()) { - originalRequest.newBuilder().newAuthBuilder().build() - } else { - originalRequest - } + val authRequest = + if (userRepository.getAccessToken().isNotBlank()) { + originalRequest.newBuilder().newAuthBuilder().build() + } else { + originalRequest + } - val response = chain.proceed(authRequest) + val response = chain.proceed(authRequest) - if (response.code == CODE_TOKEN_EXPIRED) { - try { - runBlocking { - authRepository.postReissueTokens( - ReissueRequestModel( - userRepository.getAccessToken(), - userRepository.getRefreshToken(), - ), - ) - }.onSuccess { data -> - userRepository.setTokens( - data.accessToken, - data.refreshToken, - ) - response.close() + if (response.code == CODE_TOKEN_EXPIRED) { + try { + runBlocking { + authRepository.postReissueTokens( + ReissueRequestModel( + userRepository.getAccessToken(), + userRepository.getRefreshToken(), + ), + ) + }.onSuccess { data -> + userRepository.setTokens( + data.accessToken, + data.refreshToken, + ) + response.close() - val newRequest = - authRequest.newBuilder().removeHeader(AUTHORIZATION).newAuthBuilder() - .build() - return chain.proceed(newRequest) - } - } catch (t: Throwable) { - Timber.d(t.message) + val newRequest = + authRequest.newBuilder().removeHeader(AUTHORIZATION).newAuthBuilder() + .build() + return chain.proceed(newRequest) } + } catch (t: Throwable) { + Timber.d(t.message) + } - userRepository.clearInfo() + userRepository.clearInfo() - Handler(Looper.getMainLooper()).post { - context.toast(TOKEN_EXPIRED_ERROR) - Intent(context, LoginActivity::class.java).apply { - addFlags(Intent.FLAG_ACTIVITY_NEW_TASK or Intent.FLAG_ACTIVITY_CLEAR_TASK) - context.startActivity(this) - } + Handler(Looper.getMainLooper()).post { + context.toast(TOKEN_EXPIRED_ERROR) + Intent(context, LoginActivity::class.java).apply { + addFlags(Intent.FLAG_ACTIVITY_NEW_TASK or Intent.FLAG_ACTIVITY_CLEAR_TASK) + context.startActivity(this) } } - return response } + return response + } - private fun Request.Builder.newAuthBuilder() = this.addHeader(AUTHORIZATION, userRepository.getAccessToken()) + private fun Request.Builder.newAuthBuilder() = + this.addHeader(AUTHORIZATION, userRepository.getAccessToken()) - companion object { - private const val CODE_TOKEN_EXPIRED = 401 - private const val TOKEN_EXPIRED_ERROR = "토큰이 만료되었어요\n다시 로그인 해주세요" - private const val AUTHORIZATION = "Authorization" - } + companion object { + private const val CODE_TOKEN_EXPIRED = 401 + private const val TOKEN_EXPIRED_ERROR = "토큰이 만료되었어요\n다시 로그인 해주세요" + private const val AUTHORIZATION = "Authorization" } +} diff --git a/build-logic/convention/build.gradle.kts b/build-logic/convention/build.gradle.kts new file mode 100644 index 00000000..f38db1c9 --- /dev/null +++ b/build-logic/convention/build.gradle.kts @@ -0,0 +1,47 @@ +plugins { + `kotlin-dsl` +} + +java { + sourceCompatibility = JavaVersion.VERSION_17 + targetCompatibility = JavaVersion.VERSION_17 +} + +dependencies { + implementation(libs.android.gradlePlugin) + implementation(libs.kotlin.gradlePlugin) +} + +gradlePlugin { + plugins { + register("AndroidApplicationPlugin") { + id = "kr.genti.androidApplication" + implementationClass = "kr.genti.convention.plugin.AndroidApplicationPlugin" + } + register("AndroidLibraryPlugin") { + id = "kr.genti.androidLibrary" + implementationClass = "kr.genti.convention.plugin.AndroidLibraryPlugin" + } + register("JavaLibraryPlugin") { + id = "kr.genti.javaLibrary" + implementationClass = "kr.genti.convention.plugin.JavaLibraryPlugin" + } + + register("KotlinPlugin") { + id = "kr.genti.kotlin" + implementationClass = "kr.genti.convention.plugin.KotlinPlugin" + } + register("HiltPlugin") { + id = "kr.genti.hilt" + implementationClass = "kr.genti.convention.plugin.HiltPlugin" + } + register("TestPlugin") { + id = "kr.genti.test" + implementationClass = "kr.genti.convention.plugin.TestPlugin" + } + register("versionPlugin") { + id = "kr.genti.version" + implementationClass = "kr.genti.convention.plugin.VersionPlugin" + } + } +} \ No newline at end of file diff --git a/build-logic/convention/src/main/java/kr/genti/convention/Constants.kt b/build-logic/convention/src/main/java/kr/genti/convention/Constants.kt new file mode 100644 index 00000000..3dd7e4ce --- /dev/null +++ b/build-logic/convention/src/main/java/kr/genti/convention/Constants.kt @@ -0,0 +1,17 @@ +package kr.genti.convention + +import org.gradle.api.JavaVersion + +object Constants { + const val packageName = "kr.genti.android" + + const val compileSdk = 34 + const val minSdk = 28 + const val targetSdk = 34 + + const val versionCode = 19 + const val versionName = "2.1.0" + + const val jvmVersion = "17" + val JAVA_VERSION = JavaVersion.VERSION_17 +} diff --git a/build-logic/convention/src/main/java/kr/genti/convention/config/CommonPluginConfig.kt b/build-logic/convention/src/main/java/kr/genti/convention/config/CommonPluginConfig.kt new file mode 100644 index 00000000..448685b5 --- /dev/null +++ b/build-logic/convention/src/main/java/kr/genti/convention/config/CommonPluginConfig.kt @@ -0,0 +1,28 @@ +package kr.genti.convention.config + +import kr.genti.convention.extension.getLibrary +import kr.genti.convention.extension.implementation +import kr.genti.convention.plugin.HiltPlugin +import kr.genti.convention.plugin.KotlinPlugin +import kr.genti.convention.plugin.TestPlugin +import org.gradle.api.Project +import org.gradle.api.artifacts.VersionCatalogsExtension +import org.gradle.kotlin.dsl.apply +import org.gradle.kotlin.dsl.dependencies +import org.gradle.kotlin.dsl.getByType + +fun Project.configureAndroidCommonPlugin() { + apply() + apply() + apply() + with(plugins) { + apply("kotlin-parcelize") + apply("org.jetbrains.kotlin.plugin.serialization") + } + + val libs = extensions.getByType().named("libs") + dependencies { + implementation(libs.getLibrary("material-design")) + implementation(libs.getLibrary("timber")) + } +} \ No newline at end of file diff --git a/build-logic/convention/src/main/java/kr/genti/convention/extension/CommonExt.kt b/build-logic/convention/src/main/java/kr/genti/convention/extension/CommonExt.kt new file mode 100644 index 00000000..89880084 --- /dev/null +++ b/build-logic/convention/src/main/java/kr/genti/convention/extension/CommonExt.kt @@ -0,0 +1,9 @@ +package kr.genti.convention.extension + +import com.android.build.api.dsl.CommonExtension +import org.gradle.api.plugins.ExtensionAware +import org.jetbrains.kotlin.gradle.dsl.KotlinJvmOptions + +fun CommonExtension<*, *, *, *, *>.kotlinOptions(block: KotlinJvmOptions.() -> Unit) { + (this as ExtensionAware).extensions.configure("kotlinOptions", block) +} \ No newline at end of file diff --git a/build-logic/convention/src/main/java/kr/genti/convention/extension/DependencyHandlerScopeExt.kt b/build-logic/convention/src/main/java/kr/genti/convention/extension/DependencyHandlerScopeExt.kt new file mode 100644 index 00000000..ec5fb735 --- /dev/null +++ b/build-logic/convention/src/main/java/kr/genti/convention/extension/DependencyHandlerScopeExt.kt @@ -0,0 +1,47 @@ +package kr.genti.convention.extension + +import org.gradle.api.Project +import org.gradle.api.file.ConfigurableFileCollection +import org.gradle.api.file.ConfigurableFileTree +import org.gradle.api.provider.Provider +import org.gradle.kotlin.dsl.DependencyHandlerScope + +fun DependencyHandlerScope.implementation(project: Project) { + "implementation"(project) +} + +fun DependencyHandlerScope.implementation(provider: Provider<*>) { + "implementation"(provider) +} + +fun DependencyHandlerScope.implementation(fileTree: ConfigurableFileTree) { + "implementation"(fileTree) +} + +fun DependencyHandlerScope.implementation(fileCollection: ConfigurableFileCollection) { + "implementation"(fileCollection) +} + +fun DependencyHandlerScope.debugImplementation(provider: Provider<*>) { + "debugImplementation"(provider) +} + +fun DependencyHandlerScope.releaseImplementation(provider: Provider<*>) { + "releaseImplementation"(provider) +} + +fun DependencyHandlerScope.kapt(provider: Provider<*>) { + "kapt"(provider) +} + +fun DependencyHandlerScope.coreLibraryDesugaring(provider: Provider<*>) { + "coreLibraryDesugaring"(provider) +} + +fun DependencyHandlerScope.androidTestImplementation(provider: Provider<*>) { + "androidTestImplementation"(provider) +} + +fun DependencyHandlerScope.testImplementation(provider: Provider<*>) { + "testImplementation"(provider) +} \ No newline at end of file diff --git a/build-logic/convention/src/main/java/kr/genti/convention/extension/ProjectExt.kt b/build-logic/convention/src/main/java/kr/genti/convention/extension/ProjectExt.kt new file mode 100644 index 00000000..47746cd5 --- /dev/null +++ b/build-logic/convention/src/main/java/kr/genti/convention/extension/ProjectExt.kt @@ -0,0 +1,9 @@ +package kr.genti.convention.extension + +import org.gradle.api.Project +import org.gradle.api.artifacts.VersionCatalog +import org.gradle.api.artifacts.VersionCatalogsExtension +import org.gradle.kotlin.dsl.getByType + +val Project.libs: VersionCatalog + get() = extensions.getByType().named("libs") \ No newline at end of file diff --git a/build-logic/convention/src/main/java/kr/genti/convention/extension/VersionCatalogExt.kt b/build-logic/convention/src/main/java/kr/genti/convention/extension/VersionCatalogExt.kt new file mode 100644 index 00000000..c8f9686b --- /dev/null +++ b/build-logic/convention/src/main/java/kr/genti/convention/extension/VersionCatalogExt.kt @@ -0,0 +1,16 @@ +package kr.genti.convention.extension + +import org.gradle.api.artifacts.ExternalModuleDependencyBundle +import org.gradle.api.artifacts.MinimalExternalModuleDependency +import org.gradle.api.artifacts.VersionCatalog +import org.gradle.api.provider.Provider + +fun VersionCatalog.getBundle(bundleName: String): Provider = + findBundle(bundleName).orElseThrow { + NoSuchElementException("Bundle with name $bundleName not found in the catalog") + } + +fun VersionCatalog.getLibrary(libraryName: String): Provider = + findLibrary(libraryName).orElseThrow { + NoSuchElementException("Library with name $libraryName not found in the catalog") + } \ No newline at end of file diff --git a/build-logic/convention/src/main/java/kr/genti/convention/plugin/AndroidApplicationPlugin.kt b/build-logic/convention/src/main/java/kr/genti/convention/plugin/AndroidApplicationPlugin.kt new file mode 100644 index 00000000..b70907ce --- /dev/null +++ b/build-logic/convention/src/main/java/kr/genti/convention/plugin/AndroidApplicationPlugin.kt @@ -0,0 +1,60 @@ +package kr.genti.convention.plugin + +import com.android.build.api.dsl.ApplicationExtension +import kr.genti.convention.Constants +import kr.genti.convention.config.configureAndroidCommonPlugin +import kr.genti.convention.extension.kotlinOptions +import org.gradle.api.Plugin +import org.gradle.api.Project +import org.gradle.kotlin.dsl.configure + +class AndroidApplicationPlugin : Plugin { + override fun apply(target: Project) = + with(target) { + with(pluginManager) { + apply("com.android.application") + } + + extensions.configure { + configureAndroidCommonPlugin() + + namespace = Constants.packageName + compileSdk = Constants.compileSdk + + defaultConfig { + applicationId = Constants.packageName + targetSdk = Constants.targetSdk + minSdk = Constants.minSdk + versionCode = Constants.versionCode + versionName = Constants.versionName + + testInstrumentationRunner = "androidx.test.runner.AndroidJUnitRunner" + } + + compileOptions { + sourceCompatibility = Constants.JAVA_VERSION + targetCompatibility = Constants.JAVA_VERSION + } + + kotlinOptions { + jvmTarget = Constants.jvmVersion + } + + buildFeatures { + buildConfig = true + viewBinding = true + dataBinding = true + } + + buildTypes { + release { + isMinifyEnabled = false + proguardFiles( + getDefaultProguardFile("proguard-android-optimize.txt"), + "proguard-rules.pro", + ) + } + } + } + } +} \ No newline at end of file diff --git a/build-logic/convention/src/main/java/kr/genti/convention/plugin/AndroidLibraryPlugin.kt b/build-logic/convention/src/main/java/kr/genti/convention/plugin/AndroidLibraryPlugin.kt new file mode 100644 index 00000000..bec5c5d6 --- /dev/null +++ b/build-logic/convention/src/main/java/kr/genti/convention/plugin/AndroidLibraryPlugin.kt @@ -0,0 +1,47 @@ +package kr.genti.convention.plugin + +import com.android.build.gradle.LibraryExtension +import kr.genti.convention.Constants +import kr.genti.convention.config.configureAndroidCommonPlugin +import kr.genti.convention.extension.kotlinOptions +import org.gradle.api.Plugin +import org.gradle.api.Project +import org.gradle.kotlin.dsl.configure + +class AndroidLibraryPlugin : Plugin { + override fun apply(target: Project) { + with(target) { + with(pluginManager) { + apply("com.android.library") + } + + extensions.configure { + configureAndroidCommonPlugin() + + compileSdk = Constants.compileSdk + + defaultConfig { + minSdk = Constants.minSdk + + testInstrumentationRunner = "androidx.test.runner.AndroidJUnitRunner" + consumerProguardFiles("consumer-rules.pro") + } + + compileOptions { + sourceCompatibility = Constants.JAVA_VERSION + targetCompatibility = Constants.JAVA_VERSION + } + + kotlinOptions { + jvmTarget = Constants.jvmVersion + } + + buildFeatures { + buildConfig = true + dataBinding = true + viewBinding = true + } + } + } + } +} \ No newline at end of file diff --git a/build-logic/convention/src/main/java/kr/genti/convention/plugin/HiltPlugin.kt b/build-logic/convention/src/main/java/kr/genti/convention/plugin/HiltPlugin.kt new file mode 100644 index 00000000..630cba21 --- /dev/null +++ b/build-logic/convention/src/main/java/kr/genti/convention/plugin/HiltPlugin.kt @@ -0,0 +1,23 @@ +package kr.genti.convention.plugin + +import kr.genti.convention.extension.getLibrary +import kr.genti.convention.extension.implementation +import kr.genti.convention.extension.kapt +import kr.genti.convention.extension.libs +import org.gradle.api.Plugin +import org.gradle.api.Project +import org.gradle.kotlin.dsl.dependencies + +class HiltPlugin : Plugin { + override fun apply(target: Project) = with(target) { + with(pluginManager) { + apply("org.jetbrains.kotlin.kapt") + apply("dagger.hilt.android.plugin") + } + + dependencies { + implementation(libs.getLibrary("hilt")) + kapt(libs.getLibrary("hilt-compiler")) + } + } +} \ No newline at end of file diff --git a/build-logic/convention/src/main/java/kr/genti/convention/plugin/JavaLibraryPlugin.kt b/build-logic/convention/src/main/java/kr/genti/convention/plugin/JavaLibraryPlugin.kt new file mode 100644 index 00000000..4866aa5b --- /dev/null +++ b/build-logic/convention/src/main/java/kr/genti/convention/plugin/JavaLibraryPlugin.kt @@ -0,0 +1,31 @@ +package kr.genti.convention.plugin + +import kr.genti.convention.Constants +import kr.genti.convention.extension.getBundle +import kr.genti.convention.extension.implementation +import kr.genti.convention.extension.libs +import org.gradle.api.Plugin +import org.gradle.api.Project +import org.gradle.api.plugins.JavaPluginExtension +import org.gradle.kotlin.dsl.configure +import org.gradle.kotlin.dsl.dependencies + +class JavaLibraryPlugin : Plugin { + override fun apply(target: Project) { + with(target) { + with(pluginManager) { + apply("org.jetbrains.kotlin.jvm") + apply("java-library") + } + + extensions.configure { + sourceCompatibility = Constants.JAVA_VERSION + targetCompatibility = Constants.JAVA_VERSION + } + + dependencies { + implementation(libs.getBundle("kotlin")) + } + } + } +} \ No newline at end of file diff --git a/build-logic/convention/src/main/java/kr/genti/convention/plugin/KotlinPlugin.kt b/build-logic/convention/src/main/java/kr/genti/convention/plugin/KotlinPlugin.kt new file mode 100644 index 00000000..98042b25 --- /dev/null +++ b/build-logic/convention/src/main/java/kr/genti/convention/plugin/KotlinPlugin.kt @@ -0,0 +1,20 @@ +package kr.genti.convention.plugin + +import kr.genti.convention.extension.getBundle +import kr.genti.convention.extension.implementation +import kr.genti.convention.extension.libs +import org.gradle.api.Plugin +import org.gradle.api.Project +import org.gradle.kotlin.dsl.dependencies + +class KotlinPlugin : Plugin { + override fun apply(target: Project) = with(target) { + with(pluginManager) { + apply("kotlin-android") + } + + dependencies { + implementation(libs.getBundle("kotlin")) + } + } +} \ No newline at end of file diff --git a/build-logic/convention/src/main/java/kr/genti/convention/plugin/TestPlugin.kt b/build-logic/convention/src/main/java/kr/genti/convention/plugin/TestPlugin.kt new file mode 100644 index 00000000..07f27941 --- /dev/null +++ b/build-logic/convention/src/main/java/kr/genti/convention/plugin/TestPlugin.kt @@ -0,0 +1,20 @@ +package kr.genti.convention.plugin + +import kr.genti.convention.extension.androidTestImplementation +import kr.genti.convention.extension.getLibrary +import kr.genti.convention.extension.libs +import kr.genti.convention.extension.testImplementation +import org.gradle.api.Plugin +import org.gradle.api.Project +import org.gradle.kotlin.dsl.dependencies + +class TestPlugin : Plugin { + override fun apply(target: Project) = with(target) { + + dependencies { + testImplementation(libs.getLibrary("j-unit")) + androidTestImplementation(libs.getLibrary("j-unit-androidx-test")) + androidTestImplementation(libs.getLibrary("espresso-core")) + } + } +} \ No newline at end of file diff --git a/build-logic/convention/src/main/java/kr/genti/convention/plugin/VersionPlugin.kt b/build-logic/convention/src/main/java/kr/genti/convention/plugin/VersionPlugin.kt new file mode 100644 index 00000000..eea141d9 --- /dev/null +++ b/build-logic/convention/src/main/java/kr/genti/convention/plugin/VersionPlugin.kt @@ -0,0 +1,14 @@ +package kr.genti.convention.plugin + +import kr.genti.convention.Constants +import org.gradle.api.Plugin +import org.gradle.api.Project + +class VersionPlugin : Plugin { + override fun apply(target: Project) = with(target) { + with(extensions) { + extraProperties["versionName"] = Constants.versionName + extraProperties["versionCode"] = Constants.versionCode + } + } +} \ No newline at end of file diff --git a/build-logic/gradle.properties b/build-logic/gradle.properties new file mode 100644 index 00000000..6977b719 --- /dev/null +++ b/build-logic/gradle.properties @@ -0,0 +1,3 @@ +org.gradle.parallel=true +org.gradle.caching=true +org.gradle.configureondemand=true \ No newline at end of file diff --git a/build-logic/settings.gradle.kts b/build-logic/settings.gradle.kts new file mode 100644 index 00000000..fd968aff --- /dev/null +++ b/build-logic/settings.gradle.kts @@ -0,0 +1,16 @@ +enableFeaturePreview("TYPESAFE_PROJECT_ACCESSORS") + +dependencyResolutionManagement { + repositories { + google() + mavenCentral() + } + versionCatalogs { + create("libs") { + from(files("../gradle/libs.versions.toml")) + } + } +} + +rootProject.name = "build-logic" +include(":convention") \ No newline at end of file diff --git a/build.gradle.kts b/build.gradle.kts index ed5abd78..5e93e036 100644 --- a/build.gradle.kts +++ b/build.gradle.kts @@ -3,18 +3,13 @@ buildscript { google() mavenCentral() } - - dependencies { - ClassPathPlugins.run { - classpath(gradle) - classpath(kotlinGradlePlugin) - classpath(hiltGradlePlugin) - classpath(googleServices) - classpath(crashlyticsGradle) - } - } } -tasks.register("clean", Delete::class) { - delete(rootProject.buildDir) -} +plugins { + alias(libs.plugins.android.application) apply false + alias(libs.plugins.kotlin.android) apply false + alias(libs.plugins.kotlin.serialization) apply false + alias(libs.plugins.hilt) apply false + alias(libs.plugins.google.services) apply false + alias(libs.plugins.google.crashlytics) apply false +} \ No newline at end of file diff --git a/buildSrc/build.gradle.kts b/buildSrc/build.gradle.kts deleted file mode 100644 index b22ed732..00000000 --- a/buildSrc/build.gradle.kts +++ /dev/null @@ -1,7 +0,0 @@ -plugins { - `kotlin-dsl` -} - -repositories { - mavenCentral() -} \ No newline at end of file diff --git a/buildSrc/src/main/kotlin/Constants.kt b/buildSrc/src/main/kotlin/Constants.kt deleted file mode 100644 index 8a436472..00000000 --- a/buildSrc/src/main/kotlin/Constants.kt +++ /dev/null @@ -1,8 +0,0 @@ -object Constants { - const val packageName = "kr.genti.android" - const val compileSdk = 34 - const val minSdk = 28 - const val targetSdk = 34 - const val versionCode = 15 - const val versionName = "1.1.6" -} diff --git a/buildSrc/src/main/kotlin/Dependencies.kt b/buildSrc/src/main/kotlin/Dependencies.kt deleted file mode 100644 index 496a1319..00000000 --- a/buildSrc/src/main/kotlin/Dependencies.kt +++ /dev/null @@ -1,91 +0,0 @@ -object KotlinDependencies { - const val kotlin = "org.jetbrains.kotlin:kotlin-stdlib:${Versions.kotlinVersion}" - const val coroutines = - "org.jetbrains.kotlinx:kotlinx-coroutines-android:${Versions.coroutinesAndroidVersion}" - const val jsonSerialization = - "org.jetbrains.kotlinx:kotlinx-serialization-json:${Versions.kotlinSerializationJsonVersion}" - const val dateTime = "org.jetbrains.kotlinx:kotlinx-datetime:${Versions.kotlinDateTimeVersion}" -} - -object AndroidXDependencies { - const val coreKtx = "androidx.core:core-ktx:${Versions.coreKtxVersion}" - const val splashScreen = "androidx.core:core-splashscreen:${Versions.splashVersion}" - - const val appCompat = "androidx.appcompat:appcompat:${Versions.appCompatVersion}" - const val constraintLayout = - "androidx.constraintlayout:constraintlayout:${Versions.constraintLayoutVersion}" - const val startup = "androidx.startup:startup-runtime:${Versions.appStartUpVersion}" - const val fragment = "androidx.fragment:fragment-ktx:${Versions.fragmentKtxVersion}" - const val legacy = "androidx.legacy:legacy-support-v4:${Versions.legacySupportVersion}" - const val security = "androidx.security:security-crypto:${Versions.securityVersion}" - - const val navigationFragment = - "androidx.navigation:navigation-fragment-ktx:${Versions.navigationVersion}" - const val navigationUi = "androidx.navigation:navigation-ui-ktx:${Versions.navigationVersion}" - - const val lifeCycleKtx = "androidx.lifecycle:lifecycle-runtime-ktx:${Versions.lifecycleVersion}" - const val lifecycleJava8 = - "androidx.lifecycle:lifecycle-common-java8:${Versions.lifecycleVersion}" - - const val hilt = "com.google.dagger:hilt-android:${Versions.hiltVersion}" - const val appUpdate = "com.google.android.play:app-update-ktx:${Versions.appUpdateVersion}" -} - -object TestDependencies { - const val jUnit = "junit:junit:${Versions.junitVersion}" - const val androidTest = "androidx.test.ext:junit:${Versions.androidTestVersion}" - const val espresso = "androidx.test.espresso:espresso-core:${Versions.espressoVersion}" -} - -object MaterialDesignDependencies { - const val materialDesign = - "com.google.android.material:material:${Versions.materialDesignVersion}" -} - -object KaptDependencies { - const val hiltCompiler = "com.google.dagger:hilt-compiler:${Versions.hiltVersion}" -} - -object RetrofitDependencies { - const val retrofit = "com.squareup.retrofit2:retrofit:${Versions.retrofitVersion}" - const val retrofitJsonConverter = - "com.jakewharton.retrofit:retrofit2-kotlinx-serialization-converter:${Versions.jsonConverterVersion}" - - const val okHttpBom = "com.squareup.okhttp3:okhttp-bom:${Versions.okHttpVersion}" - const val okHttp = "com.squareup.okhttp3:okhttp" - const val okHttpLoggingInterceptor = "com.squareup.okhttp3:logging-interceptor" -} - -object ThirdPartyDependencies { - const val coil = "io.coil-kt:coil:${Versions.coilVersion}" - const val amplitude = "com.amplitude:analytics-android:${Versions.amplitudeVersion}" - const val timber = "com.jakewharton.timber:timber:${Versions.timberVersion}" - const val progressView = "com.github.skydoves:progressview:${Versions.progressViewVersion}" - const val balloon = "com.github.skydoves:balloon:${Versions.balloonVersion}" - const val lottie = "com.airbnb.android:lottie:${Versions.lottieVersion}" - const val circularProgressBar = - "com.mikhaellopez:circularprogressbar:${Versions.circularProgressBar}" - const val circleIndicator = "me.relex:circleindicator:${Versions.circleIndicatorVersion}" - const val shimmer = "com.facebook.shimmer:shimmer:${Versions.shimmerVersion}" -} - -object ClassPathPlugins { - const val gradle = "com.android.tools.build:gradle:${Versions.gradleVersion}" - const val kotlinGradlePlugin = "org.jetbrains.kotlin:kotlin-gradle-plugin:${Versions.kotlinVersion}" - const val hiltGradlePlugin = "com.google.dagger:hilt-android-gradle-plugin:${Versions.hiltVersion}" - - const val googleServices = "com.google.gms:google-services:${Versions.googleServicesVersion}" - const val crashlyticsGradle = - "com.google.firebase:firebase-crashlytics-gradle:${Versions.crashlyticsVersion}" -} - -object FirebaseDependencies { - const val firebaseBom = "com.google.firebase:firebase-bom:${Versions.firebaseBomVersion}" - const val messaging = "com.google.firebase:firebase-messaging-ktx" - const val crashlytics = "com.google.firebase:firebase-crashlytics-ktx" - const val analytics = "com.google.firebase:firebase-analytics-ktx" -} - -object KakaoDependencies { - const val user = "com.kakao.sdk:v2-user:${Versions.kakaoVersion}" -} diff --git a/buildSrc/src/main/kotlin/Versions.kt b/buildSrc/src/main/kotlin/Versions.kt deleted file mode 100644 index 8ac3fc3f..00000000 --- a/buildSrc/src/main/kotlin/Versions.kt +++ /dev/null @@ -1,45 +0,0 @@ -import org.gradle.api.JavaVersion - -object Versions { - const val gradleVersion = "8.0.2" - const val kotlinVersion = "1.8.20" - - val javaVersion = JavaVersion.VERSION_17 - const val jvmVersion = "17" - - const val coroutinesAndroidVersion = "1.7.1" - const val kotlinSerializationJsonVersion = "1.5.1" - const val kotlinDateTimeVersion = "0.4.0" - const val coreKtxVersion = "1.10.1" - const val appCompatVersion = "1.6.1" - const val materialDesignVersion = "1.9.0" - const val constraintLayoutVersion = "2.1.4" - const val appStartUpVersion = "1.1.1" - const val legacySupportVersion = "1.0.0" - const val securityVersion = "1.1.0-alpha06" - const val hiltVersion = "2.46.1" - const val fragmentKtxVersion = "1.5.7" - const val navigationVersion = "2.7.7" - const val lifecycleVersion = "2.6.1" - const val splashVersion = "1.0.1" - const val coilVersion = "2.4.0" - const val retrofitVersion = "2.9.0" - const val jsonConverterVersion = "1.0.0" - const val okHttpVersion = "4.11.0" - const val amplitudeVersion = "1.17.3" - const val timberVersion = "5.0.1" - const val progressViewVersion = "1.1.3" - const val balloonVersion = "1.4.5" - const val lottieVersion = "6.0.0" - const val circularProgressBar = "3.1.0" - const val kakaoVersion = "2.20.3" - const val circleIndicatorVersion = "2.1.6" - const val shimmerVersion = "0.5.0" - const val junitVersion = "4.13.2" - const val espressoVersion = "3.3.0" - const val androidTestVersion = "1.1.2" - const val firebaseBomVersion = "33.1.2" - const val googleServicesVersion = "4.4.2" - const val crashlyticsVersion = "2.9.9" - const val appUpdateVersion = "2.1.0" -} diff --git a/core/build.gradle.kts b/core/build.gradle.kts index 15e47907..041fe5a5 100644 --- a/core/build.gradle.kts +++ b/core/build.gradle.kts @@ -1,50 +1,7 @@ plugins { - id("com.android.library") - kotlin("android") - kotlin("kapt") - id("dagger.hilt.android.plugin") + id("kr.genti.androidLibrary") } android { namespace = "kr.genti.core" - compileSdk = Constants.compileSdk - - defaultConfig { - minSdk = Constants.minSdk - - testInstrumentationRunner = "androidx.test.runner.AndroidJUnitRunner" - consumerProguardFiles("consumer-rules.pro") - } - compileOptions { - sourceCompatibility = Versions.javaVersion - targetCompatibility = Versions.javaVersion - } - kotlinOptions { - jvmTarget = Versions.jvmVersion - } - - buildFeatures { - dataBinding = true - viewBinding = true - } -} - -dependencies { - // Kotlin - implementation(KotlinDependencies.kotlin) - - // Lifecycle Ktx - implementation(AndroidXDependencies.lifeCycleKtx) - - // Material Design - implementation(MaterialDesignDependencies.materialDesign) - - // Hilt - implementation(AndroidXDependencies.hilt) - kapt(KaptDependencies.hiltCompiler) - - // Test Dependency - testImplementation(TestDependencies.jUnit) - androidTestImplementation(TestDependencies.androidTest) - androidTestImplementation(TestDependencies.espresso) } diff --git a/core/src/main/java/kr/genti/core/base/BaseBottomSheet.kt b/core/src/main/java/kr/genti/core/base/BaseBottomSheet.kt index fbab6e6d..8bc93a0d 100644 --- a/core/src/main/java/kr/genti/core/base/BaseBottomSheet.kt +++ b/core/src/main/java/kr/genti/core/base/BaseBottomSheet.kt @@ -1,5 +1,6 @@ package kr.genti.core.base +import android.app.Dialog import android.os.Bundle import android.view.LayoutInflater import android.view.View @@ -26,6 +27,12 @@ abstract class BaseBottomSheet( return binding.root } + override fun onCreateDialog(savedInstanceState: Bundle?): Dialog { + return super.onCreateDialog(savedInstanceState).apply { + window?.setDimAmount(0.8f) + } + } + override fun onDestroyView() { super.onDestroyView() _binding = null diff --git a/core/src/main/java/kr/genti/core/base/BaseDialog.kt b/core/src/main/java/kr/genti/core/base/BaseDialog.kt index cad653b0..67e6d5a7 100644 --- a/core/src/main/java/kr/genti/core/base/BaseDialog.kt +++ b/core/src/main/java/kr/genti/core/base/BaseDialog.kt @@ -1,5 +1,6 @@ package kr.genti.core.base +import android.app.Dialog import android.os.Bundle import android.view.LayoutInflater import android.view.View @@ -8,6 +9,7 @@ import androidx.annotation.LayoutRes import androidx.databinding.DataBindingUtil import androidx.databinding.ViewDataBinding import androidx.fragment.app.DialogFragment +import kr.genti.core.R abstract class BaseDialog( @LayoutRes private val layoutRes: Int, @@ -26,6 +28,13 @@ abstract class BaseDialog( return binding.root } + override fun onCreateDialog(savedInstanceState: Bundle?): Dialog { + return super.onCreateDialog(savedInstanceState).apply { + window?.setWindowAnimations(R.style.DialogAnimation) + window?.setDimAmount(0.8f) + } + } + override fun onDestroyView() { super.onDestroyView() _binding = null diff --git a/core/src/main/java/kr/genti/core/extension/StringExt.kt b/core/src/main/java/kr/genti/core/extension/StringExt.kt index b0a1764a..d60841ea 100644 --- a/core/src/main/java/kr/genti/core/extension/StringExt.kt +++ b/core/src/main/java/kr/genti/core/extension/StringExt.kt @@ -18,3 +18,5 @@ fun String.getGraphemeLength(): Int { return count } + +fun String.breakLines(): String = this.replace(Regex("\\s+"), " ") \ No newline at end of file diff --git a/core/src/main/java/kr/genti/core/extension/ViewExt.kt b/core/src/main/java/kr/genti/core/extension/ViewExt.kt index 3ef6859c..1ead55e2 100644 --- a/core/src/main/java/kr/genti/core/extension/ViewExt.kt +++ b/core/src/main/java/kr/genti/core/extension/ViewExt.kt @@ -1,9 +1,14 @@ package kr.genti.core.extension +import android.graphics.LinearGradient import android.graphics.RenderEffect import android.graphics.Shader import android.os.Build +import android.text.Spannable +import android.text.SpannableString +import android.text.style.ImageSpan import android.view.View +import android.widget.TextView inline fun View.setOnSingleClickListener( delay: Long = 1000L, @@ -30,3 +35,22 @@ fun View.setGusianBlur(radius: Float?) { } } } + +fun TextView.setGradientText(startColor: Int, endColor: Int) { + paint.shader = LinearGradient( + 0f, + 0f, + paint.measureText(text.toString()), + textSize, + intArrayOf(startColor, endColor), + null, + Shader.TileMode.CLAMP + ) +} + +fun TextView.setTextWithImage(text: String, imageResId: Int) { + val spannableString = SpannableString(" $text") + val imageSpan = ImageSpan(context, imageResId, ImageSpan.ALIGN_BOTTOM) + spannableString.setSpan(imageSpan, 0, 1, Spannable.SPAN_EXCLUSIVE_EXCLUSIVE) + this.text = spannableString +} \ No newline at end of file diff --git a/core/src/main/res/anim/fade_in.xml b/core/src/main/res/anim/fade_in.xml new file mode 100644 index 00000000..712773e4 --- /dev/null +++ b/core/src/main/res/anim/fade_in.xml @@ -0,0 +1,5 @@ + + \ No newline at end of file diff --git a/core/src/main/res/anim/fade_out.xml b/core/src/main/res/anim/fade_out.xml new file mode 100644 index 00000000..6f15c325 --- /dev/null +++ b/core/src/main/res/anim/fade_out.xml @@ -0,0 +1,5 @@ + + \ No newline at end of file diff --git a/core/src/main/res/values/styles.xml b/core/src/main/res/values/styles.xml new file mode 100644 index 00000000..a45c00f0 --- /dev/null +++ b/core/src/main/res/values/styles.xml @@ -0,0 +1,8 @@ + + + + + \ No newline at end of file diff --git a/data/build.gradle.kts b/data/build.gradle.kts index 56c5dcb2..61889a67 100644 --- a/data/build.gradle.kts +++ b/data/build.gradle.kts @@ -1,66 +1,14 @@ plugins { - id("com.android.library") - kotlin("android") - kotlin("kapt") - kotlin("plugin.serialization") version Versions.kotlinVersion + id("kr.genti.androidLibrary") } android { namespace = "kr.genti.data" - compileSdk = Constants.compileSdk - - defaultConfig { - minSdk = Constants.minSdk - - testInstrumentationRunner = "androidx.test.runner.AndroidJUnitRunner" - consumerProguardFiles("consumer-rules.pro") - } - - compileOptions { - sourceCompatibility = Versions.javaVersion - targetCompatibility = Versions.javaVersion - } - - kotlinOptions { - jvmTarget = Versions.jvmVersion - } - - buildFeatures { - buildConfig = true - } } dependencies { - implementation(project(":domain")) - - AndroidXDependencies.run { - implementation(hilt) - implementation(security) - implementation(coreKtx) - } - - KotlinDependencies.run { - implementation(kotlin) - implementation(jsonSerialization) - implementation(coroutines) - implementation(dateTime) - } - - RetrofitDependencies.run { - implementation(platform(okHttpBom)) - implementation(okHttp) - implementation(okHttpLoggingInterceptor) - implementation(retrofit) - implementation(retrofitJsonConverter) - } - - ThirdPartyDependencies.run { - implementation(timber) - } + implementation(projects.domain) - TestDependencies.run { - testImplementation(jUnit) - androidTestImplementation(androidTest) - androidTestImplementation(espresso) - } + implementation(platform(libs.okhttp.bom)) + implementation(libs.bundles.networking) } diff --git a/data/src/main/java/kr/genti/data/dataSource/CreateDataSource.kt b/data/src/main/java/kr/genti/data/dataSource/CreateDataSource.kt index a2797965..ec7e929d 100644 --- a/data/src/main/java/kr/genti/data/dataSource/CreateDataSource.kt +++ b/data/src/main/java/kr/genti/data/dataSource/CreateDataSource.kt @@ -2,8 +2,10 @@ package kr.genti.data.dataSource import kr.genti.data.dto.BaseResponse import kr.genti.data.dto.request.CreateRequestDto +import kr.genti.data.dto.request.CreateTwoRequestDto import kr.genti.data.dto.request.KeyRequestDto import kr.genti.data.dto.request.S3RequestDto +import kr.genti.data.dto.response.PromptExampleDto import kr.genti.data.dto.response.S3PresignedUrlDto interface CreateDataSource { @@ -13,5 +15,11 @@ interface CreateDataSource { suspend fun postToCreate(request: CreateRequestDto): BaseResponse + suspend fun postToCreateOne(request: CreateRequestDto): BaseResponse + + suspend fun postToCreateTwo(request: CreateTwoRequestDto): BaseResponse + suspend fun postToVerify(request: KeyRequestDto): BaseResponse + + suspend fun getPromptExample(): BaseResponse> } diff --git a/data/src/main/java/kr/genti/data/dataSource/GenerateDataSource.kt b/data/src/main/java/kr/genti/data/dataSource/GenerateDataSource.kt index 1ae9ae3c..0a1fd96d 100644 --- a/data/src/main/java/kr/genti/data/dataSource/GenerateDataSource.kt +++ b/data/src/main/java/kr/genti/data/dataSource/GenerateDataSource.kt @@ -33,4 +33,6 @@ interface GenerateDataSource { suspend fun getIsUserVerified(): BaseResponse suspend fun getIsServerAvailable(): BaseResponse + + suspend fun patchStatusInDevelop(): BaseResponse } diff --git a/data/src/main/java/kr/genti/data/dataSourceImpl/CreateDataSourceImpl.kt b/data/src/main/java/kr/genti/data/dataSourceImpl/CreateDataSourceImpl.kt index 82b55e2f..cbb8815b 100644 --- a/data/src/main/java/kr/genti/data/dataSourceImpl/CreateDataSourceImpl.kt +++ b/data/src/main/java/kr/genti/data/dataSourceImpl/CreateDataSourceImpl.kt @@ -3,23 +3,37 @@ package kr.genti.data.dataSourceImpl import kr.genti.data.dataSource.CreateDataSource import kr.genti.data.dto.BaseResponse import kr.genti.data.dto.request.CreateRequestDto +import kr.genti.data.dto.request.CreateTwoRequestDto import kr.genti.data.dto.request.KeyRequestDto import kr.genti.data.dto.request.S3RequestDto +import kr.genti.data.dto.response.PromptExampleDto import kr.genti.data.dto.response.S3PresignedUrlDto import kr.genti.data.service.CreateService import javax.inject.Inject data class CreateDataSourceImpl - @Inject - constructor( - private val createService: CreateService, - ) : CreateDataSource { - override suspend fun getSingleS3Url(request: S3RequestDto): BaseResponse = createService.getSingleS3Url(request) +@Inject +constructor( + private val createService: CreateService, +) : CreateDataSource { + override suspend fun getSingleS3Url(request: S3RequestDto): BaseResponse = + createService.getSingleS3Url(request) - override suspend fun getMultiS3Url(request: List): BaseResponse> = - createService.getMultiS3Url(request) + override suspend fun getMultiS3Url(request: List): BaseResponse> = + createService.getMultiS3Url(request) - override suspend fun postToCreate(request: CreateRequestDto): BaseResponse = createService.postToCreate(request) + override suspend fun postToCreate(request: CreateRequestDto): BaseResponse = + createService.postToCreate(request) - override suspend fun postToVerify(request: KeyRequestDto): BaseResponse = createService.postToVerify(request) - } + override suspend fun postToCreateOne(request: CreateRequestDto): BaseResponse = + createService.postToCreateOne(request) + + override suspend fun postToCreateTwo(request: CreateTwoRequestDto): BaseResponse = + createService.postToCreateTwo(request) + + override suspend fun postToVerify(request: KeyRequestDto): BaseResponse = + createService.postToVerify(request) + + override suspend fun getPromptExample(): BaseResponse> = + createService.getPromptExample() +} diff --git a/data/src/main/java/kr/genti/data/dataSourceImpl/GenerateDataSourceImpl.kt b/data/src/main/java/kr/genti/data/dataSourceImpl/GenerateDataSourceImpl.kt index b4bed88e..afa881f4 100644 --- a/data/src/main/java/kr/genti/data/dataSourceImpl/GenerateDataSourceImpl.kt +++ b/data/src/main/java/kr/genti/data/dataSourceImpl/GenerateDataSourceImpl.kt @@ -11,35 +11,44 @@ import kr.genti.data.service.GenerateService import javax.inject.Inject data class GenerateDataSourceImpl - @Inject - constructor( - private val generateService: GenerateService, - ) : GenerateDataSource { - override suspend fun getGenerateStatus(): BaseResponse = generateService.getGenerateStatus() - - override suspend fun getGeneratedPictureList( - page: Int, - size: Int, - sortBy: String?, - direction: String?, - ): BaseResponse = generateService.getGeneratedPictureList(page, size, sortBy, direction) - - override suspend fun postGenerateReport(request: ReportRequestDto): BaseResponse = - generateService.postGenerateReport(request) - - override suspend fun postGenerateRate( - responseId: Int, - star: Int, - ): BaseResponse = generateService.postGenerateRate(responseId, star) - - override suspend fun postVerifyGenerateState(responseId: Int): BaseResponse = - generateService.postVerifyGenerateState(responseId) - - override suspend fun getCanceledToReset(requestId: String): BaseResponse = generateService.getCanceledToReset(requestId) - - override suspend fun getOpenchatData(): BaseResponse = generateService.getOpenchatData() - - override suspend fun getIsUserVerified(): BaseResponse = generateService.getIsUserVerified() - - override suspend fun getIsServerAvailable(): BaseResponse = generateService.getIsServerAvailable() - } +@Inject +constructor( + private val generateService: GenerateService, +) : GenerateDataSource { + override suspend fun getGenerateStatus(): BaseResponse = + generateService.getGenerateStatus() + + override suspend fun getGeneratedPictureList( + page: Int, + size: Int, + sortBy: String?, + direction: String?, + ): BaseResponse = + generateService.getGeneratedPictureList(page, size, sortBy, direction) + + override suspend fun postGenerateReport(request: ReportRequestDto): BaseResponse = + generateService.postGenerateReport(request) + + override suspend fun postGenerateRate( + responseId: Int, + star: Int, + ): BaseResponse = generateService.postGenerateRate(responseId, star) + + override suspend fun postVerifyGenerateState(responseId: Int): BaseResponse = + generateService.postVerifyGenerateState(responseId) + + override suspend fun getCanceledToReset(requestId: String): BaseResponse = + generateService.getCanceledToReset(requestId) + + override suspend fun getOpenchatData(): BaseResponse = + generateService.getOpenchatData() + + override suspend fun getIsUserVerified(): BaseResponse = + generateService.getIsUserVerified() + + override suspend fun getIsServerAvailable(): BaseResponse = + generateService.getIsServerAvailable() + + override suspend fun patchStatusInDevelop(): BaseResponse = + generateService.patchStatusInDevelop() +} diff --git a/data/src/main/java/kr/genti/data/dto/request/CreateRequestDto.kt b/data/src/main/java/kr/genti/data/dto/request/CreateRequestDto.kt index f50f5795..3a95e1f2 100644 --- a/data/src/main/java/kr/genti/data/dto/request/CreateRequestDto.kt +++ b/data/src/main/java/kr/genti/data/dto/request/CreateRequestDto.kt @@ -4,22 +4,14 @@ import kotlinx.serialization.SerialName import kotlinx.serialization.Serializable import kr.genti.data.dto.request.KeyRequestDto.Companion.toDto import kr.genti.domain.entity.request.CreateRequestModel -import kr.genti.domain.enums.CameraAngle import kr.genti.domain.enums.PictureRatio -import kr.genti.domain.enums.ShotCoverage @Serializable data class CreateRequestDto( @SerialName("prompt") val prompt: String, - @SerialName("posePicture") - val posePicture: KeyRequestDto?, @SerialName("facePictureList") val facePictureList: List, - @SerialName("cameraAngle") - val cameraAngle: CameraAngle, - @SerialName("shotCoverage") - val shotCoverage: ShotCoverage, @SerialName("pictureRatio") val pictureRatio: PictureRatio, ) { @@ -27,10 +19,7 @@ data class CreateRequestDto( fun CreateRequestModel.toDto() = CreateRequestDto( prompt, - posePicture?.toDto(), facePictureList.map { it.toDto() }, - cameraAngle, - shotCoverage, pictureRatio, ) } diff --git a/data/src/main/java/kr/genti/data/dto/request/CreateTwoRequestDto.kt b/data/src/main/java/kr/genti/data/dto/request/CreateTwoRequestDto.kt new file mode 100644 index 00000000..4876c55e --- /dev/null +++ b/data/src/main/java/kr/genti/data/dto/request/CreateTwoRequestDto.kt @@ -0,0 +1,29 @@ +package kr.genti.data.dto.request + +import kotlinx.serialization.SerialName +import kotlinx.serialization.Serializable +import kr.genti.data.dto.request.KeyRequestDto.Companion.toDto +import kr.genti.domain.entity.request.CreateTwoRequestModel +import kr.genti.domain.enums.PictureRatio + +@Serializable +data class CreateTwoRequestDto( + @SerialName("prompt") + val prompt: String, + @SerialName("facePictureList") + val facePictureList: List, + @SerialName("otherFacePictureList") + val otherFacePictureList: List, + @SerialName("pictureRatio") + val pictureRatio: PictureRatio, +) { + companion object { + fun CreateTwoRequestModel.toDto() = + CreateTwoRequestDto( + prompt, + facePictureList.map { it.toDto() }, + otherFacePictureList.map { it.toDto() }, + pictureRatio, + ) + } +} diff --git a/data/src/main/java/kr/genti/data/dto/response/GenerateStatusDto.kt b/data/src/main/java/kr/genti/data/dto/response/GenerateStatusDto.kt index 04abf79c..a6995d07 100644 --- a/data/src/main/java/kr/genti/data/dto/response/GenerateStatusDto.kt +++ b/data/src/main/java/kr/genti/data/dto/response/GenerateStatusDto.kt @@ -13,6 +13,8 @@ data class GenerateStatusDto( val status: GenerateStatus, @SerialName("pictureGenerateResponse") val pictureGenerateResponse: GenerateResponseDto?, + @SerialName("paid") + val paid: Boolean?, ) { @Serializable data class GenerateResponseDto( @@ -33,5 +35,6 @@ data class GenerateStatusDto( pictureGenerateRequestId, status, pictureGenerateResponse?.toModel(), + paid, ) } diff --git a/data/src/main/java/kr/genti/data/dto/response/PromptExampleDto.kt b/data/src/main/java/kr/genti/data/dto/response/PromptExampleDto.kt new file mode 100644 index 00000000..2bf433ee --- /dev/null +++ b/data/src/main/java/kr/genti/data/dto/response/PromptExampleDto.kt @@ -0,0 +1,15 @@ +package kr.genti.data.dto.response + +import kotlinx.serialization.SerialName +import kotlinx.serialization.Serializable +import kr.genti.domain.entity.response.PromptExampleModel + +@Serializable +data class PromptExampleDto( + @SerialName("url") + val url: String, + @SerialName("prompt") + val prompt: String, +) { + fun toModel() = PromptExampleModel(url, prompt) +} \ No newline at end of file diff --git a/data/src/main/java/kr/genti/data/local/UserSharedPref.kt b/data/src/main/java/kr/genti/data/local/UserSharedPref.kt index 35f3fd8b..2214945b 100644 --- a/data/src/main/java/kr/genti/data/local/UserSharedPref.kt +++ b/data/src/main/java/kr/genti/data/local/UserSharedPref.kt @@ -4,8 +4,6 @@ interface UserSharedPref { var accessToken: String var refreshToken: String var userRole: String - var isGuideNeeded: Boolean - var isChatAccessible: Boolean fun clearInfo() } diff --git a/data/src/main/java/kr/genti/data/local/UserSharedPrefImpl.kt b/data/src/main/java/kr/genti/data/local/UserSharedPrefImpl.kt index 50c6eaba..03bb6571 100644 --- a/data/src/main/java/kr/genti/data/local/UserSharedPrefImpl.kt +++ b/data/src/main/java/kr/genti/data/local/UserSharedPrefImpl.kt @@ -21,14 +21,6 @@ class UserSharedPrefImpl get() = dataStore.getString(USER_ROLE, "").orEmpty() set(value) = dataStore.edit { putString(USER_ROLE, value) } - override var isGuideNeeded: Boolean - get() = dataStore.getBoolean(IS_GUIDE_NEEDED, true) - set(value) = dataStore.edit { putBoolean(IS_GUIDE_NEEDED, value) } - - override var isChatAccessible: Boolean - get() = dataStore.getBoolean(IS_CHAT_ACCESSIBLE, true) - set(value) = dataStore.edit { putBoolean(IS_CHAT_ACCESSIBLE, value) } - override fun clearInfo() { dataStore.edit().clear().apply() } @@ -37,7 +29,6 @@ class UserSharedPrefImpl private const val ACCESS_TOKEN = "ACCESS_TOKEN" private const val REFRESH_TOKEN = "REFRESH_TOKEN" private const val USER_ROLE = "USER_ROLE" - private const val IS_GUIDE_NEEDED = "IS_GUIDE_NEEDED" private const val IS_CHAT_ACCESSIBLE = "IS_CHAT_ACCESSIBLE" } } diff --git a/data/src/main/java/kr/genti/data/repositoryImpl/CreateRepositoryImpl.kt b/data/src/main/java/kr/genti/data/repositoryImpl/CreateRepositoryImpl.kt index dc3db02e..d3e4435f 100644 --- a/data/src/main/java/kr/genti/data/repositoryImpl/CreateRepositoryImpl.kt +++ b/data/src/main/java/kr/genti/data/repositoryImpl/CreateRepositoryImpl.kt @@ -2,37 +2,55 @@ package kr.genti.data.repositoryImpl import kr.genti.data.dataSource.CreateDataSource import kr.genti.data.dto.request.CreateRequestDto.Companion.toDto +import kr.genti.data.dto.request.CreateTwoRequestDto.Companion.toDto import kr.genti.data.dto.request.KeyRequestDto.Companion.toDto import kr.genti.data.dto.request.S3RequestDto.Companion.toDto import kr.genti.domain.entity.request.CreateRequestModel +import kr.genti.domain.entity.request.CreateTwoRequestModel import kr.genti.domain.entity.request.KeyRequestModel import kr.genti.domain.entity.request.S3RequestModel +import kr.genti.domain.entity.response.PromptExampleModel import kr.genti.domain.entity.response.S3PresignedUrlModel import kr.genti.domain.repository.CreateRepository import javax.inject.Inject class CreateRepositoryImpl - @Inject - constructor( - private val createDataSource: CreateDataSource, - ) : CreateRepository { - override suspend fun getS3SingleUrl(request: S3RequestModel): Result = - runCatching { - createDataSource.getSingleS3Url(request.toDto()).response.toModel() - } - - override suspend fun getS3MultiUrl(request: List): Result> = - runCatching { - createDataSource.getMultiS3Url(request.map { it.toDto() }).response.map { it.toModel() } - } - - override suspend fun postToCreate(request: CreateRequestModel): Result = - runCatching { - createDataSource.postToCreate(request.toDto()).response - } - - override suspend fun postToVerify(request: KeyRequestModel): Result = - runCatching { - createDataSource.postToVerify(request.toDto()).response - } - } +@Inject +constructor( + private val createDataSource: CreateDataSource, +) : CreateRepository { + override suspend fun getS3SingleUrl(request: S3RequestModel): Result = + runCatching { + createDataSource.getSingleS3Url(request.toDto()).response.toModel() + } + + override suspend fun getS3MultiUrl(request: List): Result> = + runCatching { + createDataSource.getMultiS3Url(request.map { it.toDto() }).response.map { it.toModel() } + } + + override suspend fun postToCreate(request: CreateRequestModel): Result = + runCatching { + createDataSource.postToCreate(request.toDto()).response + } + + override suspend fun postToCreateOne(request: CreateRequestModel): Result = + runCatching { + createDataSource.postToCreateOne(request.toDto()).response + } + + override suspend fun postToCreateTwo(request: CreateTwoRequestModel): Result = + runCatching { + createDataSource.postToCreateTwo(request.toDto()).response + } + + override suspend fun postToVerify(request: KeyRequestModel): Result = + runCatching { + createDataSource.postToVerify(request.toDto()).response + } + + override suspend fun getPromptExample(): Result> = + runCatching { + createDataSource.getPromptExample().response.map { it.toModel() } + } +} diff --git a/data/src/main/java/kr/genti/data/repositoryImpl/GenerateRepositoryImpl.kt b/data/src/main/java/kr/genti/data/repositoryImpl/GenerateRepositoryImpl.kt index 50a31f7c..5220b5c8 100644 --- a/data/src/main/java/kr/genti/data/repositoryImpl/GenerateRepositoryImpl.kt +++ b/data/src/main/java/kr/genti/data/repositoryImpl/GenerateRepositoryImpl.kt @@ -11,67 +11,72 @@ import kr.genti.domain.repository.GenerateRepository import javax.inject.Inject class GenerateRepositoryImpl - @Inject - constructor( - private val generateDataSource: GenerateDataSource, - ) : GenerateRepository { - override suspend fun getGenerateStatus(): Result = - runCatching { - generateDataSource.getGenerateStatus().response.toModel() - } +@Inject +constructor( + private val generateDataSource: GenerateDataSource, +) : GenerateRepository { + override suspend fun getGenerateStatus(): Result = + runCatching { + generateDataSource.getGenerateStatus().response.toModel() + } - override suspend fun getGeneratedPictureList( - page: Int, - size: Int, - sortBy: String?, - direction: String?, - ): Result = - runCatching { - generateDataSource - .getGeneratedPictureList( - page, - size, - sortBy, - direction, - ).response - .toModel() - } + override suspend fun getGeneratedPictureList( + page: Int, + size: Int, + sortBy: String?, + direction: String?, + ): Result = + runCatching { + generateDataSource + .getGeneratedPictureList( + page, + size, + sortBy, + direction, + ).response + .toModel() + } - override suspend fun postGenerateReport(request: ReportRequestModel): Result = - runCatching { - generateDataSource.postGenerateReport(request.toDto()).response - } + override suspend fun postGenerateReport(request: ReportRequestModel): Result = + runCatching { + generateDataSource.postGenerateReport(request.toDto()).response + } - override suspend fun postGenerateRate( - responseId: Int, - star: Int, - ): Result = - runCatching { - generateDataSource.postGenerateRate(responseId, star).response - } + override suspend fun postGenerateRate( + responseId: Int, + star: Int, + ): Result = + runCatching { + generateDataSource.postGenerateRate(responseId, star).response + } - override suspend fun postVerifyGenerateState(responseId: Int): Result = - runCatching { - generateDataSource.postVerifyGenerateState(responseId).response - } + override suspend fun postVerifyGenerateState(responseId: Int): Result = + runCatching { + generateDataSource.postVerifyGenerateState(responseId).response + } - override suspend fun getCanceledToReset(requestId: String): Result = - runCatching { - generateDataSource.getCanceledToReset(requestId).response - } + override suspend fun getCanceledToReset(requestId: String): Result = + runCatching { + generateDataSource.getCanceledToReset(requestId).response + } - override suspend fun getOpenchatData(): Result = - runCatching { - generateDataSource.getOpenchatData().response.toModel() - } + override suspend fun getOpenchatData(): Result = + runCatching { + generateDataSource.getOpenchatData().response.toModel() + } - override suspend fun getIsUserVerified(): Result = - runCatching { - generateDataSource.getIsUserVerified().response - } + override suspend fun getIsUserVerified(): Result = + runCatching { + generateDataSource.getIsUserVerified().response + } - override suspend fun getIsServerAvailable(): Result = - runCatching { - generateDataSource.getIsServerAvailable().response.toModel() - } - } + override suspend fun getIsServerAvailable(): Result = + runCatching { + generateDataSource.getIsServerAvailable().response.toModel() + } + + override suspend fun patchStatusInDevelop(): Result = + runCatching { + generateDataSource.patchStatusInDevelop().response + } +} diff --git a/data/src/main/java/kr/genti/data/repositoryImpl/UserRepositoryImpl.kt b/data/src/main/java/kr/genti/data/repositoryImpl/UserRepositoryImpl.kt index 4200e4ab..876214a7 100644 --- a/data/src/main/java/kr/genti/data/repositoryImpl/UserRepositoryImpl.kt +++ b/data/src/main/java/kr/genti/data/repositoryImpl/UserRepositoryImpl.kt @@ -15,10 +15,6 @@ class UserRepositoryImpl override fun getUserRole(): String = userSharedPref.userRole - override fun getIsGuideNeeded(): Boolean = userSharedPref.isGuideNeeded - - override fun getIsChatAccessible(): Boolean = userSharedPref.isChatAccessible - override fun setTokens( accessToken: String, refreshToken: String, @@ -31,14 +27,6 @@ class UserRepositoryImpl userSharedPref.userRole = userRole } - override fun setIsGuideNeeded(isGuideNeeded: Boolean) { - userSharedPref.isGuideNeeded = isGuideNeeded - } - - override fun setIsChatAccessible(isChatAccessible: Boolean) { - userSharedPref.isChatAccessible = isChatAccessible - } - override fun clearInfo() { userSharedPref.clearInfo() } diff --git a/data/src/main/java/kr/genti/data/service/CreateService.kt b/data/src/main/java/kr/genti/data/service/CreateService.kt index df5c76b0..b84cf4be 100644 --- a/data/src/main/java/kr/genti/data/service/CreateService.kt +++ b/data/src/main/java/kr/genti/data/service/CreateService.kt @@ -2,10 +2,13 @@ package kr.genti.data.service import kr.genti.data.dto.BaseResponse import kr.genti.data.dto.request.CreateRequestDto +import kr.genti.data.dto.request.CreateTwoRequestDto import kr.genti.data.dto.request.KeyRequestDto import kr.genti.data.dto.request.S3RequestDto +import kr.genti.data.dto.response.PromptExampleDto import kr.genti.data.dto.response.S3PresignedUrlDto import retrofit2.http.Body +import retrofit2.http.GET import retrofit2.http.POST interface CreateService { @@ -24,8 +27,21 @@ interface CreateService { @Body request: CreateRequestDto, ): BaseResponse + @POST("api/v1/users/picture-generate-requests/paid/one") + suspend fun postToCreateOne( + @Body request: CreateRequestDto, + ): BaseResponse + + @POST("api/v1/users/picture-generate-requests/paid/two") + suspend fun postToCreateTwo( + @Body request: CreateTwoRequestDto, + ): BaseResponse + @POST("api/v1/user-verification") suspend fun postToVerify( @Body request: KeyRequestDto, ): BaseResponse + + @GET("api/v1/users/examples/with-picture-square") + suspend fun getPromptExample(): BaseResponse> } diff --git a/data/src/main/java/kr/genti/data/service/GenerateService.kt b/data/src/main/java/kr/genti/data/service/GenerateService.kt index 1b4e77f5..c90c0a13 100644 --- a/data/src/main/java/kr/genti/data/service/GenerateService.kt +++ b/data/src/main/java/kr/genti/data/service/GenerateService.kt @@ -53,4 +53,7 @@ interface GenerateService { @GET("api/v1/maintenance") suspend fun getIsServerAvailable(): BaseResponse + + @POST("api/v1/frontend/picture-generate-responses") + suspend fun patchStatusInDevelop(): BaseResponse } diff --git a/domain/build.gradle.kts b/domain/build.gradle.kts index 649449cf..5b9bf4c5 100644 --- a/domain/build.gradle.kts +++ b/domain/build.gradle.kts @@ -1,18 +1,3 @@ plugins { - id("java-library") - kotlin("jvm") - kotlin("kapt") -} - -java { - sourceCompatibility = Versions.javaVersion - targetCompatibility = Versions.javaVersion -} - -dependencies { - KotlinDependencies.run { - implementation(kotlin) - implementation(coroutines) - implementation(dateTime) - } + id("kr.genti.javaLibrary") } \ No newline at end of file diff --git a/domain/src/main/kotlin/kr/genti/domain/entity/request/CreateRequestModel.kt b/domain/src/main/kotlin/kr/genti/domain/entity/request/CreateRequestModel.kt index 742e7ae3..75e522ea 100644 --- a/domain/src/main/kotlin/kr/genti/domain/entity/request/CreateRequestModel.kt +++ b/domain/src/main/kotlin/kr/genti/domain/entity/request/CreateRequestModel.kt @@ -1,14 +1,9 @@ package kr.genti.domain.entity.request -import kr.genti.domain.enums.CameraAngle import kr.genti.domain.enums.PictureRatio -import kr.genti.domain.enums.ShotCoverage data class CreateRequestModel( val prompt: String, - val posePicture: KeyRequestModel?, val facePictureList: List, - val cameraAngle: CameraAngle, - val shotCoverage: ShotCoverage, val pictureRatio: PictureRatio, ) diff --git a/domain/src/main/kotlin/kr/genti/domain/entity/request/CreateTwoRequestModel.kt b/domain/src/main/kotlin/kr/genti/domain/entity/request/CreateTwoRequestModel.kt new file mode 100644 index 00000000..e96793ed --- /dev/null +++ b/domain/src/main/kotlin/kr/genti/domain/entity/request/CreateTwoRequestModel.kt @@ -0,0 +1,10 @@ +package kr.genti.domain.entity.request + +import kr.genti.domain.enums.PictureRatio + +data class CreateTwoRequestModel( + val prompt: String, + val facePictureList: List, + val otherFacePictureList: List, + val pictureRatio: PictureRatio, +) diff --git a/domain/src/main/kotlin/kr/genti/domain/entity/response/GenerateStatusModel.kt b/domain/src/main/kotlin/kr/genti/domain/entity/response/GenerateStatusModel.kt index cd946bd2..0ed62e7c 100644 --- a/domain/src/main/kotlin/kr/genti/domain/entity/response/GenerateStatusModel.kt +++ b/domain/src/main/kotlin/kr/genti/domain/entity/response/GenerateStatusModel.kt @@ -3,12 +3,13 @@ package kr.genti.domain.entity.response import kr.genti.domain.enums.GenerateStatus data class GenerateStatusModel( - val pictureGenerateRequestId: Long?, + val requestId: Long?, val status: GenerateStatus, - val pictureGenerateResponse: GenerateResponseModel?, + val response: GenerateResponseModel?, + val paid: Boolean?, ) { data class GenerateResponseModel( - val pictureGenerateResponseId: Long, - val pictureCompleted: ImageModel?, + val responseId: Long, + val picture: ImageModel?, ) } diff --git a/domain/src/main/kotlin/kr/genti/domain/entity/response/PromptExampleModel.kt b/domain/src/main/kotlin/kr/genti/domain/entity/response/PromptExampleModel.kt new file mode 100644 index 00000000..8c9a8e20 --- /dev/null +++ b/domain/src/main/kotlin/kr/genti/domain/entity/response/PromptExampleModel.kt @@ -0,0 +1,6 @@ +package kr.genti.domain.entity.response + +data class PromptExampleModel( + val url: String, + val prompt: String, +) \ No newline at end of file diff --git a/domain/src/main/kotlin/kr/genti/domain/enums/CameraAngle.kt b/domain/src/main/kotlin/kr/genti/domain/enums/CameraAngle.kt deleted file mode 100644 index 0fb02510..00000000 --- a/domain/src/main/kotlin/kr/genti/domain/enums/CameraAngle.kt +++ /dev/null @@ -1,13 +0,0 @@ -package kr.genti.domain.enums - -enum class CameraAngle(private val description: String) { - ANY("앵글은 자유롭게 맡길래요"), - HIGH("위에서 촬영"), - MIDDLE("눈높이에서 촬영"), - LOW("아래에서 촬영"), - ; - - override fun toString(): String { - return description - } -} diff --git a/domain/src/main/kotlin/kr/genti/domain/enums/PictureNumber.kt b/domain/src/main/kotlin/kr/genti/domain/enums/PictureNumber.kt new file mode 100644 index 00000000..b16d57eb --- /dev/null +++ b/domain/src/main/kotlin/kr/genti/domain/enums/PictureNumber.kt @@ -0,0 +1,17 @@ +package kr.genti.domain.enums + +enum class PictureNumber { + ONE, + TWO, + ; + + companion object { + fun String.toPictureNumber(): PictureNumber { + return if (this == "ONE") { + ONE + } else { + TWO + } + } + } +} \ No newline at end of file diff --git a/domain/src/main/kotlin/kr/genti/domain/enums/ShotCoverage.kt b/domain/src/main/kotlin/kr/genti/domain/enums/ShotCoverage.kt deleted file mode 100644 index 14128e66..00000000 --- a/domain/src/main/kotlin/kr/genti/domain/enums/ShotCoverage.kt +++ /dev/null @@ -1,13 +0,0 @@ -package kr.genti.domain.enums - -enum class ShotCoverage(private val description: String) { - ANY("프레임은 자유롭게 맡길래요"), - UPPER_BODY("바스트샷\n(상반신)"), - KNEE_SHOT("니샷\n(무릎 위)"), - FULL_BODY("풀샷\n(전신)"), - ; - - override fun toString(): String { - return description - } -} diff --git a/domain/src/main/kotlin/kr/genti/domain/repository/CreateRepository.kt b/domain/src/main/kotlin/kr/genti/domain/repository/CreateRepository.kt index 9535244f..bab39b95 100644 --- a/domain/src/main/kotlin/kr/genti/domain/repository/CreateRepository.kt +++ b/domain/src/main/kotlin/kr/genti/domain/repository/CreateRepository.kt @@ -1,8 +1,10 @@ package kr.genti.domain.repository import kr.genti.domain.entity.request.CreateRequestModel +import kr.genti.domain.entity.request.CreateTwoRequestModel import kr.genti.domain.entity.request.KeyRequestModel import kr.genti.domain.entity.request.S3RequestModel +import kr.genti.domain.entity.response.PromptExampleModel import kr.genti.domain.entity.response.S3PresignedUrlModel interface CreateRepository { @@ -12,5 +14,11 @@ interface CreateRepository { suspend fun postToCreate(request: CreateRequestModel): Result + suspend fun postToCreateOne(request: CreateRequestModel): Result + + suspend fun postToCreateTwo(request: CreateTwoRequestModel): Result + suspend fun postToVerify(request: KeyRequestModel): Result + + suspend fun getPromptExample(): Result> } diff --git a/domain/src/main/kotlin/kr/genti/domain/repository/GenerateRepository.kt b/domain/src/main/kotlin/kr/genti/domain/repository/GenerateRepository.kt index 7a34143a..3ea3b1bf 100644 --- a/domain/src/main/kotlin/kr/genti/domain/repository/GenerateRepository.kt +++ b/domain/src/main/kotlin/kr/genti/domain/repository/GenerateRepository.kt @@ -32,4 +32,6 @@ interface GenerateRepository { suspend fun getIsUserVerified(): Result suspend fun getIsServerAvailable(): Result + + suspend fun patchStatusInDevelop(): Result } diff --git a/domain/src/main/kotlin/kr/genti/domain/repository/UserRepository.kt b/domain/src/main/kotlin/kr/genti/domain/repository/UserRepository.kt index e103fbaa..2868d969 100644 --- a/domain/src/main/kotlin/kr/genti/domain/repository/UserRepository.kt +++ b/domain/src/main/kotlin/kr/genti/domain/repository/UserRepository.kt @@ -7,10 +7,6 @@ interface UserRepository { fun getUserRole(): String - fun getIsGuideNeeded(): Boolean - - fun getIsChatAccessible(): Boolean - fun setTokens( accessToken: String, refreshToken: String, @@ -18,9 +14,5 @@ interface UserRepository { fun setUserRole(userRole: String) - fun setIsGuideNeeded(isGuideNeeded: Boolean) - - fun setIsChatAccessible(isChatAccessible: Boolean) - fun clearInfo() } diff --git a/gradle/libs.versions.toml b/gradle/libs.versions.toml index 54596282..d09d4de3 100644 --- a/gradle/libs.versions.toml +++ b/gradle/libs.versions.toml @@ -1,24 +1,189 @@ [versions] -agp = "8.3.0" -kotlin = "1.9.0" -coreKtx = "1.13.0" +# Build Tools +agp = "8.1.2" + +# Kotlin +kotlin = "1.8.20" + +# AndroidX +androidx-core-ktx = "1.13.1" +androidx-core-splashscreen = "1.0.1" +androidx-appcompat = "1.7.0" +androidx-activity-ktx = "1.9.0" +androidx-lifecycle-runtime-ktx = "2.8.7" +androidx-navigation-ktx = "2.7.7" +androidx-fragment-ktx = "1.5.7" +androidx-legacy-support = "1.0.0" +androidx-constraint-layout = "2.1.4" + +# Google +material-design = "1.9.0" +app-update = "2.1.0" +google-services = "4.4.2" + +# Hilt +hilt = "2.46.1" + +# Networking +retrofit = "2.9.0" +okhttp = "4.11.0" +retrofit-serialization-converter = "1.0.0" + +# Kotlin Extensions +kotlin-serialization = "1.5.1" +kotlin-datetime = "0.4.0" +kotlin-coroutines = "1.7.1" + +# UI Libraries +coil = "2.4.0" +glide = "4.12.0" +glide-transformations = "4.3.0" +lottie = "6.0.0" +shimmer = "0.5.0" +circular-progress-bar = "3.1.0" +progress-view = "1.1.3" +balloon = "1.4.5" +circle-indicator = "2.1.6" + +# Amplitude +amplitude = "1.17.3" + +# Firebase +firebase-bom = "33.1.2" +crashlytics = "2.9.9" + +# Kakao +kakao = "2.20.3" + +# Testing junit = "4.13.2" -junitVersion = "1.1.5" -espressoCore = "3.5.1" -appcompat = "1.6.1" -material = "1.11.0" -navigationFragmentKtx = "2.7.7" -navigationUiKtx = "2.7.7" +espresso = "3.5.1" +androidx-test = "1.1.5" + +# Timber +timber = "5.0.1" [libraries] -androidx-core-ktx = { group = "androidx.core", name = "core-ktx", version.ref = "coreKtx" } -junit = { group = "junit", name = "junit", version.ref = "junit" } -androidx-junit = { group = "androidx.test.ext", name = "junit", version.ref = "junitVersion" } -androidx-espresso-core = { group = "androidx.test.espresso", name = "espresso-core", version.ref = "espressoCore" } -androidx-appcompat = { group = "androidx.appcompat", name = "appcompat", version.ref = "appcompat" } -material = { group = "com.google.android.material", name = "material", version.ref = "material" } +# Gradle Plugin +android-gradlePlugin = { group = "com.android.tools.build", name = "gradle", version.ref = "agp" } +kotlin-gradlePlugin = { module = "org.jetbrains.kotlin:kotlin-gradle-plugin", version.ref = "kotlin" } -[plugins] -androidApplication = { id = "com.android.application", version.ref = "agp" } -jetbrainsKotlinAndroid = { id = "org.jetbrains.kotlin.android", version.ref = "kotlin" } +# Kotlin +kotlin-stdlib = { group = "org.jetbrains.kotlin", name = "kotlin-stdlib", version.ref = "kotlin" } +kotlin-serialization = { group = "org.jetbrains.kotlinx", name = "kotlinx-serialization-json", version.ref = "kotlin-serialization" } +kotlin-datetime = { group = "org.jetbrains.kotlinx", name = "kotlinx-datetime", version.ref = "kotlin-datetime" } +kotlin-coroutines = { group = "org.jetbrains.kotlinx", name = "kotlinx-coroutines-android", version.ref = "kotlin-coroutines" } + +# AndroidX +androidx-core-ktx = { group = "androidx.core", name = "core-ktx", version.ref = "androidx-core-ktx" } +androidx-core-splashscreen = { group = "androidx.core", name = "core-splashscreen", version.ref = "androidx-core-splashscreen" } +androidx-appcompat = { group = "androidx.appcompat", name = "appcompat", version.ref = "androidx-appcompat" } +androidx-activity-ktx = { group = "androidx.activity", name = "activity-ktx", version.ref = "androidx-activity-ktx" } +androidx-lifecycle-runtime-ktx = { group = "androidx.lifecycle", name = "lifecycle-runtime-ktx", version.ref = "androidx-lifecycle-runtime-ktx" } +androidx-navigation-fragment-ktx = { group = "androidx.navigation", name = "navigation-fragment-ktx", version.ref = "androidx-navigation-ktx" } +androidx-navigation-ui-ktx = { group = "androidx.navigation", name = "navigation-ui-ktx", version.ref = "androidx-navigation-ktx" } +androidx-fragment-ktx = { group = "androidx.fragment", name = "fragment-ktx", version.ref = "androidx-fragment-ktx" } +androidx-legacy-support = { group = "androidx.legacy", name = "legacy-support-v4", version.ref = "androidx-legacy-support" } +androidx-constraint-layout = { group = "androidx.constraintlayout", name = "constraintlayout", version.ref = "androidx-constraint-layout" } + +# MaterialDesign +material-design = { group = "com.google.android.material", name = "material", version.ref = "material-design" } +# Hilt +hilt = { group = "com.google.dagger", name = "hilt-android", version.ref = "hilt" } +hilt-compiler = { group = "com.google.dagger", name = "hilt-compiler", version.ref = "hilt" } + +# Networking +retrofit = { group = "com.squareup.retrofit2", name = "retrofit", version.ref = "retrofit" } +retrofit-serialization-converter = { group = "com.jakewharton.retrofit", name = "retrofit2-kotlinx-serialization-converter", version.ref = "retrofit-serialization-converter" } +okhttp-bom = { group = "com.squareup.okhttp3", name = "okhttp-bom", version.ref = "okhttp" } +okhttp = { group = "com.squareup.okhttp3", name = "okhttp" } +okhttp-logging-interceptor = { group = "com.squareup.okhttp3", name = "logging-interceptor" } + +# UI Libraries +coil = { group = "io.coil-kt", name = "coil", version.ref = "coil" } +glide = { group = "com.github.bumptech.glide", name = "glide", version.ref = "glide" } +glide-transformations = { group = "jp.wasabeef", name = "glide-transformations", version.ref = "glide-transformations" } +lottie = { group = "com.airbnb.android", name = "lottie", version.ref = "lottie" } +shimmer = { group = "com.facebook.shimmer", name = "shimmer", version.ref = "shimmer" } +circular-progress-bar = { group = "com.mikhaellopez", name = "circularprogressbar", version.ref = "circular-progress-bar" } +progress-view = { group = "com.github.skydoves", name = "progressview", version.ref = "progress-view" } +balloon = { group = "com.github.skydoves", name = "balloon", version.ref = "balloon" } +circle-indicator = { group = "me.relex", name = "circleindicator", version.ref = "circle-indicator" } + +# Analytics +amplitude = { group = "com.amplitude", name = "analytics-android", version.ref = "amplitude" } + +# Firebase +firebase-bom = { group = "com.google.firebase", name = "firebase-bom", version.ref = "firebase-bom" } +firebase-crashlytics = { group = "com.google.firebase", name = "firebase-crashlytics-ktx" } +firebase-analytics = { group = "com.google.firebase", name = "firebase-analytics-ktx" } +firebase-messaging = { group = "com.google.firebase", name = "firebase-messaging-ktx" } + +# Kakao +kakao = { group = "com.kakao.sdk", name = "v2-user", version.ref = "kakao" } + +# Testing +j-unit = { group = "junit", name = "junit", version.ref = "junit" } +j-unit-androidx-test = { group = "androidx.test.ext", name = "junit", version.ref = "androidx-test" } +espresso-core = { group = "androidx.test.espresso", name = "espresso-core", version.ref = "espresso" } + +# App Update +app-update = { group = "com.google.android.play", name = "app-update-ktx", version.ref = "app-update" } + +# Timber +timber = { group = "com.jakewharton.timber", name = "timber", version.ref = "timber" } + +[bundles] +kotlin = [ + "kotlin-stdlib", + "kotlin-serialization", + "kotlin-datetime", + "kotlin-coroutines", +] + +androidx = [ + "androidx-core-ktx", + "androidx-core-splashscreen", + "androidx-appcompat", + "androidx-activity-ktx", + "androidx-lifecycle-runtime-ktx", + "androidx-navigation-fragment-ktx", + "androidx-navigation-ui-ktx", + "androidx-fragment-ktx", + "androidx-legacy-support", + "androidx-constraint-layout" +] + +networking = [ + "retrofit", + "retrofit-serialization-converter", + "okhttp", + "okhttp-logging-interceptor" +] + +ui = [ + "coil", + "glide", + "glide-transformations", + "lottie", + "shimmer", + "circular-progress-bar", + "progress-view", + "balloon", + "circle-indicator" +] + +firebase = [ + "firebase-crashlytics", + "firebase-analytics", + "firebase-messaging" +] + +[plugins] +android-application = { id = "com.android.application", version.ref = "agp" } +kotlin-android = { id = "org.jetbrains.kotlin.android", version.ref = "kotlin" } +kotlin-serialization = { id = "org.jetbrains.kotlin.plugin.serialization", version.ref = "kotlin" } +hilt = { id = "com.google.dagger.hilt.android", version.ref = "hilt" } +google-services = { id = "com.google.gms.google-services", version.ref = "google-services" } +google-crashlytics = { id = "com.google.firebase.crashlytics", version.ref = "crashlytics" } diff --git a/gradle/wrapper/gradle-wrapper.properties b/gradle/wrapper/gradle-wrapper.properties index d7c014f1..43368ec1 100644 --- a/gradle/wrapper/gradle-wrapper.properties +++ b/gradle/wrapper/gradle-wrapper.properties @@ -1,6 +1,6 @@ #Wed May 01 02:58:14 KST 2024 distributionBase=GRADLE_USER_HOME distributionPath=wrapper/dists -distributionUrl=https\://services.gradle.org/distributions/gradle-8.4-bin.zip +distributionUrl=https\://services.gradle.org/distributions/gradle-8.0-bin.zip zipStoreBase=GRADLE_USER_HOME zipStorePath=wrapper/dists diff --git a/presentation/build.gradle.kts b/presentation/build.gradle.kts index 25ab42f3..29284339 100644 --- a/presentation/build.gradle.kts +++ b/presentation/build.gradle.kts @@ -1,102 +1,30 @@ plugins { - id("com.android.library") - kotlin("android") - kotlin("kapt") - id("kotlin-parcelize") - id("dagger.hilt.android.plugin") + id("kr.genti.androidLibrary") + id("kr.genti.version") } android { namespace = "kr.genti.presentation" - compileSdk = Constants.compileSdk defaultConfig { - minSdk = Constants.minSdk - - testInstrumentationRunner = "androidx.test.runner.AndroidJUnitRunner" - consumerProguardFiles("consumer-rules.pro") - - buildConfigField("String", "VERSION_NAME", "\"${Constants.versionName}\"") - buildConfigField("String", "VERSION_CODE", "\"${Constants.versionCode}\"") - } - - compileOptions { - sourceCompatibility = Versions.javaVersion - targetCompatibility = Versions.javaVersion - } - - kotlinOptions { - jvmTarget = Versions.jvmVersion - } - - buildFeatures { - buildConfig = true - dataBinding = true - viewBinding = true + buildConfigField("String", "VERSION_NAME", "\"${extra["versionName"]}\"") + buildConfigField("String", "VERSION_CODE", "\"${extra["versionCode"]}\"") } } dependencies { - implementation(project(":core")) - implementation(project(":domain")) + implementation(projects.core) + implementation(projects.domain) - KotlinDependencies.run { - implementation(kotlin) - implementation(coroutines) - implementation(jsonSerialization) - implementation(dateTime) - } + implementation(platform(libs.okhttp.bom)) + implementation(libs.bundles.networking) - AndroidXDependencies.run { - implementation(coreKtx) - implementation(appCompat) - implementation(constraintLayout) - implementation(fragment) - implementation(navigationFragment) - implementation(navigationUi) - implementation(startup) - implementation(legacy) - implementation(security) - implementation(lifeCycleKtx) - implementation(lifecycleJava8) - implementation(splashScreen) - implementation(hilt) - implementation(appUpdate) - } + implementation(platform(libs.firebase.bom)) + implementation(libs.bundles.firebase) - KaptDependencies.run { - kapt(hiltCompiler) - } - - MaterialDesignDependencies.run { - implementation(materialDesign) - } - - TestDependencies.run { - testImplementation(jUnit) - androidTestImplementation(androidTest) - androidTestImplementation(espresso) - } - - ThirdPartyDependencies.run { - implementation(coil) - implementation(timber) - implementation(amplitude) - implementation(progressView) - implementation(balloon) - implementation(lottie) - implementation(circularProgressBar) - implementation(circleIndicator) - } - - FirebaseDependencies.run { - implementation(platform(firebaseBom)) - implementation(messaging) - implementation(crashlytics) - implementation(analytics) - } - - KakaoDependencies.run { - implementation(user) - } + implementation(libs.bundles.androidx) + implementation(libs.bundles.ui) + implementation(libs.kakao) + implementation(libs.app.update) + implementation(libs.amplitude) } diff --git a/presentation/src/main/java/kr/genti/presentation/auth/login/LoginActivity.kt b/presentation/src/main/java/kr/genti/presentation/auth/login/LoginActivity.kt index 0d79d5cd..08494f9e 100644 --- a/presentation/src/main/java/kr/genti/presentation/auth/login/LoginActivity.kt +++ b/presentation/src/main/java/kr/genti/presentation/auth/login/LoginActivity.kt @@ -1,14 +1,12 @@ package kr.genti.presentation.auth.login +import android.animation.ObjectAnimator import android.app.Activity import android.app.ActivityOptions import android.content.Intent +import android.graphics.Matrix import android.os.Bundle import androidx.activity.viewModels -import androidx.core.view.ViewCompat -import androidx.core.view.WindowCompat -import androidx.core.view.WindowInsetsCompat -import androidx.core.view.updatePadding import androidx.lifecycle.flowWithLifecycle import androidx.lifecycle.lifecycleScope import dagger.hilt.android.AndroidEntryPoint @@ -16,7 +14,6 @@ import kotlinx.coroutines.flow.distinctUntilChanged import kotlinx.coroutines.flow.launchIn import kotlinx.coroutines.flow.onEach import kr.genti.core.base.BaseActivity -import kr.genti.core.extension.colorOf import kr.genti.core.extension.initOnBackPressedListener import kr.genti.core.extension.setOnSingleClickListener import kr.genti.core.extension.stringOf @@ -37,8 +34,7 @@ class LoginActivity : BaseActivity(R.layout.activity_login initLoginBtnListener() initOnBackPressedListener(binding.root) - setStatusBarTransparent() - setNavigationBarGreen() + setBackgroundAnimation() observeAppLoginAvailable() observeGetDeviceTokenResult() observeChangeTokenState() @@ -50,16 +46,29 @@ class LoginActivity : BaseActivity(R.layout.activity_login } } - private fun setStatusBarTransparent() { - WindowCompat.setDecorFitsSystemWindows(window, false) - ViewCompat.setOnApplyWindowInsetsListener(binding.root) { v, insets -> - v.updatePadding(bottom = insets.getInsets(WindowInsetsCompat.Type.navigationBars()).bottom) - insets - } - } + private fun setBackgroundAnimation() { + val imageView = binding.ivLoginBackground + imageView.post { + val matrix = Matrix() + + val scale = imageView.height.toFloat() / imageView.drawable.intrinsicHeight.toFloat() + matrix.setScale(scale, scale) + val scaledWidth = imageView.drawable.intrinsicWidth.toFloat() * scale - private fun setNavigationBarGreen() { - this.window.navigationBarColor = colorOf(R.color.genti_green) + matrix.postTranslate(0f, 0f) + imageView.imageMatrix = matrix + + val animator = ObjectAnimator.ofFloat(0f, imageView.width - scaledWidth).apply { + duration = 8000 + addUpdateListener { animation -> + val translateX = animation.animatedValue as Float + matrix.setScale(scale, scale) + matrix.postTranslate(translateX, 0f) + imageView.imageMatrix = matrix + } + } + animator.start() + } } private fun observeAppLoginAvailable() { diff --git a/presentation/src/main/java/kr/genti/presentation/auth/onboarding/OnboardingActivity.kt b/presentation/src/main/java/kr/genti/presentation/auth/onboarding/OnboardingActivity.kt index f1cebf9b..90d9ee93 100644 --- a/presentation/src/main/java/kr/genti/presentation/auth/onboarding/OnboardingActivity.kt +++ b/presentation/src/main/java/kr/genti/presentation/auth/onboarding/OnboardingActivity.kt @@ -1,13 +1,12 @@ package kr.genti.presentation.auth.onboarding -import android.animation.ValueAnimator +import android.animation.ObjectAnimator import android.content.Intent import android.os.Bundle import androidx.core.view.isVisible -import androidx.core.view.updateLayoutParams import dagger.hilt.android.AndroidEntryPoint import kr.genti.core.base.BaseActivity -import kr.genti.core.extension.dpToPx +import kr.genti.core.extension.initOnBackPressedListener import kr.genti.core.extension.setOnSingleClickListener import kr.genti.presentation.R import kr.genti.presentation.databinding.ActivityOnboardingBinding @@ -20,12 +19,13 @@ import kr.genti.presentation.util.AmplitudeManager.PROPERTY_PAGE @AndroidEntryPoint class OnboardingActivity : BaseActivity(R.layout.activity_onboarding) { private var _onboardingAdapter: OnboardingAdapter? = null - val onboardingAdapter + private val onboardingAdapter get() = requireNotNull(_onboardingAdapter) { getString(R.string.adapter_not_initialized_error_msg) } override fun onCreate(savedInstanceState: Bundle?) { super.onCreate(savedInstanceState) + initOnBackPressedListener(binding.root) initViewPager() initNextBtnListener() initFinishBtnListener() @@ -47,26 +47,17 @@ class OnboardingActivity : BaseActivity(R.layout.acti mapOf(PROPERTY_BTN to "next"), ) with(binding) { - btnNext.setOnSingleClickListener { - vpOnboarding.currentItem = 1 - btnNext.isVisible = false - btnFinish.isVisible = true - tvOnboardingTitle.isVisible = false - btnExit.isVisible = false - startLogoAnimation() - } - } - } - - private fun startLogoAnimation() { - ValueAnimator.ofInt(60.dpToPx(this), 150.dpToPx(this)).apply { - duration = 300 - addUpdateListener { animator -> - binding.ivOnboardingLogo.updateLayoutParams { - width = animator.animatedValue as Int + btnNext.setOnClickListener { + vpOnboarding.currentItem += 1 + if (vpOnboarding.currentItem == 2) { + btnNext.isVisible = false + btnFinish.isVisible = true + ObjectAnimator.ofFloat(ivOnboardingThird, "alpha", 0f, 1f).apply { + duration = 1000 + start() + } } } - start() } } diff --git a/presentation/src/main/java/kr/genti/presentation/auth/onboarding/OnboardingAdapter.kt b/presentation/src/main/java/kr/genti/presentation/auth/onboarding/OnboardingAdapter.kt index 99d4a12b..ed8dbf8a 100644 --- a/presentation/src/main/java/kr/genti/presentation/auth/onboarding/OnboardingAdapter.kt +++ b/presentation/src/main/java/kr/genti/presentation/auth/onboarding/OnboardingAdapter.kt @@ -25,6 +25,6 @@ class OnboardingAdapter : RecyclerView.Adapter() { override fun getItemCount(): Int = TOTAL_VIEW_COUNT companion object { - const val TOTAL_VIEW_COUNT = 2 + const val TOTAL_VIEW_COUNT = 3 } } diff --git a/presentation/src/main/java/kr/genti/presentation/auth/onboarding/OnboardingViewHolder.kt b/presentation/src/main/java/kr/genti/presentation/auth/onboarding/OnboardingViewHolder.kt index fd66f379..8893b807 100644 --- a/presentation/src/main/java/kr/genti/presentation/auth/onboarding/OnboardingViewHolder.kt +++ b/presentation/src/main/java/kr/genti/presentation/auth/onboarding/OnboardingViewHolder.kt @@ -1,7 +1,9 @@ package kr.genti.presentation.auth.onboarding +import androidx.appcompat.content.res.AppCompatResources import androidx.core.view.isVisible import androidx.recyclerview.widget.RecyclerView +import kr.genti.presentation.R import kr.genti.presentation.databinding.ItemOnboardingBinding class OnboardingViewHolder( @@ -9,8 +11,17 @@ class OnboardingViewHolder( ) : RecyclerView.ViewHolder(binding.root) { fun onBind(position: Int) { with(binding) { - layoutOnboardingFirst.isVisible = position == 0 - layoutOnboardingSecond.isVisible = position != 0 + ivOnboardingFirst.isVisible = position == 0 + ivOnboardingSecond.isVisible = position == 1 + if (position == 1) { + tvOnboardingSubtitle.text = "얼굴 사진 3장만 골라요." + tvOnboardingBody.text = "사진 생성에 이용할 본인 사진 3장을 골라주세요." + } + if (position == 2) { + tvOnboardingSubtitle.text = "나만의 특별한 사진 완성!" + tvOnboardingBody.text = "지금 젠티하러 가볼까요?" + root.background = AppCompatResources.getDrawable(root.context, R.color.transparent) + } } } } diff --git a/presentation/src/main/java/kr/genti/presentation/auth/signup/SignupActivity.kt b/presentation/src/main/java/kr/genti/presentation/auth/signup/SignupActivity.kt index 7406fa78..d2615b08 100644 --- a/presentation/src/main/java/kr/genti/presentation/auth/signup/SignupActivity.kt +++ b/presentation/src/main/java/kr/genti/presentation/auth/signup/SignupActivity.kt @@ -3,10 +3,6 @@ package kr.genti.presentation.auth.signup import android.content.Intent import android.os.Bundle import androidx.activity.viewModels -import androidx.core.view.ViewCompat -import androidx.core.view.WindowCompat -import androidx.core.view.WindowInsetsCompat -import androidx.core.view.updatePadding import androidx.lifecycle.flowWithLifecycle import androidx.lifecycle.lifecycleScope import dagger.hilt.android.AndroidEntryPoint @@ -14,7 +10,7 @@ import kotlinx.coroutines.flow.distinctUntilChanged import kotlinx.coroutines.flow.launchIn import kotlinx.coroutines.flow.onEach import kr.genti.core.base.BaseActivity -import kr.genti.core.extension.colorOf +import kr.genti.core.extension.hideKeyboard import kr.genti.core.extension.initOnBackPressedListener import kr.genti.core.extension.setOnSingleClickListener import kr.genti.core.extension.stringOf @@ -25,7 +21,6 @@ import kr.genti.presentation.R import kr.genti.presentation.auth.onboarding.OnboardingActivity import kr.genti.presentation.databinding.ActivitySignupBinding import kr.genti.presentation.util.AmplitudeManager -import java.util.Calendar @AndroidEntryPoint class SignupActivity : BaseActivity(R.layout.activity_signup) { @@ -36,10 +31,8 @@ class SignupActivity : BaseActivity(R.layout.activity_sig initView() initSubmitBtnListener() - setYearPicker() - setStatusBarTransparent() - setNavigationBarGreen() observePostSignupState() + observeYearInputState() } private fun initView() { @@ -54,31 +47,6 @@ class SignupActivity : BaseActivity(R.layout.activity_sig } } - private fun setYearPicker() { - binding.npSignupBirth.apply { - val currentYear = Calendar.getInstance()[Calendar.YEAR] - maxValue = currentYear - minValue = 1900 - value = currentYear - viewModel.selectBirthYear(currentYear) - setOnValueChangedListener { _, _, newVal -> - viewModel.selectBirthYear(newVal) - } - } - } - - private fun setStatusBarTransparent() { - WindowCompat.setDecorFitsSystemWindows(window, false) - ViewCompat.setOnApplyWindowInsetsListener(binding.root) { v, insets -> - v.updatePadding(bottom = insets.getInsets(WindowInsetsCompat.Type.navigationBars()).bottom) - insets - } - } - - private fun setNavigationBarGreen() { - this.window.navigationBarColor = colorOf(R.color.green_5) - } - private fun observePostSignupState() { viewModel.postSignupState .flowWithLifecycle(lifecycle) @@ -99,6 +67,15 @@ class SignupActivity : BaseActivity(R.layout.activity_sig }.launchIn(lifecycleScope) } + private fun observeYearInputState() { + viewModel.isYearAllSelected.flowWithLifecycle(lifecycle).distinctUntilChanged() + .onEach { isAllSelected -> + if (isAllSelected) { + hideKeyboard(binding.root) + } + }.launchIn(lifecycleScope) + } + private fun setAmplitudeUserProperty(state: UiState.Success) { AmplitudeManager.apply { trackEvent("complete_infoget") diff --git a/presentation/src/main/java/kr/genti/presentation/auth/signup/SignupViewModel.kt b/presentation/src/main/java/kr/genti/presentation/auth/signup/SignupViewModel.kt index 1083c479..364c0087 100644 --- a/presentation/src/main/java/kr/genti/presentation/auth/signup/SignupViewModel.kt +++ b/presentation/src/main/java/kr/genti/presentation/auth/signup/SignupViewModel.kt @@ -17,62 +17,60 @@ import javax.inject.Inject @HiltViewModel class SignupViewModel - @Inject - constructor( - private val infoRepository: InfoRepository, - private val userRepository: UserRepository, - ) : ViewModel() { - val selectedGender = MutableLiveData(Gender.NONE) - val isGenderSelected = MutableLiveData(false) +@Inject +constructor( + private val infoRepository: InfoRepository, + private val userRepository: UserRepository, +) : ViewModel() { + val selectedGender = MutableLiveData(Gender.NONE) + private val isGenderSelected = MutableLiveData(false) - val selectedYear = MutableLiveData() - val selectedYearText = MutableLiveData() - val isYearSelected = MutableLiveData(false) + val selectedYear = MutableLiveData() + val isYearSelected = MutableLiveData(false) - val isAllSelected = MutableLiveData(false) + private val _isYearAllSelected = MutableStateFlow(false) + val isYearAllSelected: StateFlow = _isYearAllSelected - private val _postSignupState = MutableStateFlow>(UiState.Empty) - val postSignupState: StateFlow> = _postSignupState + val isAllSelected = MutableLiveData(false) - fun selectGender(gender: Gender) { - selectedGender.value = gender - isGenderSelected.value = true - checkAllSelected() - } + private val _postSignupState = MutableStateFlow>(UiState.Empty) + val postSignupState: StateFlow> = _postSignupState - fun selectBirthYear(year: Int) { - selectedYear.value = year - if (isYearSelected.value == true) selectedYearText.value = year.toString() + "년" - } + fun selectGender(gender: Gender) { + selectedGender.value = gender + isGenderSelected.value = true + checkAllSelected() + } - fun showYearPicker() { - isYearSelected.value = true - selectedYearText.value = selectedYear.value.toString() + "년" - checkAllSelected() - } + fun checkYear() { + isYearSelected.value = selectedYear.value?.isNotEmpty() + _isYearAllSelected.value = selectedYear.value?.length == 4 && + selectedYear.value?.toIntOrNull()?.let { it in 1950..2025 } == true + checkAllSelected() + } - private fun checkAllSelected() { - isAllSelected.value = isGenderSelected.value == true && isYearSelected.value == true - } + private fun checkAllSelected() { + isAllSelected.value = isGenderSelected.value == true && isYearAllSelected.value == true + } - fun postSignupDataToServer() { - _postSignupState.value = UiState.Loading - viewModelScope.launch { - infoRepository.postSignupData( - SignupRequestModel( - selectedYear.value.toString(), - selectedGender.value.toString(), - ), - ).onSuccess { - userRepository.setUserRole(ROLE_USER) - _postSignupState.value = UiState.Success(it) - }.onFailure { - _postSignupState.value = UiState.Failure(it.message.orEmpty()) - } + fun postSignupDataToServer() { + _postSignupState.value = UiState.Loading + viewModelScope.launch { + infoRepository.postSignupData( + SignupRequestModel( + selectedYear.value.toString(), + selectedGender.value.toString(), + ), + ).onSuccess { + userRepository.setUserRole(ROLE_USER) + _postSignupState.value = UiState.Success(it) + }.onFailure { + _postSignupState.value = UiState.Failure(it.message.orEmpty()) } } + } - companion object { - private const val ROLE_USER = "USER" - } + companion object { + private const val ROLE_USER = "USER" } +} diff --git a/presentation/src/main/java/kr/genti/presentation/auth/splash/SplashActivity.kt b/presentation/src/main/java/kr/genti/presentation/auth/splash/SplashActivity.kt index ebb23508..d47529e2 100644 --- a/presentation/src/main/java/kr/genti/presentation/auth/splash/SplashActivity.kt +++ b/presentation/src/main/java/kr/genti/presentation/auth/splash/SplashActivity.kt @@ -20,7 +20,6 @@ import com.google.android.play.core.appupdate.AppUpdateOptions import com.google.android.play.core.install.model.AppUpdateType.IMMEDIATE import com.google.android.play.core.install.model.UpdateAvailability.UPDATE_AVAILABLE import dagger.hilt.android.AndroidEntryPoint -import kotlinx.coroutines.flow.distinctUntilChanged import kotlinx.coroutines.flow.launchIn import kotlinx.coroutines.flow.onEach import kr.genti.core.base.BaseActivity @@ -47,11 +46,16 @@ class SplashActivity : BaseActivity(R.layout.activity_spl super.onCreate(savedInstanceState) setSystemWindowsTransparent() - checkAppUpdateAvailable() observeAutoLoginState() observeReissueTokenResult() } + override fun onResume() { + super.onResume() + + checkAppUpdateAvailable() + } + private fun setSystemWindowsTransparent() { WindowCompat.setDecorFitsSystemWindows(window, false) WindowInsetsControllerCompat(window, window.decorView).apply { @@ -79,7 +83,7 @@ class SplashActivity : BaseActivity(R.layout.activity_spl private fun isAppUpdateNeeded(appUpdateInfo: AppUpdateInfo) = appUpdateInfo.updateAvailability() == UPDATE_AVAILABLE && - appUpdateInfo.isUpdateTypeAllowed(IMMEDIATE) + appUpdateInfo.isUpdateTypeAllowed(IMMEDIATE) private fun requestUpdate(appUpdateInfo: AppUpdateInfo) { runCatching { @@ -92,25 +96,23 @@ class SplashActivity : BaseActivity(R.layout.activity_spl } private fun observeAutoLoginState() { - viewModel.isAutoLogined.flowWithLifecycle(lifecycle).distinctUntilChanged() - .onEach { isAutoLogined -> - if (isAutoLogined) { - viewModel.postToReissueToken() - } else { - navigateTo() - } - }.launchIn(lifecycleScope) + viewModel.isAutoLogined.flowWithLifecycle(lifecycle).onEach { isAutoLogined -> + if (isAutoLogined) { + viewModel.postToReissueToken() + } else { + navigateTo() + } + }.launchIn(lifecycleScope) } private fun observeReissueTokenResult() { - viewModel.reissueTokenResult.flowWithLifecycle(lifecycle).distinctUntilChanged() - .onEach { isSuccess -> - if (isSuccess) { - navigateTo() - } else { - navigateTo() - } - }.launchIn(lifecycleScope) + viewModel.reissueTokenResult.flowWithLifecycle(lifecycle).onEach { isSuccess -> + if (isSuccess) { + navigateTo() + } else { + navigateTo() + } + }.launchIn(lifecycleScope) } private inline fun navigateTo() { diff --git a/presentation/src/main/java/kr/genti/presentation/create/CreateActivity.kt b/presentation/src/main/java/kr/genti/presentation/create/CreateActivity.kt new file mode 100644 index 00000000..8d53b305 --- /dev/null +++ b/presentation/src/main/java/kr/genti/presentation/create/CreateActivity.kt @@ -0,0 +1,120 @@ +package kr.genti.presentation.create + +import android.animation.ObjectAnimator +import android.content.Context +import android.content.Intent +import android.os.Bundle +import android.view.animation.LinearInterpolator +import androidx.activity.OnBackPressedCallback +import androidx.activity.viewModels +import androidx.core.view.isVisible +import androidx.lifecycle.flowWithLifecycle +import androidx.lifecycle.lifecycleScope +import androidx.navigation.NavController +import androidx.navigation.fragment.NavHostFragment +import dagger.hilt.android.AndroidEntryPoint +import kotlinx.coroutines.flow.launchIn +import kotlinx.coroutines.flow.onEach +import kr.genti.core.base.BaseActivity +import kr.genti.core.state.UiState +import kr.genti.presentation.R +import kr.genti.presentation.databinding.ActivityCreateBinding +import kr.genti.presentation.util.AmplitudeManager +import kr.genti.presentation.util.AmplitudeManager.EVENT_CLICK_BTN +import kr.genti.presentation.util.AmplitudeManager.PROPERTY_BTN +import kr.genti.presentation.util.AmplitudeManager.PROPERTY_PAGE + +@AndroidEntryPoint +class CreateActivity() : BaseActivity(R.layout.activity_create) { + private val viewModel by viewModels() + private lateinit var navController: NavController + + override fun onCreate(savedInstanceState: Bundle?) { + super.onCreate(savedInstanceState) + + initView() + initBackBtnListener() + setParentPicWithIntent() + observeProgressBar() + observeGeneratingState() + } + + private fun initView() { + navController = binding.fcvCreate.getFragment().navController + } + + private fun initBackBtnListener() { + binding.btnBack.setOnClickListener { + when (navController.currentDestination?.id) { + R.id.numberFragment -> finish() + R.id.defineFragment -> { + if (viewModel.isCreatingParentPic) { + navigateBackFragment("create1", -33) + } else { + finish() + } + } + + R.id.poseFragment -> navigateBackFragment("create2", -33) + R.id.selfieFragment -> navigateBackFragment("create3", -34) + } + } + } + + private fun navigateBackFragment(tag: String, amount: Int) { + AmplitudeManager.trackEvent( + EVENT_CLICK_BTN, + mapOf(PROPERTY_PAGE to tag), + mapOf(PROPERTY_BTN to "back"), + ) + navController.popBackStack() + viewModel.modCurrentPercent(amount) + } + + private fun setParentPicWithIntent() { + viewModel.isCreatingParentPic = intent.getBooleanExtra(EXTRA_IS_CREATING_PARENT_PIC, false) + if (viewModel.isCreatingParentPic) { + binding.tvCreateTitle.text = getString(R.string.create_parent_tv_title) + } else { + navController.navigate(R.id.action_number_to_define_without_anim) + viewModel.modCurrentPercent(33) + } + } + + private fun observeProgressBar() { + viewModel.currentPercent.flowWithLifecycle(lifecycle).onEach { percent -> + ObjectAnimator.ofInt( + binding.progressbarCreate, + PROPERTY_PROGRESS, + binding.progressbarCreate.progress, + percent, + ).apply { + duration = 300 + interpolator = LinearInterpolator() + start() + } + binding.tvCreatePhase.text = getString(R.string.create_phase_text, percent / 33) + }.launchIn(lifecycleScope) + } + + private fun observeGeneratingState() { + viewModel.totalGeneratingState.flowWithLifecycle(lifecycle).onEach { state -> + binding.layoutLoading.isVisible = state == UiState.Loading + }.launchIn(lifecycleScope) + } + + companion object { + const val PROPERTY_PROGRESS = "progress" + + private const val EXTRA_IS_CREATING_PARENT_PIC = "EXTRA_IS_CREATING_PARENT_PIC" + + @JvmStatic + fun createIntent( + context: Context, + isCreatingParentPic: Boolean, + ): Intent = + Intent(context, CreateActivity::class.java).apply { + putExtra(EXTRA_IS_CREATING_PARENT_PIC, isCreatingParentPic) + } + } +} diff --git a/presentation/src/main/java/kr/genti/presentation/create/CreateViewModel.kt b/presentation/src/main/java/kr/genti/presentation/create/CreateViewModel.kt new file mode 100644 index 00000000..d0bb1674 --- /dev/null +++ b/presentation/src/main/java/kr/genti/presentation/create/CreateViewModel.kt @@ -0,0 +1,237 @@ +package kr.genti.presentation.create + +import androidx.lifecycle.MutableLiveData +import androidx.lifecycle.ViewModel +import androidx.lifecycle.viewModelScope +import dagger.hilt.android.lifecycle.HiltViewModel +import kotlinx.coroutines.async +import kotlinx.coroutines.awaitAll +import kotlinx.coroutines.coroutineScope +import kotlinx.coroutines.flow.MutableStateFlow +import kotlinx.coroutines.flow.StateFlow +import kotlinx.coroutines.launch +import kr.genti.core.state.UiState +import kr.genti.domain.entity.request.CreateRequestModel +import kr.genti.domain.entity.request.CreateTwoRequestModel +import kr.genti.domain.entity.request.KeyRequestModel +import kr.genti.domain.entity.request.S3RequestModel +import kr.genti.domain.entity.response.ImageFileModel +import kr.genti.domain.entity.response.PromptExampleModel +import kr.genti.domain.entity.response.S3PresignedUrlModel +import kr.genti.domain.enums.FileType +import kr.genti.domain.enums.PictureNumber +import kr.genti.domain.enums.PictureRatio +import kr.genti.domain.repository.CreateRepository +import kr.genti.domain.repository.UploadRepository +import javax.inject.Inject + +@HiltViewModel +class CreateViewModel +@Inject +constructor( + private val createRepository: CreateRepository, + private val uploadRepository: UploadRepository, +) : ViewModel() { + var isCreatingParentPic = false + + val prompt = MutableLiveData() + val isWritten = MutableLiveData(false) + + val selectedNumber = MutableLiveData() + val isNumberSelected = MutableLiveData(false) + + val selectedRatio = MutableLiveData() + val isRatioSelected = MutableLiveData(false) + + var currentAddingList = 0 + var imageList = listOf() + var firstImageList = listOf() + var secondImageList = listOf() + + var isFirstListCompleted = MutableLiveData(false) + var isSecondListCompleted = MutableLiveData(false) + var isCompleted = MutableLiveData(false) + + private val _currentPercent = MutableStateFlow(0) + val currentPercent: StateFlow = _currentPercent + + private val _getExampleState = + MutableStateFlow>>(UiState.Empty) + val getExampleState: StateFlow>> = _getExampleState + + private val _totalGeneratingState = MutableStateFlow>(UiState.Empty) + val totalGeneratingState: StateFlow> = _totalGeneratingState + + private var imageS3KeyList = listOf() + private var firstImageS3KeyList = listOf() + private var secondImageS3KeyList = listOf() + + init { + getExamplePrompt() + } + + fun modCurrentPercent(amount: Int) { + _currentPercent.value += amount + } + + fun checkWritten() { + isWritten.value = prompt.value?.isNotEmpty() + } + + fun selectNumber(item: PictureNumber) { + selectedNumber.value = item + isNumberSelected.value = selectedNumber.value != null + } + + fun selectRatio(item: PictureRatio) { + selectedRatio.value = item + isRatioSelected.value = selectedRatio.value != null + } + + fun updateCompletionState(uriSize: Int) { + when (currentAddingList) { + 0 -> isCompleted.value = uriSize == 3 + 1 -> { + isFirstListCompleted.value = uriSize == 3 + isCompleted.value = + isFirstListCompleted.value == true && isSecondListCompleted.value == true + } + + 2 -> { + isSecondListCompleted.value = uriSize == 3 + isCompleted.value = + isFirstListCompleted.value == true && isSecondListCompleted.value == true + } + } + } + + fun resetTotalGeneratingState() { + _totalGeneratingState.value = UiState.Empty + } + + private fun getExamplePrompt() { + _getExampleState.value = UiState.Loading + viewModelScope.launch { + runCatching { + createRepository.getPromptExample() + }.onSuccess { + _getExampleState.value = UiState.Success(it.getOrThrow()) + }.onFailure { + _getExampleState.value = UiState.Failure(it.message.toString()) + } + } + } + + fun startSendingImages() { + _totalGeneratingState.value = UiState.Loading + if (selectedNumber.value != PictureNumber.TWO) { + saveThreeImages() + } else { + saveSixImages() + } + } + + private fun saveThreeImages() { + viewModelScope.launch { + runCatching { + coroutineScope { + val urlModelList = getThreeS3Urls(imageList) + postThreeImage(urlModelList, imageList) + urlModelList.map { KeyRequestModel(it.s3Key) } + } + }.onSuccess { keyList -> + imageS3KeyList = keyList + postThreeToGenerateImage() + }.onFailure { + _totalGeneratingState.value = UiState.Failure(it.message.toString()) + } + } + } + + private fun saveSixImages() { + viewModelScope.launch { + runCatching { + coroutineScope { + val firstResult = async { + val urlModelList = getThreeS3Urls(firstImageList) + postThreeImage(urlModelList, firstImageList) + urlModelList.map { KeyRequestModel(it.s3Key) } + } + val secondResult = async { + val urlModelList = getThreeS3Urls(secondImageList) + postThreeImage(urlModelList, secondImageList) + urlModelList.map { KeyRequestModel(it.s3Key) } + } + firstImageS3KeyList = firstResult.await() + secondImageS3KeyList = secondResult.await() + } + }.onSuccess { + postSixToGenerateImage() + }.onFailure { + _totalGeneratingState.value = UiState.Failure(it.message.toString()) + } + } + } + + private suspend fun getThreeS3Urls(imageList: List): List { + return createRepository.getS3MultiUrl( + imageList.map { image -> + S3RequestModel(FileType.USER_UPLOADED_IMAGE, image.name) + } + ).getOrThrow() + } + + private suspend fun postThreeImage( + urlModelList: List, + imageList: List + ) { + coroutineScope { + urlModelList.mapIndexed { i, urlModel -> + async { + runCatching { + uploadRepository.uploadImage(urlModel.url, imageList[i].url) + }.onFailure { + throw Exception(it.message) + } + } + }.awaitAll() + } + } + + private fun postThreeToGenerateImage() { + viewModelScope.launch { + val request = CreateRequestModel( + prompt.value ?: return@launch, + imageS3KeyList, + selectedRatio.value ?: return@launch, + ) + val result = if (isCreatingParentPic) { + createRepository.postToCreateOne(request) + } else { + createRepository.postToCreate(request) + } + result.onSuccess { + _totalGeneratingState.value = UiState.Success(it) + }.onFailure { + _totalGeneratingState.value = UiState.Failure(it.message.toString()) + } + } + } + + private fun postSixToGenerateImage() { + viewModelScope.launch { + createRepository.postToCreateTwo( + CreateTwoRequestModel( + prompt.value ?: return@launch, + firstImageS3KeyList, + secondImageS3KeyList, + selectedRatio.value ?: return@launch, + ) + ).onSuccess { + _totalGeneratingState.value = UiState.Success(isCreatingParentPic) + }.onFailure { + _totalGeneratingState.value = UiState.Failure(it.message.toString()) + } + } + } +} diff --git a/presentation/src/main/java/kr/genti/presentation/create/DefineAdapter.kt b/presentation/src/main/java/kr/genti/presentation/create/DefineAdapter.kt new file mode 100644 index 00000000..7e0edd78 --- /dev/null +++ b/presentation/src/main/java/kr/genti/presentation/create/DefineAdapter.kt @@ -0,0 +1,37 @@ +package kr.genti.presentation.create + +import android.view.LayoutInflater +import android.view.ViewGroup +import androidx.recyclerview.widget.ListAdapter +import kr.genti.core.util.ItemDiffCallback +import kr.genti.domain.entity.response.PromptExampleModel +import kr.genti.presentation.databinding.ItemDefineExampleBinding + +class DefineAdapter : ListAdapter(diffUtil) { + + override fun onCreateViewHolder( + parent: ViewGroup, + viewType: Int, + ): DefineViewHolder { + val inflater = LayoutInflater.from(parent.context) + val binding = ItemDefineExampleBinding.inflate(inflater, parent, false) + return DefineViewHolder(binding) + } + + override fun onBindViewHolder( + holder: DefineViewHolder, + position: Int, + ) { + val item = getItem(position) ?: return + holder.onBind(item) + } + + companion object { + + private val diffUtil = + ItemDiffCallback( + onItemsTheSame = { old, new -> old.prompt == new.prompt }, + onContentsTheSame = { old, new -> old == new }, + ) + } +} \ No newline at end of file diff --git a/presentation/src/main/java/kr/genti/presentation/create/DefineFragment.kt b/presentation/src/main/java/kr/genti/presentation/create/DefineFragment.kt new file mode 100644 index 00000000..511c5bd1 --- /dev/null +++ b/presentation/src/main/java/kr/genti/presentation/create/DefineFragment.kt @@ -0,0 +1,135 @@ +package kr.genti.presentation.create + +import android.os.Bundle +import android.view.View +import androidx.activity.OnBackPressedCallback +import androidx.fragment.app.activityViewModels +import androidx.lifecycle.flowWithLifecycle +import androidx.lifecycle.lifecycleScope +import androidx.navigation.fragment.findNavController +import androidx.viewpager2.widget.ViewPager2 +import dagger.hilt.android.AndroidEntryPoint +import kotlinx.coroutines.flow.launchIn +import kotlinx.coroutines.flow.onEach +import kr.genti.core.base.BaseFragment +import kr.genti.core.extension.setOnSingleClickListener +import kr.genti.core.extension.setTextWithImage +import kr.genti.core.extension.stringOf +import kr.genti.core.extension.toast +import kr.genti.core.state.UiState +import kr.genti.presentation.R +import kr.genti.presentation.databinding.FragmentDefineBinding +import kr.genti.presentation.util.AmplitudeManager +import kr.genti.presentation.util.AmplitudeManager.EVENT_CLICK_BTN +import kr.genti.presentation.util.AmplitudeManager.PROPERTY_BTN +import kr.genti.presentation.util.AmplitudeManager.PROPERTY_PAGE + +@AndroidEntryPoint +class DefineFragment() : BaseFragment(R.layout.fragment_define) { + private val viewModel by activityViewModels() + + private var _adapter: DefineAdapter? = null + private val adapter + get() = requireNotNull(_adapter) { getString(R.string.adapter_not_initialized_error_msg) } + + override fun onViewCreated( + view: View, + savedInstanceState: Bundle?, + ) { + super.onViewCreated(view, savedInstanceState) + + initView() + initCreateBtnListener() + initBackPressedListener() + initViewPager() + setTextByParent() + observeGetExampleState() + } + + private fun initView() { + binding.vm = viewModel + } + + private fun initCreateBtnListener() { + binding.btnCreateNext.setOnSingleClickListener { + AmplitudeManager.trackEvent( + EVENT_CLICK_BTN, + mapOf(PROPERTY_PAGE to "create1"), + mapOf(PROPERTY_BTN to "next"), + ) + findNavController().navigate(R.id.action_define_to_pose) + viewModel.modCurrentPercent(33) + } + } + + private fun initBackPressedListener() { + requireActivity().onBackPressedDispatcher.addCallback( + viewLifecycleOwner, + object : OnBackPressedCallback(true) { + override fun handleOnBackPressed() { + if (viewModel.isCreatingParentPic) { + viewModel.modCurrentPercent(-33) + findNavController().popBackStack() + } else { + requireActivity().finish() + } + } + }) + } + + private fun initViewPager() { + _adapter = DefineAdapter() + with(binding) { + vpCreateRandom.adapter = adapter + vpCreateRandom.registerOnPageChangeCallback(object : ViewPager2.OnPageChangeCallback() { + override fun onPageSelected(position: Int) { + super.onPageSelected(position) + AmplitudeManager.apply { + trackEvent( + EVENT_CLICK_BTN, + mapOf(PROPERTY_PAGE to "create1"), + mapOf(PROPERTY_BTN to "promptsuggest_refresh"), + ) + plusIntProperties("user_promptsuggest_refresh") + } + } + }) + } + } + + private fun setTextByParent() { + with(binding) { + val (titleRes, subtitle2Res) = if (viewModel.isCreatingParentPic) { + R.string.create_tv_script_title_parent to R.string.create_tv_script_subtitle_2_parent + } else { + R.string.create_tv_script_title to R.string.create_tv_script_subtitle_2 + } + tvCreateScriptTitle.text = stringOf(titleRes) + tvCreateScriptSubtitle1.setTextWithImage( + stringOf(R.string.create_tv_script_subtitle_1), + R.drawable.ic_check, + ) + tvCreateScriptSubtitle2.setTextWithImage( + stringOf(subtitle2Res), + R.drawable.ic_check, + ) + } + } + + private fun observeGetExampleState() { + viewModel.getExampleState + .flowWithLifecycle(lifecycle) + .onEach { state -> + when (state) { + is UiState.Success -> { + adapter.submitList(state.data) + binding.dotIndicator.setViewPager(binding.vpCreateRandom) + } + + is UiState.Failure -> toast(stringOf(R.string.error_msg)) + else -> return@onEach + } + }.launchIn(lifecycleScope) + } + +} diff --git a/presentation/src/main/java/kr/genti/presentation/create/DefineViewHolder.kt b/presentation/src/main/java/kr/genti/presentation/create/DefineViewHolder.kt new file mode 100644 index 00000000..bcbb726d --- /dev/null +++ b/presentation/src/main/java/kr/genti/presentation/create/DefineViewHolder.kt @@ -0,0 +1,18 @@ +package kr.genti.presentation.create + +import androidx.recyclerview.widget.RecyclerView +import coil.load +import kr.genti.core.extension.breakLines +import kr.genti.domain.entity.response.PromptExampleModel +import kr.genti.presentation.databinding.ItemDefineExampleBinding + +class DefineViewHolder( + val binding: ItemDefineExampleBinding, +) : RecyclerView.ViewHolder(binding.root) { + fun onBind(item: PromptExampleModel) { + with(binding) { + ivDefineExample.load(item.url) + tvDefineExample.text = item.prompt.breakLines() + } + } +} diff --git a/presentation/src/main/java/kr/genti/presentation/create/NumberFragment.kt b/presentation/src/main/java/kr/genti/presentation/create/NumberFragment.kt new file mode 100644 index 00000000..a91c6c56 --- /dev/null +++ b/presentation/src/main/java/kr/genti/presentation/create/NumberFragment.kt @@ -0,0 +1,45 @@ +package kr.genti.presentation.create + +import android.os.Bundle +import android.view.View +import androidx.fragment.app.activityViewModels +import androidx.navigation.fragment.findNavController +import kr.genti.core.base.BaseFragment +import kr.genti.core.extension.setOnSingleClickListener +import kr.genti.presentation.R +import kr.genti.presentation.databinding.FragmentNumberBinding +import kr.genti.presentation.util.AmplitudeManager +import kr.genti.presentation.util.AmplitudeManager.EVENT_CLICK_BTN +import kr.genti.presentation.util.AmplitudeManager.PROPERTY_BTN +import kr.genti.presentation.util.AmplitudeManager.PROPERTY_PAGE + +class NumberFragment() : BaseFragment(R.layout.fragment_number) { + private val viewModel by activityViewModels() + + + override fun onViewCreated( + view: View, + savedInstanceState: Bundle?, + ) { + super.onViewCreated(view, savedInstanceState) + + initView() + initNextBtnListener() + } + + private fun initView() { + binding.vm = viewModel + } + + private fun initNextBtnListener() { + binding.btnNumberNext.setOnSingleClickListener { + AmplitudeManager.trackEvent( + EVENT_CLICK_BTN, + mapOf(PROPERTY_PAGE to "create0"), + mapOf(PROPERTY_BTN to "next"), + ) + findNavController().navigate(R.id.action_number_to_define) + viewModel.modCurrentPercent(33) + } + } +} \ No newline at end of file diff --git a/presentation/src/main/java/kr/genti/presentation/main/create/PoseFragment.kt b/presentation/src/main/java/kr/genti/presentation/create/PoseFragment.kt similarity index 62% rename from presentation/src/main/java/kr/genti/presentation/main/create/PoseFragment.kt rename to presentation/src/main/java/kr/genti/presentation/create/PoseFragment.kt index db44b9f7..26edd36e 100644 --- a/presentation/src/main/java/kr/genti/presentation/main/create/PoseFragment.kt +++ b/presentation/src/main/java/kr/genti/presentation/create/PoseFragment.kt @@ -1,4 +1,4 @@ -package kr.genti.presentation.main.create +package kr.genti.presentation.create import android.os.Bundle import android.view.View @@ -19,8 +19,6 @@ import kr.genti.presentation.util.AmplitudeManager.PROPERTY_PAGE class PoseFragment() : BaseFragment(R.layout.fragment_pose) { private val viewModel by activityViewModels() - private var createGuideDialog: CreateGuideDialog? = null - override fun onViewCreated( view: View, savedInstanceState: Bundle?, @@ -30,7 +28,6 @@ class PoseFragment() : BaseFragment(R.layout.fragment_pose) initView() initNextBtnListener() initBackPressedListener() - initGuideIfNeeded() } private fun initView() { @@ -50,38 +47,13 @@ class PoseFragment() : BaseFragment(R.layout.fragment_pose) } private fun initBackPressedListener() { - val onBackPressedCallback = + requireActivity().onBackPressedDispatcher.addCallback( + viewLifecycleOwner, object : OnBackPressedCallback(true) { override fun handleOnBackPressed() { - if (isAdded) { - findNavController().popBackStack() - viewModel.modCurrentPercent(-33) - } else { - requireActivity().onBackPressedDispatcher.onBackPressed() - } + viewModel.modCurrentPercent(-33) + findNavController().popBackStack() } - } - activity?.onBackPressedDispatcher?.addCallback( - requireActivity(), - onBackPressedCallback, - ) - } - - private fun initGuideIfNeeded() { - if (viewModel.getIsGuideNeeded()) { - createGuideDialog = CreateGuideDialog() - createGuideDialog?.show(childFragmentManager, DIALOG_GUIDE) - viewModel.setIsGuideNeeded(false) - } - } - - override fun onDestroyView() { - super.onDestroyView() - - createGuideDialog = null - } - - companion object { - const val DIALOG_GUIDE = "DIALOG_GUIDE" + }) } } diff --git a/presentation/src/main/java/kr/genti/presentation/main/create/SelfieFragment.kt b/presentation/src/main/java/kr/genti/presentation/create/SelfieFragment.kt similarity index 51% rename from presentation/src/main/java/kr/genti/presentation/main/create/SelfieFragment.kt rename to presentation/src/main/java/kr/genti/presentation/create/SelfieFragment.kt index 362ed08b..66905dcf 100644 --- a/presentation/src/main/java/kr/genti/presentation/main/create/SelfieFragment.kt +++ b/presentation/src/main/java/kr/genti/presentation/create/SelfieFragment.kt @@ -1,15 +1,13 @@ -package kr.genti.presentation.main.create +package kr.genti.presentation.create import android.app.Activity.RESULT_CANCELED import android.app.Activity.RESULT_OK import android.content.Intent import android.net.Uri +import android.os.Build import android.os.Bundle -import android.text.SpannableString -import android.text.SpannableStringBuilder -import android.text.Spanned -import android.text.style.BulletSpan import android.view.View +import android.widget.ImageView import androidx.activity.OnBackPressedCallback import androidx.activity.result.ActivityResultLauncher import androidx.activity.result.PickVisualMediaRequest @@ -27,27 +25,26 @@ import kotlinx.coroutines.flow.onEach import kr.genti.core.base.BaseFragment import kr.genti.core.extension.getFileName import kr.genti.core.extension.setOnSingleClickListener +import kr.genti.core.extension.setTextWithImage import kr.genti.core.extension.stringOf import kr.genti.core.extension.toast import kr.genti.core.state.UiState import kr.genti.domain.entity.response.ImageFileModel +import kr.genti.domain.enums.PictureNumber import kr.genti.presentation.R import kr.genti.presentation.databinding.FragmentSelfieBinding import kr.genti.presentation.generate.waiting.WaitingActivity -import kr.genti.presentation.main.MainActivity -import kr.genti.presentation.main.feed.FeedFragment import kr.genti.presentation.util.AmplitudeManager import kr.genti.presentation.util.AmplitudeManager.EVENT_CLICK_BTN import kr.genti.presentation.util.AmplitudeManager.PROPERTY_BTN import kr.genti.presentation.util.AmplitudeManager.PROPERTY_PAGE -import kotlin.math.max @AndroidEntryPoint class SelfieFragment : BaseFragment(R.layout.fragment_selfie) { private val viewModel by activityViewModels() + private lateinit var photoPickerResult: ActivityResultLauncher private lateinit var galleryPickerResult: ActivityResultLauncher - private lateinit var waitingResult: ActivityResultLauncher override fun onViewCreated( view: View, @@ -59,11 +56,10 @@ class SelfieFragment : BaseFragment(R.layout.fragment_sel initBackPressedListener() initAddImageBtnListener() initRequestCreateBtnListener() + setGuideTextByParent() + setLayoutByParent() setGalleryImageWithPhotoPicker() setGalleryImageWithGalleryPicker() - setBulletPointList() - setGuideListBlur() - initWaitingResult() observeGeneratingState() } @@ -78,19 +74,14 @@ class SelfieFragment : BaseFragment(R.layout.fragment_sel } private fun initBackPressedListener() { - val onBackPressedCallback = + requireActivity().onBackPressedDispatcher.addCallback( + viewLifecycleOwner, object : OnBackPressedCallback(true) { override fun handleOnBackPressed() { - if (isAdded) { - findNavController().popBackStack() - viewModel.modCurrentPercent(-34) - } + viewModel.modCurrentPercent(-34) + findNavController().popBackStack() } - } - activity?.onBackPressedDispatcher?.addCallback( - requireActivity(), - onBackPressedCallback, - ) + }) } private fun initAddImageBtnListener() { @@ -101,21 +92,15 @@ class SelfieFragment : BaseFragment(R.layout.fragment_sel mapOf(PROPERTY_PAGE to "create3"), mapOf(PROPERTY_BTN to "selectpic"), ) - checkAndGetImages() - } - layoutAddedImage.setOnSingleClickListener { - AmplitudeManager.trackEvent( - EVENT_CLICK_BTN, - mapOf(PROPERTY_PAGE to "create3"), - mapOf(PROPERTY_BTN to "reselectpic"), - ) - checkAndGetImages() + checkAndGetImages(0) } + btnSelfieAddFirst.setOnSingleClickListener { checkAndGetImages(1) } + btnSelfieAddSecond.setOnSingleClickListener { checkAndGetImages(2) } } } private fun initRequestCreateBtnListener() { - binding.btnSelfieNext.setOnSingleClickListener { + binding.btnCreate.setOnSingleClickListener { AmplitudeManager.trackEvent( EVENT_CLICK_BTN, mapOf(PROPERTY_PAGE to "create3"), @@ -128,8 +113,52 @@ class SelfieFragment : BaseFragment(R.layout.fragment_sel } } - private fun checkAndGetImages() { - if (isPhotoPickerAvailable(requireContext())) { + private fun setGuideTextByParent() { + with(binding) { + tvSelfieGuide1.setTextWithImage( + stringOf(R.string.selfie_tv_guide_1), + R.drawable.ic_check, + ) + if (viewModel.isCreatingParentPic) { + tvSelfieGuide2.setTextWithImage( + stringOf(R.string.selfie_tv_guide_parent), + R.drawable.ic_check, + ) + tvSelfieGuide3.isVisible = false + tvSelfieWarning.isVisible = false + } else { + tvSelfieGuide2.setTextWithImage( + stringOf(R.string.selfie_tv_guide_2), + R.drawable.ic_check, + ) + tvSelfieGuide3.isVisible = true + tvSelfieWarning.isVisible = true + tvSelfieGuide3.setTextWithImage( + stringOf(R.string.selfie_tv_guide_3), + R.drawable.ic_check, + ) + } + } + } + + private fun setLayoutByParent() { + with(binding) { + if (viewModel.isCreatingParentPic) { + if (viewModel.selectedNumber.value == PictureNumber.ONE) { + tvSelfieTitle.text = stringOf(R.string.selfie_tv_title_parent_one) + } else { + tvSelfieTitle.text = stringOf(R.string.selfie_tv_title_parent_two) + layoutExampleImage.isVisible = false + layoutTwoParent.isVisible = true + btnSelfieAdd.visibility = View.INVISIBLE + } + } + } + } + + private fun checkAndGetImages(currentAddingList: Int) { + viewModel.currentAddingList = currentAddingList + if (isPhotoPickerAvailable(requireContext()) && Build.VERSION.SDK_INT >= Build.VERSION_CODES.TIRAMISU) { photoPickerResult.launch(PickVisualMediaRequest(ActivityResultContracts.PickVisualMedia.ImageOnly)) } else { galleryPickerResult.launch( @@ -172,110 +201,69 @@ class SelfieFragment : BaseFragment(R.layout.fragment_sel private fun setImageListWithUri(uris: List) { with(viewModel) { - imageList = - uris.mapIndexed { _, uri -> - ImageFileModel( - uri.hashCode().toLong(), - uri.getFileName(requireActivity().contentResolver).toString(), - uri.toString(), - ) - } - isCompleted.value = uris.size == 3 - } - with(binding) { - listOf(ivAddedImage1, ivAddedImage2, ivAddedImage3).apply { - forEach { it.setImageDrawable(null) } - uris.take(size).forEachIndexed { index, uri -> - this[index].load(uri) - } - } - } - binding.layoutAddedImage.isVisible = uris.isNotEmpty() - } - - private fun setSavedImages() { - if (viewModel.imageList.isNotEmpty()) { - val imageViews = - with(binding) { listOf(ivAddedImage1, ivAddedImage2, ivAddedImage3) } - imageViews.forEach { it.setImageDrawable(null) } - viewModel.imageList - .take(3) - .forEachIndexed { index, imageFile -> imageViews[index].load(imageFile.url) } - binding.layoutAddedImage.isVisible = true + val listMap = mapOf( + 0 to ::imageList, + 1 to ::firstImageList, + 2 to ::secondImageList + ) + listMap[currentAddingList]?.set(uris.map { uri -> + uri.toImageFileModel() + }) + updateCompletionState(uris.size) } + setSavedImages() } - private fun setBulletPointList() { - val points = - listOf( - stringOf(R.string.selfie_tv_guide_one), - stringOf(R.string.selfie_tv_guide_two), - stringOf(R.string.selfie_tv_guide_three), - stringOf(R.string.selfie_tv_guide_four), - ) - val spannableStringBuilder = SpannableStringBuilder() - points.forEach { point -> - val spannableString = - SpannableString(point).apply { - setSpan( - BulletSpan(15), - 0, - point.length, - Spanned.SPAN_EXCLUSIVE_EXCLUSIVE, - ) - } - with(spannableStringBuilder) { - append(spannableString) - append("\n") - } - } - binding.tvSelfieGuideBody.text = spannableStringBuilder + private fun Uri.toImageFileModel(): ImageFileModel { + return ImageFileModel( + hashCode().toLong(), + getFileName(requireActivity().contentResolver).toString(), + toString() + ) } - private fun setGuideListBlur() { + private fun setSavedImages() { with(binding) { - svSelfieGuide.setOnScrollChangeListener { _, _, scrollY, _, _ -> - ivSelfieBlurBottom.alpha = max(0.0, (1 - scrollY / 500f).toDouble()).toFloat() - ivSelfieBlurTop.alpha = 1 - max(0.0, (1 - scrollY / 100f).toDouble()).toFloat() + if (viewModel.selectedNumber.value != PictureNumber.TWO) { + layoutAddedImage.isVisible = viewModel.imageList.isNotEmpty() + layoutExampleImage.isVisible = viewModel.imageList.isEmpty() + btnSelfieAdd.text = + if (viewModel.imageList.isEmpty()) stringOf(R.string.selfie_tv_btn_select) + else stringOf(R.string.selfie_tv_btn_reselect) + listOf(ivAddedImage1, ivAddedImage2, ivAddedImage3) + .resetAndLoadImages(viewModel.imageList) + } else { + listOf(ivAddedFirstImage1, ivAddedFirstImage2, ivAddedFirstImage3) + .resetAndLoadImages(viewModel.firstImageList) + listOf(ivAddedSecondImage1, ivAddedSecondImage2, ivAddedSecondImage3) + .resetAndLoadImages(viewModel.secondImageList) } } } - private fun initWaitingResult() { - if (!::waitingResult.isInitialized) { - waitingResult = - registerForActivityResult( - ActivityResultContracts.StartActivityForResult(), - ) { result -> - if (result.resultCode == RESULT_OK) { - requireActivity() - .supportFragmentManager - .beginTransaction() - .replace(R.id.fcv_main, FeedFragment()) - .commit() - (requireActivity() as? MainActivity)?.initBnvItemIconTintList() - } - } + private fun List.resetAndLoadImages(imageList: List) { + this.forEach { it.load(R.drawable.img_empty_image) } + imageList.take(size).forEachIndexed { index, file -> + this[index].load(file.url) } } private fun observeGeneratingState() { - viewModel.totalGeneratingState - .flowWithLifecycle(lifecycle) - .onEach { state -> - when (state) { - is UiState.Success -> { - AmplitudeManager.plusIntProperties("user_piccreate") - waitingResult.launch(Intent(requireContext(), WaitingActivity::class.java)) - with(viewModel) { - modCurrentPercent(-67) - resetGeneratingState() - } - } + viewModel.totalGeneratingState.flowWithLifecycle(lifecycle).onEach { state -> + when (state) { + is UiState.Success -> { + AmplitudeManager.plusIntProperties("user_piccreate") + startActivity(WaitingActivity.createIntent(requireContext(), state.data)) + requireActivity().finish() + } - is UiState.Failure -> toast(stringOf(R.string.error_msg)) - else -> return@onEach + is UiState.Failure -> { + toast(stringOf(R.string.error_msg)) + viewModel.resetTotalGeneratingState() } - }.launchIn(lifecycleScope) + + else -> return@onEach + } + }.launchIn(lifecycleScope) } } diff --git a/presentation/src/main/java/kr/genti/presentation/generate/finished/FinishedActivity.kt b/presentation/src/main/java/kr/genti/presentation/generate/finished/FinishedActivity.kt index 5f6ac5bf..7172f7c3 100644 --- a/presentation/src/main/java/kr/genti/presentation/generate/finished/FinishedActivity.kt +++ b/presentation/src/main/java/kr/genti/presentation/generate/finished/FinishedActivity.kt @@ -3,37 +3,26 @@ package kr.genti.presentation.generate.finished import android.content.Context import android.content.Intent import android.os.Bundle -import android.text.SpannableStringBuilder -import android.text.Spanned -import android.text.style.AbsoluteSizeSpan -import android.text.style.ForegroundColorSpan -import android.text.style.TypefaceSpan -import android.widget.ImageView -import android.widget.TextView +import android.view.ViewGroup +import androidx.activity.OnBackPressedCallback import androidx.activity.viewModels import androidx.core.content.FileProvider -import androidx.core.content.res.ResourcesCompat -import androidx.core.view.ViewCompat -import androidx.core.view.WindowCompat -import androidx.core.view.WindowInsetsCompat import androidx.core.view.isVisible -import androidx.core.view.updatePadding import androidx.lifecycle.flowWithLifecycle import androidx.lifecycle.lifecycleScope import coil.load -import coil.transform.RoundedCornersTransformation +import com.bumptech.glide.Glide +import com.bumptech.glide.request.RequestOptions import dagger.hilt.android.AndroidEntryPoint +import jp.wasabeef.glide.transformations.BlurTransformation import kotlinx.coroutines.flow.launchIn import kotlinx.coroutines.flow.onEach import kr.genti.core.base.BaseActivity -import kr.genti.core.extension.colorOf import kr.genti.core.extension.dpToPx import kr.genti.core.extension.setOnSingleClickListener import kr.genti.core.extension.stringOf import kr.genti.core.extension.toast -import kr.genti.domain.entity.response.ImageModel -import kr.genti.domain.enums.PictureRatio.Companion.toPictureRatio -import kr.genti.domain.enums.PictureType +import kr.genti.domain.enums.PictureRatio import kr.genti.presentation.R import kr.genti.presentation.databinding.ActivityFinishedBinding import kr.genti.presentation.main.profile.ProfileImageDialog.Companion.FILE_PROVIDER_AUTORITY @@ -43,6 +32,7 @@ import kr.genti.presentation.util.AmplitudeManager import kr.genti.presentation.util.AmplitudeManager.EVENT_CLICK_BTN import kr.genti.presentation.util.AmplitudeManager.PROPERTY_BTN import kr.genti.presentation.util.AmplitudeManager.PROPERTY_PAGE +import kr.genti.presentation.util.GlideResultListener import kr.genti.presentation.util.downloadImage import java.io.File @@ -63,10 +53,11 @@ class FinishedActivity : BaseActivity(R.layout.activity initImageBtnListener() initSaveBtnListener() initShareBtnListener() - initReturnBtnListener() initUnwantedBtnListener() + initCloseBtnListener() + initBackPressedListener() + setUiWIthIsPaidIntent() getIntentInfo() - setStatusBarTransparent() observeDownloadCacheImage() } @@ -75,26 +66,21 @@ class FinishedActivity : BaseActivity(R.layout.activity } private fun initImageBtnListener() { - with(binding) { - ivFinishedImage32.setOnSingleClickListener { showImageDialog() } - ivFinishedImage23.setOnSingleClickListener { showImageDialog() } + binding.ivFinishedImage.setOnSingleClickListener { + AmplitudeManager.trackEvent("enlarge_picdone_picture") + finishedImageDialog = FinishedImageDialog() + finishedImageDialog?.show(supportFragmentManager, DIALOG_IMAGE) } } - private fun showImageDialog() { - AmplitudeManager.trackEvent("enlarge_picdone_picture") - finishedImageDialog = FinishedImageDialog() - finishedImageDialog?.show(supportFragmentManager, DIALOG_IMAGE) - } - private fun initSaveBtnListener() { with(binding) { - btnDownload23.setOnSingleClickListener { saveImage() } - btnDownload32.setOnSingleClickListener { saveImage() } + btnDownload.setOnSingleClickListener { downloadImage() } + btnSavePaid.setOnSingleClickListener { downloadImage() } } } - private fun saveImage() { + private fun downloadImage() { AmplitudeManager.apply { trackEvent( EVENT_CLICK_BTN, @@ -103,7 +89,7 @@ class FinishedActivity : BaseActivity(R.layout.activity ) plusIntProperties("user_picturedownload") } - downloadImage(viewModel.finishedImage.id, viewModel.finishedImage.url) + downloadImage(viewModel.finishedImageId, viewModel.finishedImageUrl) } private fun initShareBtnListener() { @@ -121,18 +107,6 @@ class FinishedActivity : BaseActivity(R.layout.activity } } - private fun initReturnBtnListener() { - binding.btnReturnMain.setOnSingleClickListener { - AmplitudeManager.trackEvent( - EVENT_CLICK_BTN, - mapOf(PROPERTY_PAGE to "picdone"), - mapOf(PROPERTY_BTN to "gomain"), - ) - finishedRatingDialog = FinishedRatingDialog() - finishedRatingDialog?.show(supportFragmentManager, DIALOG_RATING) - } - } - private fun initUnwantedBtnListener() { binding.btnUnwanted.setOnSingleClickListener { finishedReportDialog = FinishedReportDialog() @@ -140,30 +114,87 @@ class FinishedActivity : BaseActivity(R.layout.activity } } + private fun initCloseBtnListener() { + binding.btnClose.setOnSingleClickListener { + showFinishedRatingDialog() + } + } + + private fun initBackPressedListener() { + val onBackPressedCallback = + object : OnBackPressedCallback(true) { + override fun handleOnBackPressed() { + showFinishedRatingDialog() + } + } + + this.onBackPressedDispatcher.addCallback(this, onBackPressedCallback) + } + + private fun setUiWIthIsPaidIntent() { + if (intent.getBooleanExtra(EXTRA_IS_PAID, false)) { + with(binding) { + tvFinishedTitle.text = stringOf(R.string.finished_tv_title_paid) + btnDownload.isVisible = false + ivFinishedTooltip.isVisible = false + btnShare.isVisible = false + btnSavePaid.isVisible = true + } + } + } + + private fun showFinishedRatingDialog() { + AmplitudeManager.trackEvent( + EVENT_CLICK_BTN, + mapOf(PROPERTY_PAGE to "picdone"), + mapOf(PROPERTY_BTN to "gomain"), + ) + finishedRatingDialog = FinishedRatingDialog() + finishedRatingDialog?.show(supportFragmentManager, DIALOG_RATING) + } + private fun getIntentInfo() { - viewModel.finishedImage = - ImageModel( - intent.getLongExtra(EXTRA_RESPONSE_ID, -1), - intent.getStringExtra(EXTRA_URL).orEmpty(), - "", - intent.getStringExtra(EXTRA_RATIO)?.toPictureRatio(), - PictureType.PictureCompleted, - ) - viewModel.setPictureRatio() - setUiWithRatio() + with(viewModel) { + finishedImageId = intent.getLongExtra(EXTRA_RESPONSE_ID, -1) + finishedImageUrl = intent.getStringExtra(EXTRA_URL).orEmpty() + finishedImageRatio = intent.getStringExtra(EXTRA_RATIO).orEmpty() + } + setImageLayout() } - private fun setUiWithRatio() { + private fun setImageLayout() { with(binding) { - layout32.isVisible = viewModel.isRatioGaro - layout23.isVisible = !viewModel.isRatioGaro - if (viewModel.isRatioGaro) { - ivFinishedImage32.loadImageToView() - tvFinishedTitle32.setEmphasizedText() - } else { - ivFinishedImage23.loadImageToView() - tvFinishedTitle23.setEmphasizedText() + if (viewModel.finishedImageRatio == PictureRatio.RATIO_GARO.name) { + setGaroImageMargin() } + ivFinishedImage.load(viewModel.finishedImageUrl) + ivFinishedBackground.apply { + Glide.with(this.context) + .load(viewModel.finishedImageUrl) + .apply(RequestOptions.bitmapTransform(BlurTransformation(50))) + .listener(GlideResultListener { + binding.layoutLoading.isVisible = false + }) + .into(this) + } + } + } + + private fun setGaroImageMargin() { + with(binding) { + cvFinishedImage.layoutParams = + (cvFinishedImage.layoutParams as ViewGroup.MarginLayoutParams).apply { + marginStart = 16.dpToPx(this@FinishedActivity) + marginEnd = 16.dpToPx(this@FinishedActivity) + } + btnDownload.layoutParams = + (btnDownload.layoutParams as ViewGroup.MarginLayoutParams).apply { + marginEnd = 16.dpToPx(this@FinishedActivity) + } + tvFinishedSubtitle.layoutParams = + (tvFinishedSubtitle.layoutParams as ViewGroup.MarginLayoutParams).apply { + bottomMargin = 64.dpToPx(this@FinishedActivity) + } } } @@ -191,52 +222,6 @@ class FinishedActivity : BaseActivity(R.layout.activity }.launchIn(lifecycleScope) } - private fun ImageView.loadImageToView() { - this.load(viewModel.finishedImage.url) { - transformations( - RoundedCornersTransformation( - 15.dpToPx(this@FinishedActivity).toFloat(), - ), - ) - } - } - - private fun TextView.setEmphasizedText() { - this.apply { - text = - SpannableStringBuilder(text).apply { - setSpan( - AbsoluteSizeSpan(22, true), - 0, - 11, - Spanned.SPAN_EXCLUSIVE_EXCLUSIVE, - ) - setSpan( - ForegroundColorSpan(colorOf(R.color.green_1)), - 0, - 11, - Spanned.SPAN_EXCLUSIVE_EXCLUSIVE, - ) - setSpan( - ResourcesCompat - .getFont(context, R.font.font_pretendard_bold) - ?.let { TypefaceSpan(it) }, - 0, - 11, - Spanned.SPAN_EXCLUSIVE_EXCLUSIVE, - ) - } - } - } - - private fun setStatusBarTransparent() { - WindowCompat.setDecorFitsSystemWindows(window, false) - ViewCompat.setOnApplyWindowInsetsListener(binding.root) { v, insets -> - v.updatePadding(bottom = insets.getInsets(WindowInsetsCompat.Type.navigationBars()).bottom) - insets - } - } - override fun onDestroy() { super.onDestroy() finishedImageDialog = null @@ -253,6 +238,7 @@ class FinishedActivity : BaseActivity(R.layout.activity private const val EXTRA_RESPONSE_ID = "EXTRA_RESPONSE_ID" private const val EXTRA_URL = "EXTRA_URL" private const val EXTRA_RATIO = "EXTRA_RATIO" + private const val EXTRA_IS_PAID = "EXTRA_IS_PAID" @JvmStatic fun createIntent( @@ -260,11 +246,13 @@ class FinishedActivity : BaseActivity(R.layout.activity id: Long, url: String, ratio: String, + isPaid: Boolean? = null, ): Intent = Intent(context, FinishedActivity::class.java).apply { putExtra(EXTRA_RESPONSE_ID, id) putExtra(EXTRA_URL, url) putExtra(EXTRA_RATIO, ratio) + putExtra(EXTRA_IS_PAID, isPaid) } } } diff --git a/presentation/src/main/java/kr/genti/presentation/generate/finished/FinishedImageDialog.kt b/presentation/src/main/java/kr/genti/presentation/generate/finished/FinishedImageDialog.kt index c5133c63..6029d02e 100644 --- a/presentation/src/main/java/kr/genti/presentation/generate/finished/FinishedImageDialog.kt +++ b/presentation/src/main/java/kr/genti/presentation/generate/finished/FinishedImageDialog.kt @@ -4,13 +4,11 @@ import android.content.DialogInterface import android.os.Bundle import android.view.View import android.view.WindowManager -import androidx.constraintlayout.widget.ConstraintLayout import androidx.fragment.app.activityViewModels import coil.load import kr.genti.core.base.BaseDialog import kr.genti.core.extension.setGusianBlur import kr.genti.core.extension.setOnSingleClickListener -import kr.genti.domain.enums.PictureRatio import kr.genti.presentation.R import kr.genti.presentation.databinding.DialogFinishedImageBinding import kr.genti.presentation.util.AmplitudeManager @@ -28,9 +26,6 @@ class FinishedImageDialog : BaseDialog(R.layout.dial ) setBackgroundDrawableResource(R.color.transparent) } - requireActivity() - .window.decorView.rootView - .setGusianBlur(50f) } override fun onViewCreated( @@ -46,7 +41,7 @@ class FinishedImageDialog : BaseDialog(R.layout.dial private fun initExitBtnListener() { binding.btnExit.setOnSingleClickListener { dismiss() } - binding.ivProfileBackground.setOnSingleClickListener { dismiss() } + binding.root.setOnSingleClickListener { dismiss() } } private fun initDownloadBtnListener() { @@ -55,25 +50,11 @@ class FinishedImageDialog : BaseDialog(R.layout.dial trackEvent("download_picdone_enlargedpicture") plusIntProperties("user_picturedownload") } - requireActivity().downloadImage(viewModel.finishedImage.id, viewModel.finishedImage.url) + requireActivity().downloadImage(viewModel.finishedImageId, viewModel.finishedImageUrl) } } private fun setImage() { - with(binding.ivProfile) { - load(viewModel.finishedImage.url) - if (viewModel.finishedImage.pictureRatio == PictureRatio.RATIO_GARO) { - (layoutParams as ConstraintLayout.LayoutParams).dimensionRatio = "3:2" - } else { - (layoutParams as ConstraintLayout.LayoutParams).dimensionRatio = "2:3" - } - } - } - - override fun onDismiss(dialog: DialogInterface) { - super.onDismiss(dialog) - requireActivity() - .window.decorView.rootView - .setGusianBlur(null) + binding.ivFinished.load(viewModel.finishedImageUrl) } } diff --git a/presentation/src/main/java/kr/genti/presentation/generate/finished/FinishedRatingDialog.kt b/presentation/src/main/java/kr/genti/presentation/generate/finished/FinishedRatingDialog.kt index 0c385102..8e8d152a 100644 --- a/presentation/src/main/java/kr/genti/presentation/generate/finished/FinishedRatingDialog.kt +++ b/presentation/src/main/java/kr/genti/presentation/generate/finished/FinishedRatingDialog.kt @@ -17,25 +17,22 @@ import kr.genti.core.extension.stringOf import kr.genti.core.extension.toast import kr.genti.presentation.R import kr.genti.presentation.databinding.DialogFinishedRatingBinding -import kr.genti.presentation.generate.openchat.OpenchatActivity import kr.genti.presentation.main.MainActivity import kr.genti.presentation.util.AmplitudeManager -class FinishedRatingDialog : BaseDialog(R.layout.dialog_finished_rating) { +class FinishedRatingDialog : + BaseDialog(R.layout.dialog_finished_rating) { private val viewModel by activityViewModels() override fun onStart() { super.onStart() dialog?.window?.apply { setLayout( - WindowManager.LayoutParams.WRAP_CONTENT, + WindowManager.LayoutParams.MATCH_PARENT, WindowManager.LayoutParams.WRAP_CONTENT, ) setBackgroundDrawableResource(R.color.transparent) } - requireActivity() - .window.decorView.rootView - .setGusianBlur(50f) } override fun onViewCreated( @@ -72,13 +69,9 @@ class FinishedRatingDialog : BaseDialog(R.layout.di } private fun navigateToMain() { - if (viewModel.getIsOpenchatAccessible()) { - startActivity(Intent(requireActivity(), OpenchatActivity::class.java)) - } else { - Intent(requireActivity(), MainActivity::class.java).apply { - flags = Intent.FLAG_ACTIVITY_CLEAR_TOP or Intent.FLAG_ACTIVITY_SINGLE_TOP - startActivity(this) - } + Intent(requireActivity(), MainActivity::class.java).apply { + flags = Intent.FLAG_ACTIVITY_CLEAR_TOP or Intent.FLAG_ACTIVITY_SINGLE_TOP + startActivity(this) } dismiss() } @@ -106,11 +99,4 @@ class FinishedRatingDialog : BaseDialog(R.layout.di } }.launchIn(lifecycleScope) } - - override fun onDismiss(dialog: DialogInterface) { - super.onDismiss(dialog) - requireActivity() - .window.decorView.rootView - .setGusianBlur(null) - } } diff --git a/presentation/src/main/java/kr/genti/presentation/generate/finished/FinishedReportDialog.kt b/presentation/src/main/java/kr/genti/presentation/generate/finished/FinishedReportDialog.kt index 6dfc06a2..a76fd316 100644 --- a/presentation/src/main/java/kr/genti/presentation/generate/finished/FinishedReportDialog.kt +++ b/presentation/src/main/java/kr/genti/presentation/generate/finished/FinishedReportDialog.kt @@ -21,7 +21,8 @@ import kr.genti.presentation.R import kr.genti.presentation.databinding.DialogFinishedReportBinding import kr.genti.presentation.util.AmplitudeManager -class FinishedReportDialog : BaseDialog(R.layout.dialog_finished_report) { +class FinishedReportDialog : + BaseDialog(R.layout.dialog_finished_report) { private val viewModel by activityViewModels() override fun onStart() { @@ -33,9 +34,6 @@ class FinishedReportDialog : BaseDialog(R.layout.di ) setBackgroundDrawableResource(R.color.transparent) } - requireActivity() - .window.decorView.rootView - .setGusianBlur(50f) } override fun onViewCreated( @@ -58,10 +56,6 @@ class FinishedReportDialog : BaseDialog(R.layout.di dismiss() requireActivity().finish() } - btnOkay.setOnSingleClickListener { - dismiss() - requireActivity().finish() - } } } @@ -91,21 +85,10 @@ class FinishedReportDialog : BaseDialog(R.layout.di with(binding) { layoutErrorInput.isVisible = false layoutErrorOutput.isVisible = true - viewOutside.setOnSingleClickListener { - dismiss() - requireActivity().finish() - } } } else { toast(stringOf(R.string.error_msg)) } }.launchIn(lifecycleScope) } - - override fun onDismiss(dialog: DialogInterface) { - super.onDismiss(dialog) - requireActivity() - .window.decorView.rootView - .setGusianBlur(null) - } } diff --git a/presentation/src/main/java/kr/genti/presentation/generate/finished/FinishedViewModel.kt b/presentation/src/main/java/kr/genti/presentation/generate/finished/FinishedViewModel.kt index c659e4ac..bdf6b05a 100644 --- a/presentation/src/main/java/kr/genti/presentation/generate/finished/FinishedViewModel.kt +++ b/presentation/src/main/java/kr/genti/presentation/generate/finished/FinishedViewModel.kt @@ -13,7 +13,6 @@ import kotlinx.coroutines.launch import kotlinx.coroutines.withContext import kr.genti.domain.entity.request.ReportRequestModel import kr.genti.domain.entity.response.ImageModel -import kr.genti.domain.enums.PictureRatio import kr.genti.domain.repository.GenerateRepository import kr.genti.domain.repository.UserRepository import java.io.File @@ -23,107 +22,93 @@ import javax.inject.Inject @HiltViewModel class FinishedViewModel - @Inject - constructor( - private val generateRepository: GenerateRepository, - private val userRepository: UserRepository, - ) : ViewModel() { - val errorReport = MutableLiveData() - val isWritten = MutableLiveData(false) +@Inject +constructor( + private val generateRepository: GenerateRepository +) : ViewModel() { + val errorReport = MutableLiveData() + val isWritten = MutableLiveData(false) - var finishedImage = - ImageModel( - -1, - "", - "", - null, - null, - ) + var finishedImageId: Long = -1 + var finishedImageUrl: String = "" + var finishedImageRatio: String = "" - var isRatioGaro = true + private val _isImageDownloaded = MutableSharedFlow() + val isImageDownloaded: SharedFlow = _isImageDownloaded - private val _isImageDownloaded = MutableSharedFlow() - val isImageDownloaded: SharedFlow = _isImageDownloaded + private val _postReportResult = MutableSharedFlow() + val postReportResult: SharedFlow = _postReportResult - private val _postReportResult = MutableSharedFlow() - val postReportResult: SharedFlow = _postReportResult + private val _postRateResult = MutableSharedFlow() + val postRateResult: SharedFlow = _postRateResult - private val _postRateResult = MutableSharedFlow() - val postRateResult: SharedFlow = _postRateResult + private val _postVerifyResult = MutableSharedFlow() + val postVerifyResult: SharedFlow = _postVerifyResult - private val _postVerifyResult = MutableSharedFlow() - val postVerifyResult: SharedFlow = _postVerifyResult - - fun checkWritten() { - isWritten.value = errorReport.value?.isNotEmpty() - } - - fun setPictureRatio() { - isRatioGaro = finishedImage.pictureRatio == PictureRatio.RATIO_GARO - } + fun checkWritten() { + isWritten.value = errorReport.value?.isNotEmpty() + } - fun downloadImageToCache(outputFile: File) { - viewModelScope.launch { - runCatching { - withContext(Dispatchers.IO) { - val connection = URL(finishedImage.url).openConnection() - connection.connect() - connection.getInputStream()?.use { inputStream -> - FileOutputStream(outputFile).use { out -> - val bitmap = BitmapFactory.decodeStream(inputStream) - bitmap?.compress(Bitmap.CompressFormat.PNG, 100, out) - } + fun downloadImageToCache(outputFile: File) { + viewModelScope.launch { + runCatching { + withContext(Dispatchers.IO) { + val connection = URL(finishedImageUrl).openConnection() + connection.connect() + connection.getInputStream()?.use { inputStream -> + FileOutputStream(outputFile).use { out -> + val bitmap = BitmapFactory.decodeStream(inputStream) + bitmap?.compress(Bitmap.CompressFormat.PNG, 100, out) } } - }.onSuccess { - _isImageDownloaded.emit(true) - }.onFailure { - _isImageDownloaded.emit(false) } + }.onSuccess { + _isImageDownloaded.emit(true) + }.onFailure { + _isImageDownloaded.emit(false) } } + } - fun postGenerateRateToServer(star: Int) { - viewModelScope.launch { - generateRepository - .postGenerateRate( - finishedImage.id.toInt(), - star, - ).onSuccess { - _postRateResult.emit(true) - }.onFailure { - _postRateResult.emit(false) - } - } + fun postGenerateRateToServer(star: Int) { + viewModelScope.launch { + generateRepository + .postGenerateRate( + finishedImageId.toInt(), + star, + ).onSuccess { + _postRateResult.emit(true) + }.onFailure { + _postRateResult.emit(false) + } } + } - fun postGenerateReportToServer() { - viewModelScope.launch { - generateRepository - .postGenerateReport( - ReportRequestModel( - finishedImage.id, - errorReport.value.orEmpty(), - ), - ).onSuccess { - _postReportResult.emit(true) - }.onFailure { - _postReportResult.emit(false) - } - } + fun postGenerateReportToServer() { + viewModelScope.launch { + generateRepository + .postGenerateReport( + ReportRequestModel( + finishedImageId, + errorReport.value.orEmpty(), + ), + ).onSuccess { + _postReportResult.emit(true) + }.onFailure { + _postReportResult.emit(false) + } } + } - fun postVerifyGenerateStateToServer() { - viewModelScope.launch { - generateRepository - .postVerifyGenerateState(finishedImage.id.toInt()) - .onSuccess { - _postVerifyResult.emit(true) - }.onFailure { - _postVerifyResult.emit(false) - } - } + fun postVerifyGenerateStateToServer() { + viewModelScope.launch { + generateRepository + .postVerifyGenerateState(finishedImageId.toInt()) + .onSuccess { + _postVerifyResult.emit(true) + }.onFailure { + _postVerifyResult.emit(false) + } } - - fun getIsOpenchatAccessible() = userRepository.getIsChatAccessible() } +} diff --git a/presentation/src/main/java/kr/genti/presentation/generate/openchat/OpenchatActivity.kt b/presentation/src/main/java/kr/genti/presentation/generate/openchat/OpenchatActivity.kt deleted file mode 100644 index d22bebd4..00000000 --- a/presentation/src/main/java/kr/genti/presentation/generate/openchat/OpenchatActivity.kt +++ /dev/null @@ -1,144 +0,0 @@ -package kr.genti.presentation.generate.openchat - -import android.content.Intent -import android.graphics.Color -import android.graphics.LinearGradient -import android.graphics.Shader -import android.net.Uri -import android.os.Bundle -import android.text.Spannable -import android.text.SpannableString -import android.text.style.ForegroundColorSpan -import androidx.activity.addCallback -import androidx.activity.viewModels -import androidx.lifecycle.flowWithLifecycle -import androidx.lifecycle.lifecycleScope -import coil.load -import dagger.hilt.android.AndroidEntryPoint -import kotlinx.coroutines.flow.distinctUntilChanged -import kotlinx.coroutines.flow.launchIn -import kotlinx.coroutines.flow.onEach -import kr.genti.core.base.BaseActivity -import kr.genti.core.extension.setOnSingleClickListener -import kr.genti.core.extension.stringOf -import kr.genti.core.extension.toast -import kr.genti.core.state.UiState -import kr.genti.presentation.R -import kr.genti.presentation.databinding.ActivityOpenchatBinding -import kr.genti.presentation.main.MainActivity - -@AndroidEntryPoint -class OpenchatActivity : BaseActivity(R.layout.activity_openchat) { - private val viewModel by viewModels() - - override fun onCreate(savedInstanceState: Bundle?) { - super.onCreate(savedInstanceState) - - initExitBtnListener() - initAccessAgainBtnListener() - setTitleTextGradation() - setBackPressed() - observeGetOpenchatState() - } - - override fun onResume() { - super.onResume() - - if (viewModel.isKakaoLaunched) navigateToMain() - } - - private fun initExitBtnListener() { - binding.btnExit.setOnSingleClickListener { - viewModel.setIsChatAccessible() - navigateToMain() - } - } - - private fun navigateToMain() { - Intent(this, MainActivity::class.java).apply { - flags = Intent.FLAG_ACTIVITY_CLEAR_TOP or Intent.FLAG_ACTIVITY_SINGLE_TOP - startActivity(this) - } - } - - private fun initAccessAgainBtnListener() { - binding.btnAccessAgain.setOnClickListener { - if (viewModel.isAccessible) { - viewModel.isAccessible = false - binding.ivAccessAgain.load(R.drawable.ic_check_checked) - binding.tvAccessAgain.setTextColor(Color.parseColor("#0D2D2B")) - } else { - viewModel.isAccessible = true - binding.ivAccessAgain.load(R.drawable.ic_check_unchecked) - binding.tvAccessAgain.setTextColor(Color.parseColor("#990D2D2B")) - } - } - } - - private fun setTitleTextGradation() { - binding.tvOpenchatTitleUp.apply { - paint.shader = - LinearGradient( - 0f, - 0f, - paint.measureText(text.toString()), - textSize, - intArrayOf( - Color.parseColor("#6CEE2A"), - Color.parseColor("#1CF48B"), - ), - null, - Shader.TileMode.CLAMP, - ) - } - } - - private fun setBackPressed() { - onBackPressedDispatcher.addCallback(this) { - viewModel.setIsChatAccessible() - navigateToMain() - } - } - - private fun observeGetOpenchatState() { - viewModel.getOpenchatState - .flowWithLifecycle(lifecycle) - .distinctUntilChanged() - .onEach { state -> - when (state) { - is UiState.Success -> { - with(state.data) { - if (!accessible) navigateToMain() - url?.let { initEnterBtnListener(it) } - count?.let { setGuideTextInfo(it) } - } - } - - is UiState.Failure -> toast(stringOf(R.string.error_msg)) - else -> return@onEach - } - }.launchIn(lifecycleScope) - } - - private fun initEnterBtnListener(url: String) { - binding.btnEnterOpenchat.setOnSingleClickListener { - with(viewModel) { - setIsChatAccessible() - isKakaoLaunched = true - } - startActivity(Intent(Intent.ACTION_VIEW, Uri.parse(url))) - } - } - - private fun setGuideTextInfo(count: Int) { - binding.tvOpenchatGuide.text = - SpannableString(getString(R.string.openchat_tv_guide, count)).apply { - setSpan( - ForegroundColorSpan(Color.parseColor("#49F155")), - 3, - 7 + count.toString().length, - Spannable.SPAN_EXCLUSIVE_EXCLUSIVE, - ) - } - } -} diff --git a/presentation/src/main/java/kr/genti/presentation/generate/openchat/OpenchatViewModel.kt b/presentation/src/main/java/kr/genti/presentation/generate/openchat/OpenchatViewModel.kt deleted file mode 100644 index 6f5efcc5..00000000 --- a/presentation/src/main/java/kr/genti/presentation/generate/openchat/OpenchatViewModel.kt +++ /dev/null @@ -1,49 +0,0 @@ -package kr.genti.presentation.generate.openchat - -import androidx.lifecycle.ViewModel -import androidx.lifecycle.viewModelScope -import dagger.hilt.android.lifecycle.HiltViewModel -import kotlinx.coroutines.flow.MutableStateFlow -import kotlinx.coroutines.flow.StateFlow -import kotlinx.coroutines.launch -import kr.genti.core.state.UiState -import kr.genti.domain.entity.response.OpenchatModel -import kr.genti.domain.repository.GenerateRepository -import kr.genti.domain.repository.UserRepository -import javax.inject.Inject - -@HiltViewModel -class OpenchatViewModel - @Inject - constructor( - private val generateRepository: GenerateRepository, - private val userRepository: UserRepository, - ) : ViewModel() { - private val _getOpenchatState = MutableStateFlow>(UiState.Empty) - val getOpenchatState: StateFlow> = _getOpenchatState - - var isAccessible = true - var isKakaoLaunched = false - - init { - getOpenchatData() - } - - private fun getOpenchatData() { - _getOpenchatState.value = UiState.Loading - viewModelScope.launch { - generateRepository - .getOpenchatData() - .onSuccess { - userRepository.setIsChatAccessible(it.accessible) - _getOpenchatState.value = UiState.Success(it) - }.onFailure { - _getOpenchatState.value = UiState.Failure(it.message.toString()) - } - } - } - - fun setIsChatAccessible() { - userRepository.setIsChatAccessible(isAccessible) - } - } diff --git a/presentation/src/main/java/kr/genti/presentation/generate/verify/VerifyActivity.kt b/presentation/src/main/java/kr/genti/presentation/generate/verify/VerifyActivity.kt index 5fd04c10..79b6826a 100644 --- a/presentation/src/main/java/kr/genti/presentation/generate/verify/VerifyActivity.kt +++ b/presentation/src/main/java/kr/genti/presentation/generate/verify/VerifyActivity.kt @@ -66,8 +66,8 @@ class VerifyActivity : BaseActivity(R.layout.activity_ver private fun initView() { AmplitudeManager.trackEvent("view_verifyme1") - setStatusBarColorFromResource(R.color.verify_bg) - setNavigationBarColorFromResource(R.color.verify_bg) + setStatusBarColorFromResource(R.color.black) + setNavigationBarColorFromResource(R.color.black) } private fun initExitBtnListener() { @@ -215,7 +215,7 @@ class VerifyActivity : BaseActivity(R.layout.activity_ver ) } }.onSuccess { - cameraLauncher.launch(photoUri) + photoUri?.let { cameraLauncher.launch(it) } } } else { toast(stringOf(R.string.error_msg)) @@ -229,7 +229,8 @@ class VerifyActivity : BaseActivity(R.layout.activity_ver cacheDir, ) - private fun getFileDateFormat() = SimpleDateFormat("yyyyMMdd_HHmmss", Locale.getDefault()).format(Date()) + private fun getFileDateFormat() = + SimpleDateFormat("yyyyMMdd_HHmmss", Locale.getDefault()).format(Date()) private fun observeGeneratingState() { viewModel.totalGeneratingState diff --git a/presentation/src/main/java/kr/genti/presentation/generate/verify/VerifyExitDialog.kt b/presentation/src/main/java/kr/genti/presentation/generate/verify/VerifyExitDialog.kt index 357b4fbc..6b359848 100644 --- a/presentation/src/main/java/kr/genti/presentation/generate/verify/VerifyExitDialog.kt +++ b/presentation/src/main/java/kr/genti/presentation/generate/verify/VerifyExitDialog.kt @@ -17,7 +17,7 @@ class VerifyExitDialog : BaseDialog(R.layout.dialog_ver super.onStart() dialog?.window?.apply { setLayout( - WindowManager.LayoutParams.WRAP_CONTENT, + WindowManager.LayoutParams.MATCH_PARENT, WindowManager.LayoutParams.WRAP_CONTENT, ) setBackgroundDrawableResource(R.color.transparent) @@ -35,7 +35,7 @@ class VerifyExitDialog : BaseDialog(R.layout.dialog_ver } private fun initCloseBtnListener() { - binding.btnClose.setOnSingleClickListener { dismiss() } + binding.btnReturn.setOnSingleClickListener { dismiss() } } private fun initExitBtnListener() { diff --git a/presentation/src/main/java/kr/genti/presentation/generate/waiting/PushDialog.kt b/presentation/src/main/java/kr/genti/presentation/generate/waiting/PushDialog.kt index bfa4cb2f..3ba78fbd 100644 --- a/presentation/src/main/java/kr/genti/presentation/generate/waiting/PushDialog.kt +++ b/presentation/src/main/java/kr/genti/presentation/generate/waiting/PushDialog.kt @@ -26,7 +26,7 @@ class PushDialog : BaseDialog(R.layout.dialog_push) { super.onStart() dialog?.window?.apply { setLayout( - WindowManager.LayoutParams.WRAP_CONTENT, + WindowManager.LayoutParams.MATCH_PARENT, WindowManager.LayoutParams.WRAP_CONTENT, ) setBackgroundDrawableResource(R.color.transparent) @@ -50,7 +50,12 @@ class PushDialog : BaseDialog(R.layout.dialog_push) { } private fun initCloseBtnListener() { - binding.btnClose.setOnSingleClickListener { dismiss() } + binding.btnClose.setOnSingleClickListener { + with(requireActivity()) { + setResult(Activity.RESULT_OK) + finish() + } + } } private fun initGetAlarmBtnListener() { @@ -92,13 +97,4 @@ class PushDialog : BaseDialog(R.layout.dialog_push) { requestPermissionLauncher.launch(Manifest.permission.POST_NOTIFICATIONS) } } - - override fun onDismiss(dialog: DialogInterface) { - super.onDismiss(dialog) - - with(requireActivity()) { - setResult(Activity.RESULT_OK) - finish() - } - } } diff --git a/presentation/src/main/java/kr/genti/presentation/generate/waiting/WaitingActivity.kt b/presentation/src/main/java/kr/genti/presentation/generate/waiting/WaitingActivity.kt index 094d35ce..096fac26 100644 --- a/presentation/src/main/java/kr/genti/presentation/generate/waiting/WaitingActivity.kt +++ b/presentation/src/main/java/kr/genti/presentation/generate/waiting/WaitingActivity.kt @@ -2,6 +2,8 @@ package kr.genti.presentation.generate.waiting import android.Manifest import android.app.Activity +import android.content.Context +import android.content.Intent import android.content.pm.PackageManager import android.os.Build import android.os.Bundle @@ -16,6 +18,7 @@ import kr.genti.core.base.BaseActivity import kr.genti.core.extension.setOnSingleClickListener import kr.genti.presentation.R import kr.genti.presentation.databinding.ActivityWaitBinding +import kr.genti.presentation.generate.finished.FinishedActivity import kr.genti.presentation.util.AmplitudeManager import kr.genti.presentation.util.AmplitudeManager.EVENT_CLICK_BTN import kr.genti.presentation.util.AmplitudeManager.PROPERTY_BTN @@ -30,6 +33,7 @@ class WaitingActivity : BaseActivity(R.layout.activity_wait initReturnBtnListener() setOnBackPressed() + setUiBuIsPaidIntent() setStatusBarTransparent() } @@ -75,6 +79,12 @@ class WaitingActivity : BaseActivity(R.layout.activity_wait false } + private fun setUiBuIsPaidIntent() { + if (intent.getBooleanExtra(EXTRA_IS_PAID, false)) { + binding.tvWaitTitle.text = getString(R.string.wait_tv_title_paid) + } + } + private fun setStatusBarTransparent() { WindowCompat.setDecorFitsSystemWindows(window, false) ViewCompat.setOnApplyWindowInsetsListener(binding.root) { v, insets -> @@ -91,5 +101,16 @@ class WaitingActivity : BaseActivity(R.layout.activity_wait companion object { private const val DIALOG_PUSH = "DIALOG_PUSH" + + private const val EXTRA_IS_PAID = "EXTRA_IS_PAID" + + @JvmStatic + fun createIntent( + context: Context, + isPaid: Boolean? = null, + ): Intent = + Intent(context, WaitingActivity::class.java).apply { + putExtra(EXTRA_IS_PAID, isPaid) + } } } diff --git a/presentation/src/main/java/kr/genti/presentation/main/CreateErrorDialog.kt b/presentation/src/main/java/kr/genti/presentation/main/CreateErrorDialog.kt index 6280f848..92bcb063 100644 --- a/presentation/src/main/java/kr/genti/presentation/main/CreateErrorDialog.kt +++ b/presentation/src/main/java/kr/genti/presentation/main/CreateErrorDialog.kt @@ -22,7 +22,7 @@ class CreateErrorDialog : BaseDialog(R.layout.dialog_c super.onStart() dialog?.window?.apply { setLayout( - WindowManager.LayoutParams.WRAP_CONTENT, + WindowManager.LayoutParams.MATCH_PARENT, WindowManager.LayoutParams.WRAP_CONTENT, ) setBackgroundDrawableResource(R.color.transparent) diff --git a/presentation/src/main/java/kr/genti/presentation/main/CreateFinishedDialog.kt b/presentation/src/main/java/kr/genti/presentation/main/CreateFinishedDialog.kt index 191a3bf3..d79ffb60 100644 --- a/presentation/src/main/java/kr/genti/presentation/main/CreateFinishedDialog.kt +++ b/presentation/src/main/java/kr/genti/presentation/main/CreateFinishedDialog.kt @@ -12,14 +12,15 @@ import kr.genti.presentation.R import kr.genti.presentation.databinding.DialogCreateFinishedBinding import kr.genti.presentation.generate.finished.FinishedActivity -class CreateFinishedDialog : BaseDialog(R.layout.dialog_create_finished) { +class CreateFinishedDialog : + BaseDialog(R.layout.dialog_create_finished) { private val viewModel by activityViewModels() override fun onStart() { super.onStart() dialog?.window?.apply { setLayout( - WindowManager.LayoutParams.WRAP_CONTENT, + WindowManager.LayoutParams.MATCH_PARENT, WindowManager.LayoutParams.WRAP_CONTENT, ) setBackgroundDrawableResource(R.color.transparent) @@ -45,18 +46,17 @@ class CreateFinishedDialog : BaseDialog(R.layout.di private fun initMoveToFinishBtnListener() { binding.btnMoveToFinish.setOnSingleClickListener { if (viewModel.checkNewPictureInitialized()) { - with(viewModel.newPicture.pictureGenerateResponse) { - FinishedActivity - .createIntent( - requireContext(), - this?.pictureGenerateResponseId ?: -1, - this?.pictureCompleted?.url.orEmpty(), - this - ?.pictureCompleted - ?.pictureRatio - ?.name - .orEmpty(), - ).apply { startActivity(this) } + with(viewModel.newPicture) { + startActivity( + FinishedActivity + .createIntent( + requireContext(), + this.response?.responseId ?: -1, + this.response?.picture?.url.orEmpty(), + this.response?.picture?.pictureRatio?.name.orEmpty(), + this.paid, + ) + ) } } else { toast(stringOf(R.string.error_msg)) diff --git a/presentation/src/main/java/kr/genti/presentation/main/CreateSelectDialog.kt b/presentation/src/main/java/kr/genti/presentation/main/CreateSelectDialog.kt new file mode 100644 index 00000000..33c5aea6 --- /dev/null +++ b/presentation/src/main/java/kr/genti/presentation/main/CreateSelectDialog.kt @@ -0,0 +1,55 @@ +package kr.genti.presentation.main + +import android.os.Bundle +import android.view.View +import android.view.WindowManager +import androidx.lifecycle.lifecycleScope +import kotlinx.coroutines.delay +import kotlinx.coroutines.launch +import kr.genti.core.base.BaseDialog +import kr.genti.core.extension.setOnSingleClickListener +import kr.genti.presentation.R +import kr.genti.presentation.create.CreateActivity +import kr.genti.presentation.databinding.DialogCreateSelectBinding + +class CreateSelectDialog : BaseDialog(R.layout.dialog_create_select) { + + override fun onStart() { + super.onStart() + dialog?.window?.apply { + setLayout( + WindowManager.LayoutParams.MATCH_PARENT, + WindowManager.LayoutParams.WRAP_CONTENT, + ) + setBackgroundDrawableResource(R.color.transparent) + } + } + + override fun onViewCreated( + view: View, + savedInstanceState: Bundle?, + ) { + super.onViewCreated(view, savedInstanceState) + + initCreateBtnListeners() + } + + private fun initCreateBtnListeners() { + with(binding) { + layoutDefaultCreate.setOnSingleClickListener { + navigateToCreate(false) + } + layoutParentCreate.setOnSingleClickListener { + navigateToCreate(true) + } + } + } + + private fun navigateToCreate(isCreatingParentPic: Boolean) { + startActivity(CreateActivity.createIntent(requireContext(), isCreatingParentPic)) + lifecycleScope.launch { + delay(500) + dismiss() + } + } +} diff --git a/presentation/src/main/java/kr/genti/presentation/main/CreateUnableDialog.kt b/presentation/src/main/java/kr/genti/presentation/main/CreateUnableDialog.kt index 98f97590..9dfc89b0 100644 --- a/presentation/src/main/java/kr/genti/presentation/main/CreateUnableDialog.kt +++ b/presentation/src/main/java/kr/genti/presentation/main/CreateUnableDialog.kt @@ -13,7 +13,7 @@ class CreateUnableDialog : BaseDialog(R.layout.dialog super.onStart() dialog?.window?.apply { setLayout( - WindowManager.LayoutParams.WRAP_CONTENT, + WindowManager.LayoutParams.MATCH_PARENT, WindowManager.LayoutParams.WRAP_CONTENT, ) setBackgroundDrawableResource(R.color.transparent) @@ -31,7 +31,6 @@ class CreateUnableDialog : BaseDialog(R.layout.dialog } private fun initCloseBtnListener() { - binding.btnClose.setOnSingleClickListener { dismiss() } binding.btnReturn.setOnSingleClickListener { dismiss() } } @@ -46,10 +45,7 @@ class CreateUnableDialog : BaseDialog(R.layout.dialog @JvmStatic fun newInstance(body: String): CreateUnableDialog = CreateUnableDialog().apply { - arguments = - Bundle().apply { - putString(ARG_BODY, body) - } + arguments = Bundle().apply { putString(ARG_BODY, body) } } } } diff --git a/presentation/src/main/java/kr/genti/presentation/main/MainActivity.kt b/presentation/src/main/java/kr/genti/presentation/main/MainActivity.kt index 490b306b..bff030b7 100644 --- a/presentation/src/main/java/kr/genti/presentation/main/MainActivity.kt +++ b/presentation/src/main/java/kr/genti/presentation/main/MainActivity.kt @@ -15,18 +15,17 @@ import dagger.hilt.android.AndroidEntryPoint import kotlinx.coroutines.flow.launchIn import kotlinx.coroutines.flow.onEach import kr.genti.core.base.BaseActivity -import kr.genti.core.extension.setStatusBarColorFromResource +import kr.genti.core.extension.initOnBackPressedListener import kr.genti.core.extension.stringOf import kr.genti.core.extension.toast import kr.genti.core.state.UiState import kr.genti.domain.enums.GenerateStatus +import kr.genti.presentation.BuildConfig import kr.genti.presentation.R import kr.genti.presentation.databinding.ActivityMainBinding import kr.genti.presentation.generate.finished.FinishedActivity -import kr.genti.presentation.generate.openchat.OpenchatActivity import kr.genti.presentation.generate.verify.VerifyActivity import kr.genti.presentation.generate.waiting.WaitingActivity -import kr.genti.presentation.main.create.CreateFragment import kr.genti.presentation.main.feed.FeedFragment import kr.genti.presentation.main.profile.ProfileFragment import kr.genti.presentation.util.AmplitudeManager @@ -38,20 +37,23 @@ class MainActivity : BaseActivity(R.layout.activity_main) { private var createFinishedDialog: CreateFinishedDialog? = null private var createErrorDialog: CreateErrorDialog? = null private var createUnableDialog: CreateUnableDialog? = null + private var createSelectDialog: CreateSelectDialog? = null override fun onCreate(savedInstanceState: Bundle?) { super.onCreate(savedInstanceState) + initOnBackPressedListener(binding.root) initBnvItemIconTintList() initBnvItemSelectedListener() initCreateBtnListener() - setStatusBarColor() + initPatchBtnListener() getNotificationIntent() observeStatusResult() observeNotificationState() observeResetResult() observeServerAvailableState() observeUserVerifyState() + observePatchResult() } override fun onResume() { @@ -63,7 +65,7 @@ class MainActivity : BaseActivity(R.layout.activity_main) { } } - fun initBnvItemIconTintList() { + private fun initBnvItemIconTintList() { with(binding.bnvMain) { itemIconTintList = null selectedItemId = R.id.menu_feed @@ -76,178 +78,170 @@ class MainActivity : BaseActivity(R.layout.activity_main) { binding.bnvMain.setOnItemSelectedListener { menu -> if (binding.bnvMain.selectedItemId == menu.itemId) { + if (menu.itemId == R.id.menu_feed) { + (supportFragmentManager.findFragmentById(R.id.fcv_main) as FeedFragment).scrollFeedListToTop() + } return@setOnItemSelectedListener false + } when (menu.itemId) { R.id.menu_feed -> navigateTo("click_maintab") - R.id.menu_create -> navigateTo("click_createpictab") + R.id.menu_create -> return@setOnItemSelectedListener false R.id.menu_profile -> navigateTo("click_mypagetab") else -> return@setOnItemSelectedListener false } - binding.btnMenuCreate.isVisible = menu.itemId != R.id.menu_create true } } private fun initCreateBtnListener() { binding.btnMenuCreate.setOnClickListener { - navigateByGenerateStatus() - } - } - - private fun setStatusBarColor() { - setStatusBarColorFromResource(R.color.background_white) - } - - private fun getNotificationIntent() { - when (intent.getStringExtra(EXTRA_TYPE)) { - TYPE_SUCCESS -> viewModel.getGenerateStatusFromServer(true) - TYPE_CANCELED -> viewModel.getGenerateStatusFromServer(true) - TYPE_OPENCHAT -> startActivity(Intent(this, OpenchatActivity::class.java)) - else -> return - } - } + when (viewModel.currentStatus) { + GenerateStatus.NEW_REQUEST_AVAILABLE -> { + viewModel.getIsServerAvailable() + } - private fun navigateByGenerateStatus() { - when (viewModel.currentStatus) { - GenerateStatus.NEW_REQUEST_AVAILABLE -> { - viewModel.getIsServerAvailable() - } + GenerateStatus.AWAIT_USER_VERIFICATION -> { + createFinishedDialog = CreateFinishedDialog() + createFinishedDialog?.show(supportFragmentManager, DIALOG_FINISHED) + } - GenerateStatus.AWAIT_USER_VERIFICATION -> { - createFinishedDialog = CreateFinishedDialog() - createFinishedDialog?.show(supportFragmentManager, DIALOG_FINISHED) - } + GenerateStatus.IN_PROGRESS -> { + if (BuildConfig.DEBUG) binding.btnPatchInDevelop.isVisible = true + startActivity(WaitingActivity.createIntent(this, viewModel.newPicture.paid)) + } - GenerateStatus.IN_PROGRESS -> { - Intent(this, WaitingActivity::class.java).apply { - startActivity(this) + GenerateStatus.CANCELED -> { + createErrorDialog = CreateErrorDialog() + createErrorDialog?.show(supportFragmentManager, DIALOG_ERROR) } - } - GenerateStatus.CANCELED -> { - createErrorDialog = CreateErrorDialog() - createErrorDialog?.show(supportFragmentManager, DIALOG_ERROR) + GenerateStatus.EMPTY -> return@setOnClickListener } + } + } - GenerateStatus.EMPTY -> return + private fun initPatchBtnListener() { + binding.btnPatchInDevelop.setOnClickListener { + viewModel.patchStatusInDevelop() + } + } + + private fun getNotificationIntent() { + if (intent.getStringExtra(EXTRA_TYPE) == TYPE_SUCCESS || intent.getStringExtra(EXTRA_TYPE) == TYPE_CANCELED) { + viewModel.getGenerateStatusFromServer(true) } } private fun observeStatusResult() { - viewModel.getStatusResult - .flowWithLifecycle(lifecycle) - .onEach { result -> - if (!result) toast(stringOf(R.string.error_msg)) - }.launchIn(lifecycleScope) + viewModel.getStatusResult.flowWithLifecycle(lifecycle).onEach { result -> + if (!result) toast(stringOf(R.string.error_msg)) + }.launchIn(lifecycleScope) } private fun observeNotificationState() { - viewModel.notificationState - .flowWithLifecycle(lifecycle) - .onEach { status -> - when (status) { - GenerateStatus.AWAIT_USER_VERIFICATION -> { - if (viewModel.checkNewPictureInitialized()) { - AmplitudeManager.trackEvent( - "click_push_notification", - mapOf("push_type" to "creating_success"), - ) - with(viewModel.newPicture.pictureGenerateResponse) { - FinishedActivity - .createIntent( - this@MainActivity, - this?.pictureGenerateResponseId ?: -1, - this?.pictureCompleted?.url.orEmpty(), - this - ?.pictureCompleted - ?.pictureRatio - ?.name - .orEmpty(), - ).apply { startActivity(this) } - } - } else { - toast(stringOf(R.string.error_msg)) - } - } - - GenerateStatus.CANCELED -> { + viewModel.notificationState.flowWithLifecycle(lifecycle).onEach { status -> + when (status) { + GenerateStatus.AWAIT_USER_VERIFICATION -> { + if (viewModel.checkNewPictureInitialized()) { AmplitudeManager.trackEvent( "click_push_notification", - mapOf("push_type" to "creating_fail"), + mapOf("push_type" to "creating_success"), ) - createErrorDialog = CreateErrorDialog() - createErrorDialog?.show(supportFragmentManager, DIALOG_ERROR) + with(viewModel.newPicture) { + startActivity( + FinishedActivity.createIntent( + this@MainActivity, + this.response?.responseId ?: -1, + this.response?.picture?.url.orEmpty(), + this.response?.picture?.pictureRatio?.name.orEmpty(), + this.paid, + ) + ) + } + } else { + toast(stringOf(R.string.error_msg)) } + } - else -> return@onEach + GenerateStatus.CANCELED -> { + AmplitudeManager.trackEvent( + "click_push_notification", + mapOf("push_type" to "creating_fail"), + ) + createErrorDialog = CreateErrorDialog() + createErrorDialog?.show(supportFragmentManager, DIALOG_ERROR) } - viewModel.resetNotificationState() - }.launchIn(lifecycleScope) + + else -> return@onEach + } + viewModel.resetNotificationState() + }.launchIn(lifecycleScope) } private fun observeResetResult() { - viewModel.postResetResult - .flowWithLifecycle(lifecycle) - .onEach { result -> - if (!result) { - toast(stringOf(R.string.error_msg)) - } else { - binding.bnvMain.selectedItemId = R.id.menu_create - } - }.launchIn(lifecycleScope) + viewModel.postResetResult.flowWithLifecycle(lifecycle).onEach { result -> + if (!result) { + toast(stringOf(R.string.error_msg)) + } else { + navigateToCreate() + } + }.launchIn(lifecycleScope) } private fun observeServerAvailableState() { - viewModel.serverAvailableState - .flowWithLifecycle(lifecycle) - .onEach { state -> - when (state) { - is UiState.Success -> { - if (state.data.status) { - viewModel.getIsUserVerifiedFromServer() - } else { - createUnableDialog = CreateUnableDialog.newInstance(state.data.message.orEmpty()) - createUnableDialog?.show(supportFragmentManager, DIALOG_UNABLE) - } + viewModel.serverAvailableState.flowWithLifecycle(lifecycle).onEach { state -> + when (state) { + is UiState.Success -> { + if (state.data.status) { + viewModel.getIsUserVerifiedFromServer() + } else { + createUnableDialog = + CreateUnableDialog.newInstance(state.data.message.orEmpty()) + createUnableDialog?.show(supportFragmentManager, DIALOG_UNABLE) } - - is UiState.Failure -> toast(stringOf(R.string.error_msg)) - else -> return@onEach } - viewModel.resetIsUserVerified() - }.launchIn(lifecycleScope) + + is UiState.Failure -> toast(stringOf(R.string.error_msg)) + else -> return@onEach + } + viewModel.resetIsServerAvailable() + }.launchIn(lifecycleScope) } private fun observeUserVerifyState() { - viewModel.userVerifyState - .flowWithLifecycle(lifecycle) - .onEach { state -> - when (state) { - is UiState.Success -> { - if (!viewModel.isUserTryingVerify) { - if (state.data) { - binding.bnvMain.selectedItemId = R.id.menu_create - } else { - viewModel.isUserTryingVerify = true - startActivity(Intent(this, VerifyActivity::class.java)) - } + viewModel.userVerifyState.flowWithLifecycle(lifecycle).onEach { state -> + when (state) { + is UiState.Success -> { + if (!viewModel.isUserTryingVerify) { + if (state.data) { + navigateToCreate() } else { - viewModel.isUserTryingVerify = false - if (state.data) { - binding.bnvMain.selectedItemId = R.id.menu_create - } + viewModel.isUserTryingVerify = true + startActivity(Intent(this, VerifyActivity::class.java)) } + } else { + viewModel.isUserTryingVerify = false + if (state.data) navigateToCreate() } - - is UiState.Failure -> toast(stringOf(R.string.error_msg)) - else -> return@onEach } - viewModel.resetIsUserVerified() - }.launchIn(lifecycleScope) + + is UiState.Failure -> toast(stringOf(R.string.error_msg)) + else -> return@onEach + } + viewModel.resetIsUserVerified() + }.launchIn(lifecycleScope) + } + + private fun observePatchResult() { + viewModel.patchStatusResult.flowWithLifecycle(lifecycle).onEach { result -> + if (!result) { + toast(stringOf(R.string.error_msg)) + } + }.launchIn(lifecycleScope) } private inline fun navigateTo(page: String?) { @@ -257,22 +251,29 @@ class MainActivity : BaseActivity(R.layout.activity_main) { } } + private fun navigateToCreate() { + AmplitudeManager.trackEvent("click_createpictab") + createSelectDialog = CreateSelectDialog() + createSelectDialog?.show(supportFragmentManager, DIALOG_SELECT) + } + override fun onDestroy() { super.onDestroy() createFinishedDialog = null createErrorDialog = null createUnableDialog = null + createSelectDialog = null } companion object { private const val DIALOG_FINISHED = "DIALOG_FINISHED" private const val DIALOG_ERROR = "DIALOG_ERROR" private const val DIALOG_UNABLE = "DIALOG_UNABLE" + private const val DIALOG_SELECT = "DIALOG_SELECT" const val TYPE_SUCCESS = "SUCCESS" const val TYPE_CANCELED = "CANCELED" - const val TYPE_OPENCHAT = "OPENCHAT" private const val EXTRA_TYPE = "EXTRA_DEFAULT" diff --git a/presentation/src/main/java/kr/genti/presentation/main/MainViewModel.kt b/presentation/src/main/java/kr/genti/presentation/main/MainViewModel.kt index a6400a60..d63786d6 100644 --- a/presentation/src/main/java/kr/genti/presentation/main/MainViewModel.kt +++ b/presentation/src/main/java/kr/genti/presentation/main/MainViewModel.kt @@ -17,94 +17,113 @@ import javax.inject.Inject @HiltViewModel class MainViewModel - @Inject - constructor( - private val generateRepository: GenerateRepository, - ) : ViewModel() { - private val _getStatusResult = MutableSharedFlow() - val getStatusResult: SharedFlow = _getStatusResult - - private val _postResetResult = MutableSharedFlow() - val postResetResult: SharedFlow = _postResetResult - - private val _notificationState = MutableStateFlow(GenerateStatus.EMPTY) - val notificationState: StateFlow = _notificationState - - private val _serverAvailableState = - MutableStateFlow>(UiState.Empty) - val serverAvailableState: StateFlow> = _serverAvailableState - - private val _userVerifyState = MutableStateFlow>(UiState.Empty) - val userVerifyState: StateFlow> = _userVerifyState - - var currentStatus: GenerateStatus = GenerateStatus.NEW_REQUEST_AVAILABLE - lateinit var newPicture: GenerateStatusModel - - var isUserTryingVerify = false - - fun getGenerateStatusFromServer(isNotification: Boolean) { - viewModelScope.launch { - generateRepository - .getGenerateStatus() - .onSuccess { - currentStatus = it.status - newPicture = it - if (isNotification) { - _notificationState.value = it.status - } - }.onFailure { - _getStatusResult.emit(false) +@Inject +constructor( + private val generateRepository: GenerateRepository, +) : ViewModel() { + private val _getStatusResult = MutableSharedFlow() + val getStatusResult: SharedFlow = _getStatusResult + + private val _postResetResult = MutableSharedFlow() + val postResetResult: SharedFlow = _postResetResult + + private val _patchStatusResult = MutableSharedFlow() + val patchStatusResult: SharedFlow = _patchStatusResult + + private val _notificationState = MutableStateFlow(GenerateStatus.EMPTY) + val notificationState: StateFlow = _notificationState + + private val _serverAvailableState = + MutableStateFlow>(UiState.Empty) + val serverAvailableState: StateFlow> = _serverAvailableState + + private val _userVerifyState = MutableStateFlow>(UiState.Empty) + val userVerifyState: StateFlow> = _userVerifyState + + var currentStatus: GenerateStatus = GenerateStatus.NEW_REQUEST_AVAILABLE + lateinit var newPicture: GenerateStatusModel + + var isUserTryingVerify = false + + fun getGenerateStatusFromServer(isNotification: Boolean) { + viewModelScope.launch { + generateRepository + .getGenerateStatus() + .onSuccess { + currentStatus = it.status + newPicture = it + if (isNotification) { + _notificationState.value = it.status } - } + }.onFailure { + _getStatusResult.emit(false) + } } + } - fun resetNotificationState() { - _notificationState.value = GenerateStatus.EMPTY - } + fun resetNotificationState() { + _notificationState.value = GenerateStatus.EMPTY + } - fun postResetStateToServer() { - viewModelScope.launch { - generateRepository - .getCanceledToReset( - newPicture.pictureGenerateRequestId.toString(), - ).onSuccess { - _postResetResult.emit(true) - getGenerateStatusFromServer(false) - }.onFailure { - _postResetResult.emit(false) - } - } + fun postResetStateToServer() { + viewModelScope.launch { + generateRepository + .getCanceledToReset( + newPicture.requestId.toString(), + ).onSuccess { + _postResetResult.emit(true) + getGenerateStatusFromServer(false) + }.onFailure { + _postResetResult.emit(false) + } } + } - fun checkNewPictureInitialized() = ::newPicture.isInitialized - - fun getIsServerAvailable() { - _serverAvailableState.value = UiState.Loading - viewModelScope.launch { - generateRepository - .getIsServerAvailable() - .onSuccess { - _serverAvailableState.value = UiState.Success(it) - }.onFailure { - _serverAvailableState.value = UiState.Failure(it.message.orEmpty()) - } - } + fun checkNewPictureInitialized() = ::newPicture.isInitialized + + fun getIsServerAvailable() { + _serverAvailableState.value = UiState.Loading + viewModelScope.launch { + generateRepository + .getIsServerAvailable() + .onSuccess { + _serverAvailableState.value = UiState.Success(it) + }.onFailure { + _serverAvailableState.value = UiState.Failure(it.message.orEmpty()) + } } + } - fun getIsUserVerifiedFromServer() { - _userVerifyState.value = UiState.Loading - viewModelScope.launch { - generateRepository - .getIsUserVerified() - .onSuccess { - _userVerifyState.value = UiState.Success(it) - }.onFailure { - _userVerifyState.value = UiState.Failure(it.message.orEmpty()) - } - } + fun resetIsServerAvailable() { + _serverAvailableState.value = UiState.Empty + } + + fun getIsUserVerifiedFromServer() { + _userVerifyState.value = UiState.Loading + viewModelScope.launch { + generateRepository + .getIsUserVerified() + .onSuccess { + _userVerifyState.value = UiState.Success(it) + }.onFailure { + _userVerifyState.value = UiState.Failure(it.message.orEmpty()) + } } + } + + fun resetIsUserVerified() { + _userVerifyState.value = UiState.Empty + } - fun resetIsUserVerified() { - _userVerifyState.value = UiState.Empty + fun patchStatusInDevelop() { + viewModelScope.launch { + generateRepository + .patchStatusInDevelop() + .onSuccess { + _patchStatusResult.emit(true) + }.onFailure { + _patchStatusResult.emit(false) + } } } +} diff --git a/presentation/src/main/java/kr/genti/presentation/main/create/CreateFragment.kt b/presentation/src/main/java/kr/genti/presentation/main/create/CreateFragment.kt deleted file mode 100644 index 5614c46d..00000000 --- a/presentation/src/main/java/kr/genti/presentation/main/create/CreateFragment.kt +++ /dev/null @@ -1,123 +0,0 @@ -package kr.genti.presentation.main.create - -import android.animation.ObjectAnimator -import android.os.Bundle -import android.view.View -import android.view.animation.LinearInterpolator -import androidx.core.view.isVisible -import androidx.fragment.app.activityViewModels -import androidx.lifecycle.flowWithLifecycle -import androidx.lifecycle.lifecycleScope -import androidx.navigation.NavController -import androidx.navigation.fragment.NavHostFragment -import dagger.hilt.android.AndroidEntryPoint -import kotlinx.coroutines.flow.launchIn -import kotlinx.coroutines.flow.onEach -import kr.genti.core.base.BaseFragment -import kr.genti.core.extension.setOnSingleClickListener -import kr.genti.core.extension.setStatusBarColor -import kr.genti.core.state.UiState -import kr.genti.presentation.R -import kr.genti.presentation.databinding.FragmentCreateBinding -import kr.genti.presentation.util.AmplitudeManager -import kr.genti.presentation.util.AmplitudeManager.EVENT_CLICK_BTN -import kr.genti.presentation.util.AmplitudeManager.PROPERTY_BTN -import kr.genti.presentation.util.AmplitudeManager.PROPERTY_PAGE - -@AndroidEntryPoint -class CreateFragment() : BaseFragment(R.layout.fragment_create) { - private val viewModel by activityViewModels() - lateinit var navController: NavController - - override fun onViewCreated( - view: View, - savedInstanceState: Bundle?, - ) { - super.onViewCreated(view, savedInstanceState) - - initView() - initBackBtnListener() - setCurrentFragment() - observeProgressBar() - observeGeneratingState() - } - - private fun initView() { - setStatusBarColor(R.color.background_white) - navController = binding.fcvCreate.getFragment().navController - } - - private fun initBackBtnListener() { - binding.btnBack.setOnSingleClickListener { - when (navController.currentDestination?.id) { - R.id.defineFragment -> return@setOnSingleClickListener - - R.id.poseFragment -> { - AmplitudeManager.trackEvent( - EVENT_CLICK_BTN, - mapOf(PROPERTY_PAGE to "create2"), - mapOf(PROPERTY_BTN to "back"), - ) - navController.popBackStack() - viewModel.modCurrentPercent(-33) - } - - R.id.selfieFragment -> { - AmplitudeManager.trackEvent( - EVENT_CLICK_BTN, - mapOf(PROPERTY_PAGE to "create3"), - mapOf(PROPERTY_BTN to "back"), - ) - navController.popBackStack() - viewModel.modCurrentPercent(-34) - } - } - } - } - - private fun setCurrentFragment() { - if (::navController.isInitialized) { - when (viewModel.currentPercent.value) { - 66 -> navController.navigate(R.id.poseFragment) - 100 -> { - navController.navigate(R.id.poseFragment) - navController.navigate(R.id.selfieFragment) - } - - else -> return - } - } - } - - private fun observeProgressBar() { - viewModel.currentPercent.flowWithLifecycle(lifecycle).onEach { percent -> - ObjectAnimator.ofInt( - binding.progressbarCreate, - PROPERTY_PROGRESS, - binding.progressbarCreate.progress, - percent, - ).apply { - duration = 300 - interpolator = LinearInterpolator() - start() - } - binding.btnBack.isVisible = viewModel.currentPercent.value > 33 - }.launchIn(lifecycleScope) - } - - private fun observeGeneratingState() { - viewModel.totalGeneratingState.flowWithLifecycle(lifecycle).onEach { state -> - if (state == UiState.Loading) { - setStatusBarColor(R.color.background_50) - binding.layoutLoading.isVisible = true - } else { - setStatusBarColor(R.color.background_white) - binding.layoutLoading.isVisible = false - } - }.launchIn(lifecycleScope) - } - - companion object { - const val PROPERTY_PROGRESS = "progress" - } -} diff --git a/presentation/src/main/java/kr/genti/presentation/main/create/CreateGuideDialog.kt b/presentation/src/main/java/kr/genti/presentation/main/create/CreateGuideDialog.kt deleted file mode 100644 index bc61e48f..00000000 --- a/presentation/src/main/java/kr/genti/presentation/main/create/CreateGuideDialog.kt +++ /dev/null @@ -1,35 +0,0 @@ -package kr.genti.presentation.main.create - -import android.os.Bundle -import android.view.View -import android.view.WindowManager -import kr.genti.core.base.BaseDialog -import kr.genti.core.extension.setOnSingleClickListener -import kr.genti.presentation.R -import kr.genti.presentation.databinding.DialogCreateGuideBinding - -class CreateGuideDialog : BaseDialog(R.layout.dialog_create_guide) { - override fun onStart() { - super.onStart() - dialog?.window?.apply { - setLayout( - WindowManager.LayoutParams.WRAP_CONTENT, - WindowManager.LayoutParams.WRAP_CONTENT, - ) - setBackgroundDrawableResource(R.color.transparent) - } - } - - override fun onViewCreated( - view: View, - savedInstanceState: Bundle?, - ) { - super.onViewCreated(view, savedInstanceState) - - initCloseBtnListener() - } - - private fun initCloseBtnListener() { - binding.btnClose.setOnSingleClickListener { dismiss() } - } -} diff --git a/presentation/src/main/java/kr/genti/presentation/main/create/CreateViewModel.kt b/presentation/src/main/java/kr/genti/presentation/main/create/CreateViewModel.kt deleted file mode 100644 index 18290aab..00000000 --- a/presentation/src/main/java/kr/genti/presentation/main/create/CreateViewModel.kt +++ /dev/null @@ -1,277 +0,0 @@ -package kr.genti.presentation.main.create - -import androidx.lifecycle.MutableLiveData -import androidx.lifecycle.ViewModel -import androidx.lifecycle.viewModelScope -import dagger.hilt.android.lifecycle.HiltViewModel -import kotlinx.coroutines.async -import kotlinx.coroutines.awaitAll -import kotlinx.coroutines.flow.MutableStateFlow -import kotlinx.coroutines.flow.StateFlow -import kotlinx.coroutines.launch -import kr.genti.core.state.UiState -import kr.genti.domain.entity.request.CreateRequestModel -import kr.genti.domain.entity.request.KeyRequestModel -import kr.genti.domain.entity.request.S3RequestModel -import kr.genti.domain.entity.response.ImageFileModel -import kr.genti.domain.entity.response.ImageFileModel.Companion.emptyImageFileModel -import kr.genti.domain.entity.response.S3PresignedUrlModel -import kr.genti.domain.enums.CameraAngle -import kr.genti.domain.enums.FileType -import kr.genti.domain.enums.PictureRatio -import kr.genti.domain.enums.ShotCoverage -import kr.genti.domain.repository.CreateRepository -import kr.genti.domain.repository.UploadRepository -import kr.genti.domain.repository.UserRepository -import javax.inject.Inject -import kotlin.random.Random - -@HiltViewModel -class CreateViewModel - @Inject - constructor( - private val createRepository: CreateRepository, - private val uploadRepository: UploadRepository, - private val userRepository: UserRepository, - ) : ViewModel() { - val prompt = MutableLiveData() - var plusImage = emptyImageFileModel() - val isWritten = MutableLiveData(false) - - val selectedRatio = MutableLiveData() - val selectedAngle = MutableLiveData() - val selectedCoverage = MutableLiveData() - val isSelected = MutableLiveData(false) - - var imageList = listOf() - var isCompleted = MutableLiveData(false) - - private val _currentPercent = MutableStateFlow(33) - val currentPercent: StateFlow = _currentPercent - - private var currentPrompt: String = "" - - private val _totalGeneratingState = MutableStateFlow>(UiState.Empty) - val totalGeneratingState: StateFlow> = _totalGeneratingState - - private var plusImageS3Key = KeyRequestModel(null) - private var imageS3KeyList = listOf() - - fun modCurrentPercent(amount: Int) { - _currentPercent.value += amount - } - - fun checkWritten() { - isWritten.value = prompt.value?.isNotEmpty() - } - - fun selectRatio(item: PictureRatio) { - selectedRatio.value = item - checkSelected() - } - - fun selectAngle(item: CameraAngle) { - selectedAngle.value = item - checkSelected() - } - - fun selectFrame(item: ShotCoverage) { - selectedCoverage.value = item - checkSelected() - } - - private fun checkSelected() { - isSelected.value = - selectedRatio.value != null && selectedAngle.value != null && selectedCoverage.value != null - } - - fun startSendingImages() { - _totalGeneratingState.value = UiState.Loading - viewModelScope.launch { - runCatching { - listOfNotNull( - if (plusImage.id != (-1).toLong()) async { getSingleS3Url() } else null, - async { getMultiS3Urls() }, - ).awaitAll() - }.onSuccess { - postToGenerateImage() - }.onFailure { - _totalGeneratingState.value = UiState.Failure(it.message.toString()) - } - } - } - - private suspend fun getSingleS3Url() { - createRepository.getS3SingleUrl( - S3RequestModel(FileType.USER_UPLOADED_IMAGE, plusImage.name), - ) - .onSuccess { uriModel -> - plusImageS3Key = KeyRequestModel(uriModel.s3Key) - postSingleImage(uriModel) - }.onFailure { - _totalGeneratingState.value = UiState.Failure(it.message.toString()) - } - } - - private suspend fun getMultiS3Urls() { - createRepository.getS3MultiUrl( - listOf( - S3RequestModel(FileType.USER_UPLOADED_IMAGE, imageList[0].name), - S3RequestModel(FileType.USER_UPLOADED_IMAGE, imageList[1].name), - S3RequestModel(FileType.USER_UPLOADED_IMAGE, imageList[2].name), - ), - ).onSuccess { uriList -> - imageS3KeyList = uriList.map { KeyRequestModel(it.s3Key) } - postMultiImage(uriList) - }.onFailure { - _totalGeneratingState.value = UiState.Failure(it.message.toString()) - } - } - - private suspend fun postSingleImage(urlModel: S3PresignedUrlModel) { - uploadRepository.uploadImage(urlModel.url, plusImage.url) - .onFailure { - _totalGeneratingState.value = UiState.Failure(it.message.toString()) - } - } - - private suspend fun postMultiImage(urlModelList: List) { - urlModelList.mapIndexed { i, urlModel -> - viewModelScope.async { - uploadRepository.uploadImage(urlModel.url, imageList[i].url) - .onFailure { - _totalGeneratingState.value = UiState.Failure(it.message.toString()) - } - } - }.awaitAll() - } - - private fun postToGenerateImage() { - viewModelScope.launch { - createRepository.postToCreate( - CreateRequestModel( - prompt.value ?: return@launch, - plusImageS3Key, - imageS3KeyList, - selectedAngle.value ?: return@launch, - selectedCoverage.value ?: return@launch, - selectedRatio.value ?: return@launch, - ), - ) - .onSuccess { - _totalGeneratingState.value = UiState.Success(it) - }.onFailure { - _totalGeneratingState.value = UiState.Failure(it.message.toString()) - } - } - } - - fun resetGeneratingState() { - _totalGeneratingState.value = UiState.Empty - } - - fun getIsGuideNeeded() = userRepository.getIsGuideNeeded() - - fun setIsGuideNeeded(isGuideNeeded: Boolean) = userRepository.setIsGuideNeeded(isGuideNeeded) - - fun getRandomPrompt(): String { - val randomPrompt = promptList[Random.nextInt(promptList.size)] - if (randomPrompt != currentPrompt) currentPrompt = randomPrompt - return currentPrompt - } - - companion object { - val promptList = - listOf( - "보트를 타고 바다 위를 여행하는 모습을 만들어주세요", - "모델이 되어서 패션쇼에서 런웨이를 서는 모습을 만들어주세요", - "원피스를 입고 꽃집에서 꽃을 들고 있는 모습을 만들어주세요", - "놀이공원에서 일본풍의 교복을 입고 서 있는 모습을 만들어주세요", - "정장을 입고 멋진 호텔에 앉아 있는 모습을 만들어주세요", - "콜로세움을 배경으로 로마 여행을 하는 모습을 만들어주세요", - "오토바이를 타고 광활한 사막 위를 달리는 모습을 만들어주세요", - "탑건 영화처럼 전투기 조종사가 되어 하늘에서 비행기를 조종하는 모습을 만들어주세요", - "산 꼭대기 호수 위에서 기념 사진을 찍는 모습을 만들어주세요", - "겨울 설원과 호수를 배경으로 코트를 입고 있는 모습을 만들어주세요", - "파자마를 입고 거실 소파에 편안하게 앉아 있는 모습을 만들어주세요", - "정장을 입고 스튜디오에서 사진을 찍는 모습을 만들어주세요", - "히말라야 정상에 등반한 모습을 만들어주세요", - "날씨 좋은 날 공원에서 멜빵바지를 입고 풍선을 손에 들고 있는 모습을 만들어주세요", - "한밤 중에 공원에서 운동복을 입고 러닝하는 모습을 만들어주세요", - "밤바다에서 가벼운 옷을 입고 산책하는 모습을 만들어주세요", - "파리의 에펠탑 앞에서 편안하게 앉아 있는 모습을 만들어주세요", - "바다 앞에서 원피스를 입고 걸어가는 모습을 만들어주세요", - "도심 야경을 배경으로 스트릿 스타일의 옷을 입고 있는 모습을 만들어주세요", - "한겨울에 목도리를 하고 있는 모습을 만들어주세요", - "드레스를 입고 파티장에 가 있는 모습을 만들어주세요", - "비행기에서 손님을 맞이하는 승무원차림의 모습을 만들어주세요", - "영화관에서 팝콘을 먹으면서 영화를 보는 모습을 만들어주세요", - "눈오는 날 눈사람을 완성한 모습을 만들어주세요", - "일본풍의 교복을 입고 벛꽃 아래에서 학교가는 모습을 만들어주세요", - "분위기 좋은 카페에 앉아 있는 모습을 만들어주세요", - "구름 위에 올라가서 무지개와 함께 있는 모습을 만들어주세요", - "웨딩드레스를 입고 행진하는 모습을 만들어주세요", - "승마복을 입고 말을 타고 있는 모습을 만들어주세요", - "트렌치 코트를 입고 낙옆이 쌓여 있는 거리를 걷고 있는 모습을 만들어주세요", - "이국적인 해변에서 원피스를 입고 걸어가는 사진을 만들어주세요", - "공항에서 캐리어를 끌고 걸어가는 사진을 만들어주세요", - "뉴욕의 타임스퀘어같은 전광판 아래에 서 있는 모습을 만들어주세요", - "기모노를 입고 교토 거리를 걸어가는 모습을 만들어주세요", - "강연장에서 프리젠테이션하고 있는 모습을 만들어주세요", - "겨울 옷을 입고 북극의 오로라와 함께있는 사진을 만들어주세요", - "변호사가 되어 법정에 서 있는 모습을 만들어주세요", - "런던의 빅밴을 배경으로 서 있는 모습을 만들어주세요", - "초원에서 말을 타고 있는 모습을 만들어주세요", - "이집트 피라미드 앞에서 서 있는 모습을 만들어주세요", - "무대 위에서 기타를 연주하는 모습을 만들어주세요", - "수의사가 되어 동물 병원에서 강아지를 진료하는 모습을 만들어주세요", - "제빵사가 되어 빵을 만들고 있는 모습을 만들어주세요", - "만개한 유채꽃밭에서 화려한 드레스를 입고 있는 모습을 만들어주세요", - "정장을 입고 공원에 앉아서 쉬고 있는 모습을 만들어주세요", - "강아지를 안고 있는 모습으로 만들어주세요", - "캠핑카 앞에 앉아서 캠핑을 즐기는 모습을 만들어주세요", - "정장을 입고 연설하는 모습을 만들어주세요", - "군복을 입고 있는 모습을 만들어주세요", - "경찰이 된 내 모습을 만들어주세요", - "청량한 바닷가에서 반팔과 반바지를 입고 있는 모습을 만들어주세요", - "고급스러운 레스토랑에서 식사하고 있는 모습을 만들어주세요", - "해변가에서 자전거를 타는 모습을 만들어주세요", - "꽃집 사장님이 된 모습을 만들어주세요", - "바닷가를 달리고 있는 모습을 만들어주세요", - "운동장에서 열심히 뛰고 있는 모습을 만들어주세요", - "마법학교에 입학한 마법사가 된 모습을 만들어주세요", - "아프리카 초원에서 사자와 함께 뛰고 있는 모습을 만들어주세요", - "크로스핏하는 모습을 만들어주세요", - "비행기 퍼스트 클래스를 타고 있는 모습을 만들어주세요", - "카페에서 커피 만드는 모습을 만들어주세요", - "잔디밭 위에서 디즈니 공주들이 입는 드레스를 입고 있는 모습을 만들어주세요", - "도서관에서 후드티를 입고 공부하는 모습을 만들어주세요", - "밤하늘에 반짝이는 별들 아래에서 앉아있는 사진을 만들어주세요", - "황혼이 짙어지는 사막에서 걸터앉아 멀리 바라보는 모습을 만들어주세요", - "번화가에서 커피를 들고 있는 사진을 만들어주세요", - "고요한 호수 위에 있는 작은 보트 안에서 비옷을 입고 있는 모습을 만들어주세요", - "황금빛 들녘에서 터틀넥 니트를 입고, 햇볕을 받고 있는 모습을 만들어주세요", - "가을 숲에서 단풍을 바라보는 모습을 만들어주세요", - "꽃이 만발한 정원에서 민소매 원피스를 입고, 꽃을 따고 있는 모습을 만들어주세요", - "도심 골목에서 가죽 재킷을 입고 벽에 기대어 있는 사진을 만들어주세요", - "늦은 가을 날씨에 니트와 스커트를 입고 단풍길을 걷는 사진을 만들어주세요", - "어두운 숲 속에서 검은색 망토를 입고있는 마법사의 모습을 만들어주세요", - "화창한 날씨에 티셔츠와 청바지를 입고 공원에서 피크닉을 즐기는 모습을 만들어주세요", - "빙하 위에서 털달린 점퍼를 입고 당당히 서 있는 모습을 만들어주세요", - "블라우스와 청바지를 입고 강가를 걷는 모습을 만들어주세요", - "해적 선박 위에서 항해하는 모습을 만들어주세요", - "카니발 축제에서 화려한 공연복을 입고 춤추는 모습을 만들어주세요", - "판타지 세계에서 화려한 마법 로브를 입고 마법진을 그리는 모습을 만들어주세요", - "고대 신전에서 황금색 로브를 입고 앉아 있는 모습을 만들어주세요", - "정글 속에서 군복을 입고 숨어있는 모습을 만들어주세요", - "중세시대 성에서 화려한 왕관과 드레스를 입고 공주처럼 앉아 있는 모습을 만들어주세요", - "중국 전통복을 입고 만리장성을 배경으로 사진을 찍고 있는 모습을 만들어주세요", - "이스탄불에서 터키 전통 의상을 입고 역사적인 건물을 구경하는 모습을 만들어주세요.", - "앙코르와트에서 탐험복을 입고 거대한 석조 건축물을 배경으로 사진을 찍는 모습을 만들어주세요", - "한 겨울 자작나무 숲에서 모피 재킷을 입고 산책을 즐기며 자연을 만끽하는 모습을 만들어주세요", - "클럽에서 원피스를 입고 춤추는 모습을 만들어주세요", - "가을의 따뜻한 햇빛이 느껴지는 날씨에, 갈대 밭 속에서 서있는 사진을 만들고 싶어요.", - "저격을 준비하고 있는 스나이퍼가 된 모습을 만들고 싶어요", - ) - } - } diff --git a/presentation/src/main/java/kr/genti/presentation/main/create/DefineFragment.kt b/presentation/src/main/java/kr/genti/presentation/main/create/DefineFragment.kt deleted file mode 100644 index d85d4f95..00000000 --- a/presentation/src/main/java/kr/genti/presentation/main/create/DefineFragment.kt +++ /dev/null @@ -1,157 +0,0 @@ -package kr.genti.presentation.main.create - -import android.app.Activity -import android.content.Intent -import android.net.Uri -import android.os.Bundle -import android.view.View -import androidx.activity.result.ActivityResultLauncher -import androidx.activity.result.PickVisualMediaRequest -import androidx.activity.result.contract.ActivityResultContracts -import androidx.activity.result.contract.ActivityResultContracts.PickVisualMedia.Companion.isPhotoPickerAvailable -import androidx.core.view.isVisible -import androidx.fragment.app.activityViewModels -import androidx.navigation.fragment.findNavController -import coil.load -import dagger.hilt.android.AndroidEntryPoint -import kr.genti.core.base.BaseFragment -import kr.genti.core.extension.getFileName -import kr.genti.core.extension.initOnBackPressedListener -import kr.genti.core.extension.setOnSingleClickListener -import kr.genti.core.extension.toast -import kr.genti.domain.entity.response.ImageFileModel -import kr.genti.domain.entity.response.ImageFileModel.Companion.emptyImageFileModel -import kr.genti.presentation.R -import kr.genti.presentation.databinding.FragmentDefineBinding -import kr.genti.presentation.util.AmplitudeManager -import kr.genti.presentation.util.AmplitudeManager.EVENT_CLICK_BTN -import kr.genti.presentation.util.AmplitudeManager.PROPERTY_BTN -import kr.genti.presentation.util.AmplitudeManager.PROPERTY_PAGE - -@AndroidEntryPoint -class DefineFragment() : BaseFragment(R.layout.fragment_define) { - private val viewModel by activityViewModels() - private lateinit var photoPickerResult: ActivityResultLauncher - private lateinit var galleryPickerResult: ActivityResultLauncher - - override fun onViewCreated( - view: View, - savedInstanceState: Bundle?, - ) { - super.onViewCreated(view, savedInstanceState) - - initView() - initCreateBtnListener() - initRefreshExBtnListener() - initAddImageBtnListener() - initDeleteBtnListener() - setGalleryImageWithPhotoPicker() - setGalleryImageWithGalleryPicker() - } - - override fun onResume() { - super.onResume() - setSavedImage() - } - - private fun initView() { - binding.vm = viewModel - initOnBackPressedListener(binding.root) - binding.tvCreateRandomExample.text = viewModel.getRandomPrompt() - } - - private fun initCreateBtnListener() { - binding.btnCreateNext.setOnSingleClickListener { - AmplitudeManager.trackEvent( - EVENT_CLICK_BTN, - mapOf(PROPERTY_PAGE to "create1"), - mapOf(PROPERTY_BTN to "next"), - ) - findNavController().navigate(R.id.action_define_to_pose) - viewModel.modCurrentPercent(33) - } - } - - private fun initRefreshExBtnListener() { - binding.btnRefresh.setOnClickListener { - AmplitudeManager.apply { - trackEvent( - EVENT_CLICK_BTN, - mapOf(PROPERTY_PAGE to "create1"), - mapOf(PROPERTY_BTN to "promptsuggest_refresh"), - ) - plusIntProperties("user_promptsuggest_refresh") - } - binding.tvCreateRandomExample.text = viewModel.getRandomPrompt() - } - } - - private fun initAddImageBtnListener() { - with(binding) { - btnCreatePlus.setOnSingleClickListener { checkAndGetImages() } - layoutPlusImage.setOnSingleClickListener { checkAndGetImages() } - } - } - - private fun initDeleteBtnListener() { - binding.btnDeleteImage.setOnSingleClickListener { - viewModel.plusImage = emptyImageFileModel() - binding.layoutPlusImage.isVisible = false - binding.btnDeleteImage.isVisible = false - } - } - - private fun checkAndGetImages() { - if (isPhotoPickerAvailable(requireContext())) { - photoPickerResult.launch(PickVisualMediaRequest(ActivityResultContracts.PickVisualMedia.ImageOnly)) - } else { - galleryPickerResult.launch( - Intent(Intent.ACTION_PICK).apply { type = "image/*" }, - ) - } - } - - private fun setGalleryImageWithPhotoPicker() { - photoPickerResult = - registerForActivityResult(ActivityResultContracts.PickVisualMedia()) { uri -> - if (uri != null) setImageWithUri(uri) - } - } - - private fun setGalleryImageWithGalleryPicker() { - galleryPickerResult = - registerForActivityResult( - ActivityResultContracts.StartActivityForResult(), - ) { result -> - when (result.resultCode) { - Activity.RESULT_OK -> result.data?.data?.let { setImageWithUri(it) } - Activity.RESULT_CANCELED -> return@registerForActivityResult - else -> toast(getString(R.string.selfie_toast_picker_error)) - } - } - } - - private fun setImageWithUri(uri: Uri) { - viewModel.plusImage = - ImageFileModel( - uri.hashCode().toLong(), - uri.getFileName(requireActivity().contentResolver).toString(), - uri.toString(), - ) - with(binding) { - ivAddedImage.load(uri) - layoutPlusImage.isVisible = true - btnDeleteImage.isVisible = true - } - } - - private fun setSavedImage() { - if (viewModel.plusImage.id != (-1).toLong()) { - with(binding) { - ivAddedImage.load(viewModel.plusImage.url) - layoutPlusImage.isVisible = true - btnDeleteImage.isVisible = true - } - } - } -} diff --git a/presentation/src/main/java/kr/genti/presentation/main/feed/FeedFragment.kt b/presentation/src/main/java/kr/genti/presentation/main/feed/FeedFragment.kt index 4fde9c20..cf90a603 100644 --- a/presentation/src/main/java/kr/genti/presentation/main/feed/FeedFragment.kt +++ b/presentation/src/main/java/kr/genti/presentation/main/feed/FeedFragment.kt @@ -1,7 +1,5 @@ package kr.genti.presentation.main.feed -import android.content.Intent -import android.net.Uri import android.os.Bundle import android.view.View import androidx.core.view.isVisible @@ -15,11 +13,12 @@ import kotlinx.coroutines.flow.distinctUntilChanged import kotlinx.coroutines.flow.launchIn import kotlinx.coroutines.flow.onEach import kr.genti.core.base.BaseFragment -import kr.genti.core.extension.initOnBackPressedListener -import kr.genti.core.extension.setStatusBarColor +import kr.genti.core.extension.dpToPx +import kr.genti.core.extension.setOnSingleClickListener import kr.genti.core.extension.stringOf import kr.genti.core.extension.toast import kr.genti.core.state.UiState +import kr.genti.core.util.RvItemLastDecoration import kr.genti.presentation.R import kr.genti.presentation.databinding.FragmentFeedBinding import kr.genti.presentation.util.AmplitudeManager @@ -31,6 +30,8 @@ class FeedFragment() : BaseFragment(R.layout.fragment_feed) val adapter get() = requireNotNull(_adapter) { getString(R.string.adapter_not_initialized_error_msg) } + private var feedInfoBottomSheet: FeedInfoBottomSheet? = null + private val viewModel by activityViewModels() override fun onViewCreated( @@ -41,56 +42,63 @@ class FeedFragment() : BaseFragment(R.layout.fragment_feed) initView() initAdapter() - setLightningVisibility() + initTooltipCloseBtnListener() + initListScrollListener() observeGetExampleItemsState() } private fun initView() { - initOnBackPressedListener(binding.root) - setStatusBarColor(R.color.background_50) viewModel.getExamplePromptsFromServer() } private fun initAdapter() { - _adapter = - FeedAdapter( - genBtnClick = ::initGenBtnListener, - ) + _adapter = FeedAdapter(genBtnClick = ::initGenBtnListener) binding.rvFeed.adapter = adapter } private fun initGenBtnListener(x: Boolean) { - Intent(Intent.ACTION_VIEW, Uri.parse(WEB_GENFLUENCER)).apply { - startActivity(this) + feedInfoBottomSheet = FeedInfoBottomSheet() + feedInfoBottomSheet?.show(parentFragmentManager, BOTTOM_SHEET_INFO) + } + + private fun initTooltipCloseBtnListener() { + binding.tvFeedTooltip.setOnSingleClickListener { + binding.tvFeedTooltip.isVisible = false + viewModel.isTooltipClosed = true } } - private fun setLightningVisibility() { - with(binding) { - rvFeed.addOnScrollListener( - object : RecyclerView.OnScrollListener() { - var accumScrollY = 0 - - override fun onScrolled( - recyclerView: RecyclerView, - dx: Int, - dy: Int, - ) { - super.onScrolled(recyclerView, dx, dy) - accumScrollY += dy - ivFeedLightning.alpha = - max(0.0, (1 - accumScrollY / 130f).toDouble()).toFloat() - if (accumScrollY > 4500 && !viewModel.isAmplitudeScrollTracked) { - AmplitudeManager.apply { - trackEvent("scroll_main_3pic") - plusIntProperties("user_main_scroll") - } - viewModel.isAmplitudeScrollTracked = true + private fun initListScrollListener() { + binding.rvFeed.addOnScrollListener( + object : RecyclerView.OnScrollListener() { + var accumScrollY = 0 + + override fun onScrolled( + recyclerView: RecyclerView, + dx: Int, + dy: Int, + ) { + super.onScrolled(recyclerView, dx, dy) + + accumScrollY += dy + + if (!viewModel.isTooltipClosed && accumScrollY > 250) { + binding.tvFeedTooltip.isVisible = true + } + + if (accumScrollY > 4500 && !viewModel.isAmplitudeScrollTracked) { + AmplitudeManager.apply { + trackEvent("scroll_main_3pic") + plusIntProperties("user_main_scroll") } + viewModel.isAmplitudeScrollTracked = true } - }, - ) - } + + binding.ivFeedBlurTop.alpha = + 1 - max(0.0, (1 - accumScrollY / 300f).toDouble()).toFloat() + } + } + ) } private fun observeGetExampleItemsState() { @@ -100,11 +108,18 @@ class FeedFragment() : BaseFragment(R.layout.fragment_feed) is UiState.Success -> { adapter.setItemList(state.data) delay(500) - setStatusBarColor(R.color.background_white) with(binding) { layoutLoading.isVisible = false rvFeed.isVisible = true - ivFeedLightning.isVisible = true + rvFeed.addItemDecoration( + RvItemLastDecoration( + requireContext(), + 0, + 0, + 0, + 70.dpToPx(requireContext()) + ) + ) } } @@ -116,11 +131,16 @@ class FeedFragment() : BaseFragment(R.layout.fragment_feed) override fun onDestroyView() { super.onDestroyView() + _adapter = null + feedInfoBottomSheet = null + } + + fun scrollFeedListToTop() { + binding.rvFeed.smoothScrollToPosition(0) } companion object { - private const val WEB_GENFLUENCER = - "https://stealth-goose-156.notion.site/57a00e1d610b4c1786c6ab1fdb4c4659?pvs=4" + private const val BOTTOM_SHEET_INFO = "BOTTOM_SHEET_INFO" } } diff --git a/presentation/src/main/java/kr/genti/presentation/main/feed/FeedInfoBottomSheet.kt b/presentation/src/main/java/kr/genti/presentation/main/feed/FeedInfoBottomSheet.kt new file mode 100644 index 00000000..d4583689 --- /dev/null +++ b/presentation/src/main/java/kr/genti/presentation/main/feed/FeedInfoBottomSheet.kt @@ -0,0 +1,43 @@ +package kr.genti.presentation.main.feed + +import android.content.Intent +import android.net.Uri +import android.os.Bundle +import android.view.View +import dagger.hilt.android.AndroidEntryPoint +import kr.genti.core.base.BaseBottomSheet +import kr.genti.core.extension.setOnSingleClickListener +import kr.genti.presentation.R +import kr.genti.presentation.databinding.BottomSheetFeedInfoBinding + +@AndroidEntryPoint +class FeedInfoBottomSheet : + BaseBottomSheet(R.layout.bottom_sheet_feed_info) { + + override fun onStart() { + super.onStart() + dialog?.window?.setBackgroundDrawableResource(R.color.transparent) + } + + override fun onViewCreated( + view: View, + savedInstanceState: Bundle?, + ) { + super.onViewCreated(view, savedInstanceState) + + initMoreBtnListener() + + } + + private fun initMoreBtnListener() { + binding.btnMoreInfo.setOnSingleClickListener { + startActivity(Intent(Intent.ACTION_VIEW, Uri.parse(WEB_GENFLUENCER))) + dismiss() + } + } + + companion object { + private const val WEB_GENFLUENCER = + "https://stealth-goose-156.notion.site/57a00e1d610b4c1786c6ab1fdb4c4659?pvs=4" + } +} \ No newline at end of file diff --git a/presentation/src/main/java/kr/genti/presentation/main/feed/FeedItemViewHolder.kt b/presentation/src/main/java/kr/genti/presentation/main/feed/FeedItemViewHolder.kt index 8f79481d..07180e19 100644 --- a/presentation/src/main/java/kr/genti/presentation/main/feed/FeedItemViewHolder.kt +++ b/presentation/src/main/java/kr/genti/presentation/main/feed/FeedItemViewHolder.kt @@ -1,8 +1,10 @@ package kr.genti.presentation.main.feed import androidx.constraintlayout.widget.ConstraintLayout +import androidx.core.view.isVisible import androidx.recyclerview.widget.RecyclerView import coil.load +import kr.genti.core.extension.breakLines import kr.genti.domain.entity.response.FeedItemModel import kr.genti.domain.enums.PictureRatio import kr.genti.presentation.databinding.ItemFeedItemBinding @@ -13,15 +15,21 @@ class FeedItemViewHolder( RecyclerView.ViewHolder(binding.root) { fun onBind(item: FeedItemModel) { with(binding) { - tvFeedItemDescription.text = item.prompt + tvFeedItemDescription.text = item.prompt.breakLines() if (item.picture.pictureRatio == PictureRatio.RATIO_GARO) { - (cvFeedItemImage.layoutParams as ConstraintLayout.LayoutParams).dimensionRatio = + (ivFeedItemImage.layoutParams as ConstraintLayout.LayoutParams).dimensionRatio = "3:2" } else { - (cvFeedItemImage.layoutParams as ConstraintLayout.LayoutParams).dimensionRatio = + (ivFeedItemImage.layoutParams as ConstraintLayout.LayoutParams).dimensionRatio = "2:3" } - ivFeedItemImage.load(item.picture.url) + ivFeedItemImage.load(item.picture.url) { + listener( + onSuccess = { _, _ -> + binding.lottieLoadingImage.isVisible = false + } + ) + } } } } diff --git a/presentation/src/main/java/kr/genti/presentation/main/feed/FeedViewModel.kt b/presentation/src/main/java/kr/genti/presentation/main/feed/FeedViewModel.kt index 30b06a26..a11feed6 100644 --- a/presentation/src/main/java/kr/genti/presentation/main/feed/FeedViewModel.kt +++ b/presentation/src/main/java/kr/genti/presentation/main/feed/FeedViewModel.kt @@ -22,6 +22,7 @@ class FeedViewModel val getExampleItemsState: StateFlow>> = _getExampleItemsState var isAmplitudeScrollTracked = false + var isTooltipClosed = false fun getExamplePromptsFromServer() { viewModelScope.launch { diff --git a/presentation/src/main/java/kr/genti/presentation/main/profile/ProfileAdapter.kt b/presentation/src/main/java/kr/genti/presentation/main/profile/ProfileAdapter.kt index 39f325e3..508c0492 100644 --- a/presentation/src/main/java/kr/genti/presentation/main/profile/ProfileAdapter.kt +++ b/presentation/src/main/java/kr/genti/presentation/main/profile/ProfileAdapter.kt @@ -3,29 +3,54 @@ package kr.genti.presentation.main.profile import android.view.LayoutInflater import android.view.ViewGroup import androidx.recyclerview.widget.ListAdapter +import androidx.recyclerview.widget.RecyclerView import kr.genti.core.util.ItemDiffCallback import kr.genti.domain.entity.response.ImageModel import kr.genti.presentation.databinding.ItemProfileImageBinding class ProfileAdapter( private val imageClick: (ImageModel) -> Unit, -) : ListAdapter(diffUtil) { + private val moveClick: (Boolean) -> Unit +) : ListAdapter(diffUtil) { + + var isMaking: Boolean = false + set(value) { + field = value + notifyItemChanged(currentList.size) + } + + override fun getItemViewType(position: Int): Int { + return if (position == currentList.size) VIEW_TYPE_FOOTER else VIEW_TYPE_ITEM + } + override fun onCreateViewHolder( parent: ViewGroup, viewType: Int, - ): ProfileViewHolder { - val inflater by lazy { LayoutInflater.from(parent.context) } - val binding: ItemProfileImageBinding = - ItemProfileImageBinding.inflate(inflater, parent, false) - return ProfileViewHolder(binding, imageClick) + ): RecyclerView.ViewHolder { + val inflater = LayoutInflater.from(parent.context) + val binding = ItemProfileImageBinding.inflate(inflater, parent, false) + + return when (viewType) { + VIEW_TYPE_ITEM -> ProfileItemViewHolder(binding, imageClick) + VIEW_TYPE_FOOTER -> ProfileMoveViewHolder(binding, moveClick) + else -> throw IllegalArgumentException("Invalid view type") + } } override fun onBindViewHolder( - holder: ProfileViewHolder, + holder: RecyclerView.ViewHolder, position: Int, ) { - val item = getItem(position) ?: return - holder.onBind(item) + if (holder is ProfileItemViewHolder) { + val item = getItem(position) ?: return + holder.onBind(item) + } else if (holder is ProfileMoveViewHolder) { + holder.onBind(isMaking) + } + } + + override fun getItemCount(): Int { + return if (currentList.isNotEmpty()) currentList.size + 1 else currentList.size } fun addList(newItems: List) { @@ -35,6 +60,9 @@ class ProfileAdapter( } companion object { + private const val VIEW_TYPE_ITEM = 0 + private const val VIEW_TYPE_FOOTER = 1 + private val diffUtil = ItemDiffCallback( onItemsTheSame = { old, new -> old.id == new.id }, diff --git a/presentation/src/main/java/kr/genti/presentation/main/profile/ProfileFragment.kt b/presentation/src/main/java/kr/genti/presentation/main/profile/ProfileFragment.kt index d6859d39..995836c0 100644 --- a/presentation/src/main/java/kr/genti/presentation/main/profile/ProfileFragment.kt +++ b/presentation/src/main/java/kr/genti/presentation/main/profile/ProfileFragment.kt @@ -14,15 +14,19 @@ import kotlinx.coroutines.flow.distinctUntilChanged import kotlinx.coroutines.flow.launchIn import kotlinx.coroutines.flow.onEach import kr.genti.core.base.BaseFragment -import kr.genti.core.extension.initOnBackPressedListener import kr.genti.core.extension.setOnSingleClickListener -import kr.genti.core.extension.setStatusBarColor import kr.genti.core.extension.stringOf import kr.genti.core.extension.toast import kr.genti.core.state.UiState import kr.genti.domain.entity.response.ImageModel +import kr.genti.domain.enums.GenerateStatus import kr.genti.presentation.R import kr.genti.presentation.databinding.FragmentProfileBinding +import kr.genti.presentation.generate.waiting.WaitingActivity +import kr.genti.presentation.main.CreateErrorDialog +import kr.genti.presentation.main.CreateFinishedDialog +import kr.genti.presentation.main.CreateSelectDialog +import kr.genti.presentation.main.CreateUnableDialog import kr.genti.presentation.setting.SettingActivity import kr.genti.presentation.util.AmplitudeManager @@ -35,6 +39,10 @@ class ProfileFragment() : BaseFragment(R.layout.fragment private val viewModel by activityViewModels() private var profileImageDialog: ProfileImageDialog? = null + private var createFinishedDialog: CreateFinishedDialog? = null + private var createErrorDialog: CreateErrorDialog? = null + private var createUnableDialog: CreateUnableDialog? = null + private var createSelectDialog: CreateSelectDialog? = null override fun onViewCreated( view: View, @@ -48,40 +56,68 @@ class ProfileFragment() : BaseFragment(R.layout.fragment setListWithInfinityScroll() observeGenerateStatus() observePictureListPageState() + observeServerAvailableState() } private fun initView() { - initOnBackPressedListener(binding.root) - setStatusBarColor(R.color.green_3) with(viewModel) { getGenerateStatusFromServer() + if(_adapter != null) adapter.submitList(listOf()) + resetPicturePagingValue() getPictureListFromServer() } } private fun initSettingBtnListener() { binding.btnSetting.setOnSingleClickListener { - Intent(requireActivity(), SettingActivity::class.java).apply { - startActivity(this) - } + startActivity(Intent(requireActivity(), SettingActivity::class.java)) } } private fun initAdapter() { - _adapter = - ProfileAdapter( - imageClick = ::initImageClickListener, - ) + _adapter = ProfileAdapter( + imageClick = ::initImageClickListener, + moveClick = ::initMoveClickListener, + ) binding.rvProfilePictureList.adapter = adapter } private fun initImageClickListener(item: ImageModel) { AmplitudeManager.trackEvent("enlarge_mypage_picture") profileImageDialog = - ProfileImageDialog.newInstance(item.id, item.url, item.pictureRatio?.name ?: "") + ProfileImageDialog.newInstance(item.id, item.url, item.pictureRatio?.name.orEmpty()) profileImageDialog?.show(parentFragmentManager, IMAGE_VIEWER) } + private fun initMoveClickListener(x: Boolean) { + when (viewModel.currentStatus) { + GenerateStatus.NEW_REQUEST_AVAILABLE -> { + viewModel.getIsServerAvailable() + } + + GenerateStatus.AWAIT_USER_VERIFICATION -> { + createFinishedDialog = CreateFinishedDialog() + createFinishedDialog?.show(parentFragmentManager, DIALOG_FINISHED) + } + + GenerateStatus.IN_PROGRESS -> { + startActivity( + WaitingActivity.createIntent( + requireContext(), + viewModel.isCreatingParentPic + ) + ) + } + + GenerateStatus.CANCELED -> { + createErrorDialog = CreateErrorDialog() + createErrorDialog?.show(parentFragmentManager, DIALOG_ERROR) + } + + GenerateStatus.EMPTY -> return + } + } + private fun setListWithInfinityScroll() { binding.rvProfilePictureList.addOnScrollListener( object : RecyclerView.OnScrollListener() { @@ -110,10 +146,8 @@ class ProfileFragment() : BaseFragment(R.layout.fragment viewModel.getGenerateStatusState.flowWithLifecycle(lifecycle).onEach { state -> when (state) { is UiState.Success -> { - with(binding) { - layoutProfileWaiting.isVisible = state.data == true - layoutProfileNormal.isVisible = state.data != true - } + binding.ivProfileMaking.isVisible = state.data == true + _adapter?.isMaking = state.data == true } is UiState.Failure -> toast(stringOf(R.string.error_msg)) @@ -156,13 +190,44 @@ class ProfileFragment() : BaseFragment(R.layout.fragment } } + private fun observeServerAvailableState() { + viewModel.serverAvailableState.flowWithLifecycle(lifecycle).onEach { state -> + when (state) { + is UiState.Success -> { + if (state.data.status) { + AmplitudeManager.trackEvent("click_createpictab") + createSelectDialog = CreateSelectDialog() + createSelectDialog?.show(parentFragmentManager, DIALOG_SELECT) + } else { + createUnableDialog = + CreateUnableDialog.newInstance(state.data.message.orEmpty()) + createUnableDialog?.show(parentFragmentManager, DIALOG_UNABLE) + } + } + + is UiState.Failure -> toast(stringOf(R.string.error_msg)) + else -> return@onEach + } + viewModel.resetIsServerAvailable() + }.launchIn(lifecycleScope) + } + override fun onDestroyView() { super.onDestroyView() _adapter = null profileImageDialog = null + createFinishedDialog = null + createErrorDialog = null + createUnableDialog = null + createSelectDialog = null } companion object { private const val IMAGE_VIEWER = "IMAGE_VIEWER" + + private const val DIALOG_FINISHED = "DIALOG_FINISHED" + private const val DIALOG_ERROR = "DIALOG_ERROR" + private const val DIALOG_UNABLE = "DIALOG_UNABLE" + private const val DIALOG_SELECT = "DIALOG_SELECT" } } diff --git a/presentation/src/main/java/kr/genti/presentation/main/profile/ProfileImageDialog.kt b/presentation/src/main/java/kr/genti/presentation/main/profile/ProfileImageDialog.kt index 9167c94d..d49a83eb 100644 --- a/presentation/src/main/java/kr/genti/presentation/main/profile/ProfileImageDialog.kt +++ b/presentation/src/main/java/kr/genti/presentation/main/profile/ProfileImageDialog.kt @@ -8,15 +8,12 @@ import android.net.Uri import android.os.Bundle import android.view.View import android.view.WindowManager -import androidx.constraintlayout.widget.ConstraintLayout import androidx.core.content.FileProvider import androidx.core.os.bundleOf import coil.load import kr.genti.core.base.BaseDialog import kr.genti.core.extension.setGusianBlur import kr.genti.core.extension.setOnSingleClickListener -import kr.genti.domain.enums.PictureRatio -import kr.genti.domain.enums.PictureRatio.Companion.toPictureRatio import kr.genti.presentation.R import kr.genti.presentation.databinding.DialogProfileImageBinding import kr.genti.presentation.util.AmplitudeManager @@ -35,11 +32,10 @@ class ProfileImageDialog : dialog?.window?.apply { setLayout( WindowManager.LayoutParams.MATCH_PARENT, - WindowManager.LayoutParams.WRAP_CONTENT, + WindowManager.LayoutParams.MATCH_PARENT, ) setBackgroundDrawableResource(R.color.transparent) } - requireActivity().window.decorView.rootView.setGusianBlur(50f) } override fun onViewCreated( @@ -52,7 +48,6 @@ class ProfileImageDialog : initExitBtnListener() initDownloadBtnListener() initShareBtnListener() - setImage() } private fun getBundleArgs() { @@ -60,10 +55,12 @@ class ProfileImageDialog : imageId = arguments?.getLong(ARGS_IMAGE_ID) ?: -1 imageUrl = arguments?.getString(ARGS_IMAGE_URL) ?: "" imageRatio = arguments?.getString(ARGS_IMAGE_RATIO) ?: "" + binding.ivProfile.load(imageUrl) } private fun initExitBtnListener() { binding.btnExit.setOnSingleClickListener { dismiss() } + binding.root.setOnSingleClickListener { dismiss() } } private fun initDownloadBtnListener() { @@ -100,19 +97,6 @@ class ProfileImageDialog : } } - private fun setImage() { - with(binding) { - ivProfile.load(imageUrl) - if (imageRatio.toPictureRatio() == PictureRatio.RATIO_GARO) { - (ivProfile.layoutParams as ConstraintLayout.LayoutParams).dimensionRatio = "3:2" - (ivProfileBg.layoutParams as ConstraintLayout.LayoutParams).dimensionRatio = "3:2" - } else { - (ivProfile.layoutParams as ConstraintLayout.LayoutParams).dimensionRatio = "2:3" - (ivProfileBg.layoutParams as ConstraintLayout.LayoutParams).dimensionRatio = "2:3" - } - } - } - private fun getTemporaryUri(): Uri { // Bitmap 가져온 후 임시 파일로 저장 val tempFile = File(requireActivity().cacheDir, TEMP_FILE_NAME) @@ -131,11 +115,6 @@ class ProfileImageDialog : ) } - override fun onDismiss(dialog: DialogInterface) { - super.onDismiss(dialog) - requireActivity().window.decorView.rootView.setGusianBlur(null) - } - companion object { const val IMAGE_TYPE = "image/*" const val SHARE_IMAGE_CHOOSER = "SHARE_IMAGE_CHOOSER" diff --git a/presentation/src/main/java/kr/genti/presentation/main/profile/ProfileViewHolder.kt b/presentation/src/main/java/kr/genti/presentation/main/profile/ProfileItemViewHolder.kt similarity index 80% rename from presentation/src/main/java/kr/genti/presentation/main/profile/ProfileViewHolder.kt rename to presentation/src/main/java/kr/genti/presentation/main/profile/ProfileItemViewHolder.kt index c87c4584..91932025 100644 --- a/presentation/src/main/java/kr/genti/presentation/main/profile/ProfileViewHolder.kt +++ b/presentation/src/main/java/kr/genti/presentation/main/profile/ProfileItemViewHolder.kt @@ -2,11 +2,10 @@ package kr.genti.presentation.main.profile import androidx.recyclerview.widget.RecyclerView import coil.load -import kr.genti.core.extension.setOnSingleClickListener import kr.genti.domain.entity.response.ImageModel import kr.genti.presentation.databinding.ItemProfileImageBinding -class ProfileViewHolder( +class ProfileItemViewHolder( val binding: ItemProfileImageBinding, val imageClick: (ImageModel) -> Unit, ) : @@ -14,7 +13,7 @@ class ProfileViewHolder( fun onBind(item: ImageModel) { with(binding) { ivProfileItemImage.load(item.url) - root.setOnSingleClickListener { + root.setOnClickListener { imageClick(item) } } diff --git a/presentation/src/main/java/kr/genti/presentation/main/profile/ProfileMoveViewHolder.kt b/presentation/src/main/java/kr/genti/presentation/main/profile/ProfileMoveViewHolder.kt new file mode 100644 index 00000000..c1103751 --- /dev/null +++ b/presentation/src/main/java/kr/genti/presentation/main/profile/ProfileMoveViewHolder.kt @@ -0,0 +1,24 @@ +package kr.genti.presentation.main.profile + +import androidx.recyclerview.widget.RecyclerView +import coil.load +import kr.genti.core.extension.setOnSingleClickListener +import kr.genti.presentation.R +import kr.genti.presentation.databinding.ItemProfileImageBinding + +class ProfileMoveViewHolder( + val binding: ItemProfileImageBinding, + val moveClick: (Boolean) -> Unit, +) : + RecyclerView.ViewHolder(binding.root) { + fun onBind(isMaking: Boolean) { + with(binding) { + if (isMaking) { + ivProfileItemImage.load(R.drawable.img_profile_create_inactive) + } else { + ivProfileItemImage.load(R.drawable.img_profile_create_active) + } + root.setOnSingleClickListener { moveClick(isMaking) } + } + } +} \ No newline at end of file diff --git a/presentation/src/main/java/kr/genti/presentation/main/profile/ProfileViewModel.kt b/presentation/src/main/java/kr/genti/presentation/main/profile/ProfileViewModel.kt index 985419ca..5b3956d4 100644 --- a/presentation/src/main/java/kr/genti/presentation/main/profile/ProfileViewModel.kt +++ b/presentation/src/main/java/kr/genti/presentation/main/profile/ProfileViewModel.kt @@ -8,66 +8,101 @@ import kotlinx.coroutines.flow.StateFlow import kotlinx.coroutines.launch import kr.genti.core.state.UiState import kr.genti.domain.entity.response.PicturePagedListModel +import kr.genti.domain.entity.response.ServerAvailableModel import kr.genti.domain.enums.GenerateStatus import kr.genti.domain.repository.GenerateRepository import javax.inject.Inject @HiltViewModel class ProfileViewModel - @Inject - constructor( - private val generateRepository: GenerateRepository, - ) : ViewModel() { - private val _getGenerateStatusState = - MutableStateFlow>(UiState.Empty) - val getGenerateStatusState: StateFlow> = _getGenerateStatusState +@Inject +constructor( + private val generateRepository: GenerateRepository, +) : ViewModel() { + private val _getGenerateStatusState = + MutableStateFlow>(UiState.Empty) + val getGenerateStatusState: StateFlow> = _getGenerateStatusState - private val _getPictureListState = - MutableStateFlow>(UiState.Loading) - val getPictureListState: StateFlow> = _getPictureListState + var currentStatus: GenerateStatus = GenerateStatus.NEW_REQUEST_AVAILABLE + var isCreatingParentPic: Boolean? = null - private var currentPage = -1 - private var isPagingFinish = false - private var totalPage = Int.MAX_VALUE + private val _serverAvailableState = + MutableStateFlow>(UiState.Empty) + val serverAvailableState: StateFlow> = _serverAvailableState - var isFirstLoading = true + private val _getPictureListState = + MutableStateFlow>(UiState.Loading) + val getPictureListState: StateFlow> = _getPictureListState - fun getGenerateStatusFromServer() { - viewModelScope.launch { - _getGenerateStatusState.value = UiState.Loading - generateRepository.getGenerateStatus() - .onSuccess { - _getGenerateStatusState.value = - UiState.Success(it.status == GenerateStatus.IN_PROGRESS) - } - .onFailure { t -> - _getGenerateStatusState.value = UiState.Failure(t.message.toString()) - } - } + private var currentPage = -1 + private var isPagingFinish = false + private var totalPage = Int.MAX_VALUE + + var isFirstLoading = true + + fun getGenerateStatusFromServer() { + viewModelScope.launch { + _getGenerateStatusState.value = UiState.Loading + generateRepository.getGenerateStatus() + .onSuccess { + currentStatus = it.status + isCreatingParentPic = it.paid + _getGenerateStatusState.value = + UiState.Success(it.status == GenerateStatus.IN_PROGRESS) + + } + .onFailure { t -> + _getGenerateStatusState.value = UiState.Failure(t.message.toString()) + } } + } - fun getPictureListFromServer() { - viewModelScope.launch { - if (isPagingFinish) return@launch - _getPictureListState.value = UiState.Loading - generateRepository.getGeneratedPictureList( - ++currentPage, - 10, - null, - null, - ) - .onSuccess { - totalPage = it.totalPages - 1 - if (totalPage == currentPage) isPagingFinish = true - if (it.content.isEmpty()) { - _getPictureListState.value = UiState.Empty - return@launch - } - if (isFirstLoading) isFirstLoading = false - _getPictureListState.value = UiState.Success(it) - }.onFailure { - _getPictureListState.value = UiState.Failure(it.message.toString()) + fun getIsServerAvailable() { + _serverAvailableState.value = UiState.Loading + viewModelScope.launch { + generateRepository + .getIsServerAvailable() + .onSuccess { + _serverAvailableState.value = UiState.Success(it) + }.onFailure { + _serverAvailableState.value = UiState.Failure(it.message.orEmpty()) + } + } + } + + fun resetIsServerAvailable() { + _serverAvailableState.value = UiState.Empty + } + + fun getPictureListFromServer() { + viewModelScope.launch { + if (isPagingFinish) return@launch + _getPictureListState.value = UiState.Loading + generateRepository.getGeneratedPictureList( + ++currentPage, + 10, + null, + null, + ) + .onSuccess { + totalPage = it.totalPages - 1 + if (totalPage == currentPage) isPagingFinish = true + if (it.content.isEmpty()) { + _getPictureListState.value = UiState.Empty + return@launch } - } + if (isFirstLoading) isFirstLoading = false + _getPictureListState.value = UiState.Success(it) + }.onFailure { + _getPictureListState.value = UiState.Failure(it.message.toString()) + } } } + + fun resetPicturePagingValue() { + currentPage = -1 + isPagingFinish = false + totalPage = Int.MAX_VALUE + isFirstLoading = true + } +} diff --git a/presentation/src/main/java/kr/genti/presentation/setting/SettingActivity.kt b/presentation/src/main/java/kr/genti/presentation/setting/SettingActivity.kt index a2636c83..3f3a7e5a 100644 --- a/presentation/src/main/java/kr/genti/presentation/setting/SettingActivity.kt +++ b/presentation/src/main/java/kr/genti/presentation/setting/SettingActivity.kt @@ -6,7 +6,6 @@ import android.os.Bundle import dagger.hilt.android.AndroidEntryPoint import kr.genti.core.base.BaseActivity import kr.genti.core.extension.setOnSingleClickListener -import kr.genti.core.extension.setStatusBarColorFromResource import kr.genti.presentation.BuildConfig.VERSION_NAME import kr.genti.presentation.R import kr.genti.presentation.databinding.ActivitySettingBinding @@ -19,7 +18,6 @@ class SettingActivity : BaseActivity(R.layout.activity_s override fun onCreate(savedInstanceState: Bundle?) { super.onCreate(savedInstanceState) - initView() initBackBtnListener() initInfoBtnsListener() initLogoutBtnListener() @@ -27,32 +25,23 @@ class SettingActivity : BaseActivity(R.layout.activity_s setAndroidVersion() } - private fun initView() { - setStatusBarColorFromResource(R.color.green_3) - } - private fun initBackBtnListener() { - binding.btnBack.setOnSingleClickListener { - finish() - } + binding.btnBack.setOnSingleClickListener { finish() } } private fun initInfoBtnsListener() { with(binding) { btnTermsOfService.setOnSingleClickListener { - Intent(Intent.ACTION_VIEW, Uri.parse(WEB_TERMS_OF_SERVICE)).apply { - startActivity(this) - } + startActivity(Intent(Intent.ACTION_VIEW, Uri.parse(WEB_TERMS_OF_SERVICE))) } btnPrivacyPolicy.setOnSingleClickListener { - Intent(Intent.ACTION_VIEW, Uri.parse(WEB_PRIVACY_POLICY)).apply { - startActivity(this) - } + startActivity(Intent(Intent.ACTION_VIEW, Uri.parse(WEB_PRIVACY_POLICY))) } btnCompanyInfo.setOnSingleClickListener { - Intent(Intent.ACTION_VIEW, Uri.parse(WEB_COMPANY_INFO)).apply { - startActivity(this) - } + startActivity(Intent(Intent.ACTION_VIEW, Uri.parse(WEB_COMPANY_INFO))) + } + btnQuestion.setOnSingleClickListener { + startActivity(Intent(Intent.ACTION_VIEW, Uri.parse(WEB_QUESTION))) } } } @@ -92,5 +81,7 @@ class SettingActivity : BaseActivity(R.layout.activity_s "https://stealth-goose-156.notion.site/e0f2e17a3a60437b8e62423f61cca2a9?pvs=4" private const val WEB_COMPANY_INFO = "https://stealth-goose-156.notion.site/39d39ae82a3a436fa053e5287ff9742c?pvs=4" + private const val WEB_QUESTION = + "https://stealth-goose-156.notion.site/14537d728f968060b2ffd1c19aff6298?pvs=4" } } diff --git a/presentation/src/main/java/kr/genti/presentation/setting/SettingLogoutDialog.kt b/presentation/src/main/java/kr/genti/presentation/setting/SettingLogoutDialog.kt index b0841ecd..c7156df0 100644 --- a/presentation/src/main/java/kr/genti/presentation/setting/SettingLogoutDialog.kt +++ b/presentation/src/main/java/kr/genti/presentation/setting/SettingLogoutDialog.kt @@ -29,14 +29,11 @@ class SettingLogoutDialog : BaseDialog(R.layout.dial super.onStart() dialog?.window?.apply { setLayout( - WindowManager.LayoutParams.WRAP_CONTENT, + WindowManager.LayoutParams.MATCH_PARENT, WindowManager.LayoutParams.WRAP_CONTENT, ) setBackgroundDrawableResource(R.color.transparent) } - requireActivity() - .window.decorView.rootView - .setGusianBlur(50f) } override fun onViewCreated( @@ -77,11 +74,4 @@ class SettingLogoutDialog : BaseDialog(R.layout.dial } }.launchIn(lifecycleScope) } - - override fun onDismiss(dialog: DialogInterface) { - super.onDismiss(dialog) - requireActivity() - .window.decorView.rootView - .setGusianBlur(null) - } } diff --git a/presentation/src/main/java/kr/genti/presentation/setting/SettingQuitDialog.kt b/presentation/src/main/java/kr/genti/presentation/setting/SettingQuitDialog.kt index bc7d9edf..2197e45d 100644 --- a/presentation/src/main/java/kr/genti/presentation/setting/SettingQuitDialog.kt +++ b/presentation/src/main/java/kr/genti/presentation/setting/SettingQuitDialog.kt @@ -29,14 +29,11 @@ class SettingQuitDialog : BaseDialog(R.layout.dialog_s super.onStart() dialog?.window?.apply { setLayout( - WindowManager.LayoutParams.WRAP_CONTENT, + WindowManager.LayoutParams.MATCH_PARENT, WindowManager.LayoutParams.WRAP_CONTENT, ) setBackgroundDrawableResource(R.color.transparent) } - requireActivity() - .window.decorView.rootView - .setGusianBlur(50f) } override fun onViewCreated( @@ -77,11 +74,4 @@ class SettingQuitDialog : BaseDialog(R.layout.dialog_s } }.launchIn(lifecycleScope) } - - override fun onDismiss(dialog: DialogInterface) { - super.onDismiss(dialog) - requireActivity() - .window.decorView.rootView - .setGusianBlur(null) - } } diff --git a/presentation/src/main/java/kr/genti/presentation/util/GlideResultListener.kt b/presentation/src/main/java/kr/genti/presentation/util/GlideResultListener.kt new file mode 100644 index 00000000..cea3c955 --- /dev/null +++ b/presentation/src/main/java/kr/genti/presentation/util/GlideResultListener.kt @@ -0,0 +1,32 @@ +package kr.genti.presentation.util + +import android.graphics.drawable.Drawable +import com.bumptech.glide.load.DataSource +import com.bumptech.glide.load.engine.GlideException +import com.bumptech.glide.request.RequestListener +import com.bumptech.glide.request.target.Target + +class GlideResultListener( + private val onComplete: () -> Unit +) : RequestListener { + override fun onLoadFailed( + e: GlideException?, + model: Any?, + target: Target?, + isFirstResource: Boolean + ): Boolean { + onComplete() + return false + } + + override fun onResourceReady( + resource: Drawable?, + model: Any?, + target: Target?, + dataSource: DataSource?, + isFirstResource: Boolean + ): Boolean { + onComplete() + return false + } +} \ No newline at end of file diff --git a/presentation/src/main/res/drawable/ic_add.xml b/presentation/src/main/res/drawable/ic_add.xml deleted file mode 100644 index 550480bd..00000000 --- a/presentation/src/main/res/drawable/ic_add.xml +++ /dev/null @@ -1,22 +0,0 @@ - - - - - diff --git a/presentation/src/main/res/drawable/ic_axe.png b/presentation/src/main/res/drawable/ic_axe.png deleted file mode 100644 index 390a1e01..00000000 Binary files a/presentation/src/main/res/drawable/ic_axe.png and /dev/null differ diff --git a/presentation/src/main/res/drawable/ic_back.xml b/presentation/src/main/res/drawable/ic_back.xml index e712ca40..54d8f172 100644 --- a/presentation/src/main/res/drawable/ic_back.xml +++ b/presentation/src/main/res/drawable/ic_back.xml @@ -1,18 +1,15 @@ - - + android:width="24dp" + android:height="25dp" + android:viewportWidth="24" + android:viewportHeight="25"> + + + + diff --git a/presentation/src/main/res/drawable/ic_big_lightning.png b/presentation/src/main/res/drawable/ic_big_lightning.png deleted file mode 100644 index 121d9e9b..00000000 Binary files a/presentation/src/main/res/drawable/ic_big_lightning.png and /dev/null differ diff --git a/presentation/src/main/res/drawable/ic_blank.xml b/presentation/src/main/res/drawable/ic_blank.xml new file mode 100644 index 00000000..dc764d7e --- /dev/null +++ b/presentation/src/main/res/drawable/ic_blank.xml @@ -0,0 +1,29 @@ + + + + + + + + + diff --git a/presentation/src/main/res/drawable/ic_check.xml b/presentation/src/main/res/drawable/ic_check.xml new file mode 100644 index 00000000..e8f62be3 --- /dev/null +++ b/presentation/src/main/res/drawable/ic_check.xml @@ -0,0 +1,13 @@ + + + + + + diff --git a/presentation/src/main/res/drawable/ic_check_checked.xml b/presentation/src/main/res/drawable/ic_check_checked.xml deleted file mode 100644 index be9073d7..00000000 --- a/presentation/src/main/res/drawable/ic_check_checked.xml +++ /dev/null @@ -1,12 +0,0 @@ - - - - diff --git a/presentation/src/main/res/drawable/ic_check_selected.xml b/presentation/src/main/res/drawable/ic_check_selected.xml deleted file mode 100644 index 65d3889d..00000000 --- a/presentation/src/main/res/drawable/ic_check_selected.xml +++ /dev/null @@ -1,19 +0,0 @@ - - - - diff --git a/presentation/src/main/res/drawable/ic_check_unchecked.xml b/presentation/src/main/res/drawable/ic_check_unchecked.xml deleted file mode 100644 index 49cb1b28..00000000 --- a/presentation/src/main/res/drawable/ic_check_unchecked.xml +++ /dev/null @@ -1,14 +0,0 @@ - - - - diff --git a/presentation/src/main/res/drawable/ic_check_unselected.xml b/presentation/src/main/res/drawable/ic_check_unselected.xml deleted file mode 100644 index 4ca3e2f5..00000000 --- a/presentation/src/main/res/drawable/ic_check_unselected.xml +++ /dev/null @@ -1,19 +0,0 @@ - - - - diff --git a/presentation/src/main/res/drawable/ic_close.xml b/presentation/src/main/res/drawable/ic_close.xml index d349483e..78545e31 100644 --- a/presentation/src/main/res/drawable/ic_close.xml +++ b/presentation/src/main/res/drawable/ic_close.xml @@ -6,5 +6,5 @@ + android:fillAlpha="0.40"/> diff --git a/presentation/src/main/res/drawable/ic_delete.xml b/presentation/src/main/res/drawable/ic_delete.xml deleted file mode 100644 index 08b72f7e..00000000 --- a/presentation/src/main/res/drawable/ic_delete.xml +++ /dev/null @@ -1,17 +0,0 @@ - - - - diff --git a/presentation/src/main/res/drawable/ic_download.xml b/presentation/src/main/res/drawable/ic_download.xml index d46316b0..1af6aecd 100644 --- a/presentation/src/main/res/drawable/ic_download.xml +++ b/presentation/src/main/res/drawable/ic_download.xml @@ -1,28 +1,18 @@ + android:width="32dp" + android:height="33dp" + android:viewportWidth="32" + android:viewportHeight="33"> + android:pathData="M16,0.315L16,0.315A16,16 0,0 1,32 16.315L32,16.315A16,16 0,0 1,16 32.315L16,32.315A16,16 0,0 1,0 16.315L0,16.315A16,16 0,0 1,16 0.315z" + android:fillColor="#49F155"/> + android:pathData="M8.727,22.861C8.727,23.664 9.379,24.315 10.182,24.315H21.818C22.622,24.315 23.273,23.664 23.273,22.861C23.273,22.058 22.622,21.406 21.818,21.406H10.182C9.379,21.406 8.727,22.058 8.727,22.861ZM16,19.952L20.792,15.16C21.359,14.593 21.359,13.674 20.792,13.107C20.232,12.547 19.325,12.54 18.756,13.091L17.455,14.352V9.77C17.455,8.967 16.803,8.315 16,8.315C15.197,8.315 14.545,8.967 14.545,9.77V14.352L13.244,13.091C12.675,12.54 11.769,12.547 11.208,13.107C10.641,13.674 10.641,14.593 11.208,15.16L16,19.952Z" + android:fillColor="#030F0F"/> + android:pathData="M16,19.952L20.792,15.16C21.359,14.593 21.359,13.674 20.792,13.107C20.232,12.547 19.325,12.54 18.756,13.091L17.455,14.352V9.77C17.455,8.967 16.803,8.315 16,8.315C15.197,8.315 14.545,8.967 14.545,9.77V14.352L13.244,13.091C12.675,12.54 11.769,12.547 11.208,13.107C10.641,13.674 10.641,14.593 11.208,15.16L16,19.952Z" + android:fillColor="#030F0F"/> + android:pathData="M8.727,22.861C8.727,23.664 9.379,24.315 10.182,24.315H21.818C22.622,24.315 23.273,23.664 23.273,22.861C23.273,22.058 22.622,21.406 21.818,21.406H10.182C9.379,21.406 8.727,22.058 8.727,22.861Z" + android:fillColor="#030F0F"/> diff --git a/presentation/src/main/res/drawable/ic_download_no_border.xml b/presentation/src/main/res/drawable/ic_download_no_border.xml new file mode 100644 index 00000000..97682cdc --- /dev/null +++ b/presentation/src/main/res/drawable/ic_download_no_border.xml @@ -0,0 +1,15 @@ + + + + + diff --git a/presentation/src/main/res/drawable/ic_error_icon.png b/presentation/src/main/res/drawable/ic_error_icon.png deleted file mode 100644 index 400e1fcd..00000000 Binary files a/presentation/src/main/res/drawable/ic_error_icon.png and /dev/null differ diff --git a/presentation/src/main/res/drawable/ic_exit.xml b/presentation/src/main/res/drawable/ic_exit.xml deleted file mode 100644 index 4d48caa7..00000000 --- a/presentation/src/main/res/drawable/ic_exit.xml +++ /dev/null @@ -1,19 +0,0 @@ - - - - diff --git a/presentation/src/main/res/drawable/ic_female_gray.xml b/presentation/src/main/res/drawable/ic_female_gray.xml new file mode 100644 index 00000000..856de760 --- /dev/null +++ b/presentation/src/main/res/drawable/ic_female_gray.xml @@ -0,0 +1,10 @@ + + + diff --git a/presentation/src/main/res/drawable/ic_female_green.xml b/presentation/src/main/res/drawable/ic_female_green.xml new file mode 100644 index 00000000..3ea4f98f --- /dev/null +++ b/presentation/src/main/res/drawable/ic_female_green.xml @@ -0,0 +1,9 @@ + + + diff --git a/presentation/src/main/res/drawable/ic_indicator_green.xml b/presentation/src/main/res/drawable/ic_indicator_green.xml index db608fc9..49bf08a5 100644 --- a/presentation/src/main/res/drawable/ic_indicator_green.xml +++ b/presentation/src/main/res/drawable/ic_indicator_green.xml @@ -1,9 +1,9 @@ \ No newline at end of file diff --git a/presentation/src/main/res/drawable/ic_indicator_grey5.xml b/presentation/src/main/res/drawable/ic_indicator_grey5.xml deleted file mode 100644 index e9310364..00000000 --- a/presentation/src/main/res/drawable/ic_indicator_grey5.xml +++ /dev/null @@ -1,9 +0,0 @@ - - - diff --git a/presentation/src/main/res/drawable/ic_indicator_white_30.xml b/presentation/src/main/res/drawable/ic_indicator_white_30.xml new file mode 100644 index 00000000..07697d88 --- /dev/null +++ b/presentation/src/main/res/drawable/ic_indicator_white_30.xml @@ -0,0 +1,9 @@ + + + \ No newline at end of file diff --git a/presentation/src/main/res/drawable/ic_info_white_40.xml b/presentation/src/main/res/drawable/ic_info_white_40.xml new file mode 100644 index 00000000..05499afa --- /dev/null +++ b/presentation/src/main/res/drawable/ic_info_white_40.xml @@ -0,0 +1,9 @@ + + + diff --git a/presentation/src/main/res/drawable/ic_logout.png b/presentation/src/main/res/drawable/ic_logout.png deleted file mode 100644 index 9b1fcb85..00000000 Binary files a/presentation/src/main/res/drawable/ic_logout.png and /dev/null differ diff --git a/presentation/src/main/res/drawable/ic_male_gray.xml b/presentation/src/main/res/drawable/ic_male_gray.xml new file mode 100644 index 00000000..07700b02 --- /dev/null +++ b/presentation/src/main/res/drawable/ic_male_gray.xml @@ -0,0 +1,10 @@ + + + diff --git a/presentation/src/main/res/drawable/ic_male_green.xml b/presentation/src/main/res/drawable/ic_male_green.xml new file mode 100644 index 00000000..071e84d6 --- /dev/null +++ b/presentation/src/main/res/drawable/ic_male_green.xml @@ -0,0 +1,9 @@ + + + diff --git a/presentation/src/main/res/drawable/ic_next.xml b/presentation/src/main/res/drawable/ic_next.xml new file mode 100644 index 00000000..181296bd --- /dev/null +++ b/presentation/src/main/res/drawable/ic_next.xml @@ -0,0 +1,13 @@ + + + + + + diff --git a/presentation/src/main/res/drawable/ic_play.xml b/presentation/src/main/res/drawable/ic_play.xml new file mode 100644 index 00000000..e80331c4 --- /dev/null +++ b/presentation/src/main/res/drawable/ic_play.xml @@ -0,0 +1,9 @@ + + + diff --git a/presentation/src/main/res/drawable/ic_profile_empty.png b/presentation/src/main/res/drawable/ic_profile_empty.png deleted file mode 100644 index d7aed533..00000000 Binary files a/presentation/src/main/res/drawable/ic_profile_empty.png and /dev/null differ diff --git a/presentation/src/main/res/drawable/ic_quit.png b/presentation/src/main/res/drawable/ic_quit.png deleted file mode 100644 index 74c8f9e9..00000000 Binary files a/presentation/src/main/res/drawable/ic_quit.png and /dev/null differ diff --git a/presentation/src/main/res/drawable/ic_refresh.xml b/presentation/src/main/res/drawable/ic_refresh.xml deleted file mode 100644 index 4653687f..00000000 --- a/presentation/src/main/res/drawable/ic_refresh.xml +++ /dev/null @@ -1,12 +0,0 @@ - - - diff --git a/presentation/src/main/res/drawable/ic_right_arrow.xml b/presentation/src/main/res/drawable/ic_right_arrow.xml deleted file mode 100644 index 75f8fded..00000000 --- a/presentation/src/main/res/drawable/ic_right_arrow.xml +++ /dev/null @@ -1,13 +0,0 @@ - - - diff --git a/presentation/src/main/res/drawable/ic_setting.xml b/presentation/src/main/res/drawable/ic_setting.xml index 9ba4c5b2..e40897cd 100644 --- a/presentation/src/main/res/drawable/ic_setting.xml +++ b/presentation/src/main/res/drawable/ic_setting.xml @@ -1,10 +1,11 @@ + android:width="24dp" + android:height="24dp" + android:viewportWidth="24" + android:viewportHeight="24"> + android:pathData="M10.133,22C9.628,22 9.203,21.625 9.141,21.124L8.85,18.8C8.633,18.717 8.429,18.617 8.238,18.5C8.046,18.383 7.858,18.258 7.675,18.125L5.512,19.034C5.048,19.229 4.511,19.049 4.259,18.613L2.393,15.39C2.14,14.953 2.252,14.396 2.654,14.092L4.525,12.675C4.508,12.558 4.5,12.446 4.5,12.337V11.663C4.5,11.554 4.508,11.442 4.525,11.325L2.654,9.908C2.252,9.604 2.14,9.047 2.393,8.61L4.259,5.387C4.511,4.951 5.048,4.771 5.512,4.966L7.675,5.875C7.858,5.742 8.05,5.617 8.25,5.5C8.45,5.383 8.65,5.283 8.85,5.2L9.141,2.876C9.203,2.376 9.628,2 10.133,2H13.867C14.372,2 14.797,2.376 14.859,2.876L15.15,5.2C15.367,5.283 15.571,5.383 15.762,5.5C15.954,5.617 16.142,5.742 16.325,5.875L18.488,4.966C18.952,4.771 19.489,4.951 19.741,5.387L21.607,8.61C21.86,9.047 21.748,9.604 21.346,9.908L19.475,11.325C19.492,11.442 19.5,11.554 19.5,11.663V12.337C19.5,12.446 19.483,12.558 19.45,12.675L21.32,14.092C21.723,14.396 21.835,14.953 21.582,15.39L19.718,18.611C19.465,19.047 18.926,19.227 18.462,19.031L16.325,18.125C16.142,18.258 15.95,18.383 15.75,18.5C15.55,18.617 15.35,18.717 15.15,18.8L14.859,21.124C14.797,21.625 14.372,22 13.867,22H10.133ZM12.05,15.5C13.017,15.5 13.842,15.158 14.525,14.475C15.208,13.792 15.55,12.967 15.55,12C15.55,11.033 15.208,10.208 14.525,9.525C13.842,8.842 13.017,8.5 12.05,8.5C11.067,8.5 10.238,8.842 9.563,9.525C8.888,10.208 8.55,11.033 8.55,12C8.55,12.967 8.888,13.792 9.563,14.475C10.238,15.158 11.067,15.5 12.05,15.5Z" + android:strokeAlpha="0.4" + android:fillColor="#ffffff" + android:fillAlpha="0.4"/> diff --git a/presentation/src/main/res/drawable/ic_share.xml b/presentation/src/main/res/drawable/ic_share.xml deleted file mode 100644 index 1647ce5e..00000000 --- a/presentation/src/main/res/drawable/ic_share.xml +++ /dev/null @@ -1,35 +0,0 @@ - - - - - - - diff --git a/presentation/src/main/res/drawable/img_alarm.png b/presentation/src/main/res/drawable/img_alarm.png index 2723140e..2b58393f 100644 Binary files a/presentation/src/main/res/drawable/img_alarm.png and b/presentation/src/main/res/drawable/img_alarm.png differ diff --git a/presentation/src/main/res/drawable/img_alert_circle.png b/presentation/src/main/res/drawable/img_alert_circle.png new file mode 100644 index 00000000..ebca898b Binary files /dev/null and b/presentation/src/main/res/drawable/img_alert_circle.png differ diff --git a/presentation/src/main/res/drawable/img_alert_triangle.png b/presentation/src/main/res/drawable/img_alert_triangle.png new file mode 100644 index 00000000..e6ef8858 Binary files /dev/null and b/presentation/src/main/res/drawable/img_alert_triangle.png differ diff --git a/presentation/src/main/res/drawable/img_angle_high.xml b/presentation/src/main/res/drawable/img_angle_high.xml deleted file mode 100644 index 17eed1fc..00000000 --- a/presentation/src/main/res/drawable/img_angle_high.xml +++ /dev/null @@ -1,211 +0,0 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - diff --git a/presentation/src/main/res/drawable/img_angle_low.xml b/presentation/src/main/res/drawable/img_angle_low.xml deleted file mode 100644 index 4cfbb1d6..00000000 --- a/presentation/src/main/res/drawable/img_angle_low.xml +++ /dev/null @@ -1,228 +0,0 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - diff --git a/presentation/src/main/res/drawable/img_angle_middle.xml b/presentation/src/main/res/drawable/img_angle_middle.xml deleted file mode 100644 index ab921dab..00000000 --- a/presentation/src/main/res/drawable/img_angle_middle.xml +++ /dev/null @@ -1,214 +0,0 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - diff --git a/presentation/src/main/res/drawable/img_blur_bottom.png b/presentation/src/main/res/drawable/img_blur_bottom.png deleted file mode 100644 index b406c9f0..00000000 Binary files a/presentation/src/main/res/drawable/img_blur_bottom.png and /dev/null differ diff --git a/presentation/src/main/res/drawable/img_blur_top.png b/presentation/src/main/res/drawable/img_blur_top.png deleted file mode 100644 index 206a4d0d..00000000 Binary files a/presentation/src/main/res/drawable/img_blur_top.png and /dev/null differ diff --git a/presentation/src/main/res/drawable/img_dialog_bg_green_bottom.xml b/presentation/src/main/res/drawable/img_dialog_bg_green_bottom.xml deleted file mode 100644 index 3302aff3..00000000 --- a/presentation/src/main/res/drawable/img_dialog_bg_green_bottom.xml +++ /dev/null @@ -1,18 +0,0 @@ - - - - - - - diff --git a/presentation/src/main/res/drawable/img_dialog_bg_green_top.xml b/presentation/src/main/res/drawable/img_dialog_bg_green_top.xml deleted file mode 100644 index 2dc60757..00000000 --- a/presentation/src/main/res/drawable/img_dialog_bg_green_top.xml +++ /dev/null @@ -1,18 +0,0 @@ - - - - - - - diff --git a/presentation/src/main/res/drawable/img_dialog_bg_grey_bottom.xml b/presentation/src/main/res/drawable/img_dialog_bg_grey_bottom.xml deleted file mode 100644 index 9c201a8f..00000000 --- a/presentation/src/main/res/drawable/img_dialog_bg_grey_bottom.xml +++ /dev/null @@ -1,18 +0,0 @@ - - - - - - - diff --git a/presentation/src/main/res/drawable/img_dialog_bg_grey_top.xml b/presentation/src/main/res/drawable/img_dialog_bg_grey_top.xml deleted file mode 100644 index 4e9873b4..00000000 --- a/presentation/src/main/res/drawable/img_dialog_bg_grey_top.xml +++ /dev/null @@ -1,18 +0,0 @@ - - - - - - - diff --git a/presentation/src/main/res/drawable/img_empty_image.xml b/presentation/src/main/res/drawable/img_empty_image.xml new file mode 100644 index 00000000..c93f17ee --- /dev/null +++ b/presentation/src/main/res/drawable/img_empty_image.xml @@ -0,0 +1,26 @@ + + + + + + + + + + + diff --git a/presentation/src/main/res/drawable/img_feed_gradient.png b/presentation/src/main/res/drawable/img_feed_gradient.png deleted file mode 100644 index f9c67e65..00000000 Binary files a/presentation/src/main/res/drawable/img_feed_gradient.png and /dev/null differ diff --git a/presentation/src/main/res/drawable/img_finished_bottom.png b/presentation/src/main/res/drawable/img_finished_bottom.png deleted file mode 100644 index 5ed0dd77..00000000 Binary files a/presentation/src/main/res/drawable/img_finished_bottom.png and /dev/null differ diff --git a/presentation/src/main/res/drawable/img_finished_ghost.png b/presentation/src/main/res/drawable/img_finished_ghost.png deleted file mode 100644 index 3da75f1a..00000000 Binary files a/presentation/src/main/res/drawable/img_finished_ghost.png and /dev/null differ diff --git a/presentation/src/main/res/drawable/img_frame_full.xml b/presentation/src/main/res/drawable/img_frame_full.xml deleted file mode 100644 index 41de0845..00000000 --- a/presentation/src/main/res/drawable/img_frame_full.xml +++ /dev/null @@ -1,187 +0,0 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - diff --git a/presentation/src/main/res/drawable/img_frame_knee.xml b/presentation/src/main/res/drawable/img_frame_knee.xml deleted file mode 100644 index 50078074..00000000 --- a/presentation/src/main/res/drawable/img_frame_knee.xml +++ /dev/null @@ -1,181 +0,0 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - diff --git a/presentation/src/main/res/drawable/img_frame_upper.xml b/presentation/src/main/res/drawable/img_frame_upper.xml deleted file mode 100644 index cf10705f..00000000 --- a/presentation/src/main/res/drawable/img_frame_upper.xml +++ /dev/null @@ -1,174 +0,0 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - diff --git a/presentation/src/main/res/drawable/img_glow.png b/presentation/src/main/res/drawable/img_glow.png new file mode 100644 index 00000000..fbcc58df Binary files /dev/null and b/presentation/src/main/res/drawable/img_glow.png differ diff --git a/presentation/src/main/res/drawable/img_gradation_black_bottom.xml b/presentation/src/main/res/drawable/img_gradation_black_bottom.xml new file mode 100644 index 00000000..a3bbb442 --- /dev/null +++ b/presentation/src/main/res/drawable/img_gradation_black_bottom.xml @@ -0,0 +1,21 @@ + + + + + + + + + + diff --git a/presentation/src/main/res/drawable/img_gradation_black_top.xml b/presentation/src/main/res/drawable/img_gradation_black_top.xml new file mode 100644 index 00000000..7efdaf2c --- /dev/null +++ b/presentation/src/main/res/drawable/img_gradation_black_top.xml @@ -0,0 +1,21 @@ + + + + + + + + + + diff --git a/presentation/src/main/res/drawable/img_gradation_image_bottom.xml b/presentation/src/main/res/drawable/img_gradation_image_bottom.xml new file mode 100644 index 00000000..444b382b --- /dev/null +++ b/presentation/src/main/res/drawable/img_gradation_image_bottom.xml @@ -0,0 +1,21 @@ + + + + + + + + + + diff --git a/presentation/src/main/res/drawable/img_gradation_image_top.xml b/presentation/src/main/res/drawable/img_gradation_image_top.xml new file mode 100644 index 00000000..9cf782a2 --- /dev/null +++ b/presentation/src/main/res/drawable/img_gradation_image_top.xml @@ -0,0 +1,21 @@ + + + + + + + + + + diff --git a/presentation/src/main/res/drawable/img_green_bg_2_word.xml b/presentation/src/main/res/drawable/img_green_bg_2_word.xml deleted file mode 100644 index 1a095798..00000000 --- a/presentation/src/main/res/drawable/img_green_bg_2_word.xml +++ /dev/null @@ -1,16 +0,0 @@ - - - - diff --git a/presentation/src/main/res/drawable/img_green_bg_3_word.xml b/presentation/src/main/res/drawable/img_green_bg_3_word.xml deleted file mode 100644 index 93035f86..00000000 --- a/presentation/src/main/res/drawable/img_green_bg_3_word.xml +++ /dev/null @@ -1,20 +0,0 @@ - - - - - - - diff --git a/presentation/src/main/res/drawable/img_guide_bg.png b/presentation/src/main/res/drawable/img_guide_bg.png deleted file mode 100644 index 452306cf..00000000 Binary files a/presentation/src/main/res/drawable/img_guide_bg.png and /dev/null differ diff --git a/presentation/src/main/res/drawable/img_guide_ex.png b/presentation/src/main/res/drawable/img_guide_ex.png deleted file mode 100644 index a893a728..00000000 Binary files a/presentation/src/main/res/drawable/img_guide_ex.png and /dev/null differ diff --git a/presentation/src/main/res/drawable/img_lightning.png b/presentation/src/main/res/drawable/img_lightning.png deleted file mode 100644 index 665b81cf..00000000 Binary files a/presentation/src/main/res/drawable/img_lightning.png and /dev/null differ diff --git a/presentation/src/main/res/drawable/img_login_bg.png b/presentation/src/main/res/drawable/img_login_bg.png new file mode 100644 index 00000000..cfd883a6 Binary files /dev/null and b/presentation/src/main/res/drawable/img_login_bg.png differ diff --git a/presentation/src/main/res/drawable/img_login_bg_bottom.xml b/presentation/src/main/res/drawable/img_login_bg_bottom.xml deleted file mode 100644 index 3c94ae80..00000000 --- a/presentation/src/main/res/drawable/img_login_bg_bottom.xml +++ /dev/null @@ -1,17 +0,0 @@ - - - - - - - diff --git a/presentation/src/main/res/drawable/img_login_bg_top.xml b/presentation/src/main/res/drawable/img_login_bg_top.xml deleted file mode 100644 index 8c8ed2e8..00000000 --- a/presentation/src/main/res/drawable/img_login_bg_top.xml +++ /dev/null @@ -1,17 +0,0 @@ - - - - - - - \ No newline at end of file diff --git a/presentation/src/main/res/drawable/img_number_one.xml b/presentation/src/main/res/drawable/img_number_one.xml new file mode 100644 index 00000000..633c0f52 --- /dev/null +++ b/presentation/src/main/res/drawable/img_number_one.xml @@ -0,0 +1,10 @@ + + + diff --git a/presentation/src/main/res/drawable/img_number_two.xml b/presentation/src/main/res/drawable/img_number_two.xml new file mode 100644 index 00000000..e36b857f --- /dev/null +++ b/presentation/src/main/res/drawable/img_number_two.xml @@ -0,0 +1,19 @@ + + + + + + + diff --git a/presentation/src/main/res/drawable/img_onboarding_finish.png b/presentation/src/main/res/drawable/img_onboarding_finish.png deleted file mode 100644 index d6d2b792..00000000 Binary files a/presentation/src/main/res/drawable/img_onboarding_finish.png and /dev/null differ diff --git a/presentation/src/main/res/drawable/img_onboarding_first.png b/presentation/src/main/res/drawable/img_onboarding_first.png new file mode 100644 index 00000000..fe1df654 Binary files /dev/null and b/presentation/src/main/res/drawable/img_onboarding_first.png differ diff --git a/presentation/src/main/res/drawable/img_onboarding_one.png b/presentation/src/main/res/drawable/img_onboarding_one.png deleted file mode 100644 index 147a11a2..00000000 Binary files a/presentation/src/main/res/drawable/img_onboarding_one.png and /dev/null differ diff --git a/presentation/src/main/res/drawable/img_onboarding_second.png b/presentation/src/main/res/drawable/img_onboarding_second.png new file mode 100644 index 00000000..655a0b23 Binary files /dev/null and b/presentation/src/main/res/drawable/img_onboarding_second.png differ diff --git a/presentation/src/main/res/drawable/img_onboarding_third.png b/presentation/src/main/res/drawable/img_onboarding_third.png new file mode 100644 index 00000000..efc6ca1f Binary files /dev/null and b/presentation/src/main/res/drawable/img_onboarding_third.png differ diff --git a/presentation/src/main/res/drawable/img_onboarding_three.png b/presentation/src/main/res/drawable/img_onboarding_three.png deleted file mode 100644 index 66e828b5..00000000 Binary files a/presentation/src/main/res/drawable/img_onboarding_three.png and /dev/null differ diff --git a/presentation/src/main/res/drawable/img_onboarding_two.png b/presentation/src/main/res/drawable/img_onboarding_two.png deleted file mode 100644 index 681c4adc..00000000 Binary files a/presentation/src/main/res/drawable/img_onboarding_two.png and /dev/null differ diff --git a/presentation/src/main/res/drawable/img_openchat.png b/presentation/src/main/res/drawable/img_openchat.png deleted file mode 100644 index be55895f..00000000 Binary files a/presentation/src/main/res/drawable/img_openchat.png and /dev/null differ diff --git a/presentation/src/main/res/drawable/img_photo_add.png b/presentation/src/main/res/drawable/img_photo_add.png new file mode 100644 index 00000000..7851e788 Binary files /dev/null and b/presentation/src/main/res/drawable/img_photo_add.png differ diff --git a/presentation/src/main/res/drawable/img_profile_bg_normal.png b/presentation/src/main/res/drawable/img_profile_bg_normal.png deleted file mode 100644 index 724cf7f5..00000000 Binary files a/presentation/src/main/res/drawable/img_profile_bg_normal.png and /dev/null differ diff --git a/presentation/src/main/res/drawable/img_profile_bg_waiting.png b/presentation/src/main/res/drawable/img_profile_bg_waiting.png deleted file mode 100644 index 6a9ad6a4..00000000 Binary files a/presentation/src/main/res/drawable/img_profile_bg_waiting.png and /dev/null differ diff --git a/presentation/src/main/res/drawable/img_profile_create_active.xml b/presentation/src/main/res/drawable/img_profile_create_active.xml new file mode 100644 index 00000000..8735a00a --- /dev/null +++ b/presentation/src/main/res/drawable/img_profile_create_active.xml @@ -0,0 +1,19 @@ + + + + + + + + diff --git a/presentation/src/main/res/drawable/img_profile_create_inactive.xml b/presentation/src/main/res/drawable/img_profile_create_inactive.xml new file mode 100644 index 00000000..be7b30fd --- /dev/null +++ b/presentation/src/main/res/drawable/img_profile_create_inactive.xml @@ -0,0 +1,23 @@ + + + + + + + + diff --git a/presentation/src/main/res/drawable/img_profile_making.png b/presentation/src/main/res/drawable/img_profile_making.png new file mode 100644 index 00000000..970815c3 Binary files /dev/null and b/presentation/src/main/res/drawable/img_profile_making.png differ diff --git a/presentation/src/main/res/drawable/img_purple_bg_2_3.xml b/presentation/src/main/res/drawable/img_purple_bg_2_3.xml deleted file mode 100644 index 3b77b4fe..00000000 --- a/presentation/src/main/res/drawable/img_purple_bg_2_3.xml +++ /dev/null @@ -1,21 +0,0 @@ - - - - - - - - - - diff --git a/presentation/src/main/res/drawable/img_purple_bg_3_2.xml b/presentation/src/main/res/drawable/img_purple_bg_3_2.xml deleted file mode 100644 index dac5a066..00000000 --- a/presentation/src/main/res/drawable/img_purple_bg_3_2.xml +++ /dev/null @@ -1,21 +0,0 @@ - - - - - - - - - - diff --git a/presentation/src/main/res/drawable/img_rating_bg.png b/presentation/src/main/res/drawable/img_rating_bg.png deleted file mode 100644 index 0a03b33c..00000000 Binary files a/presentation/src/main/res/drawable/img_rating_bg.png and /dev/null differ diff --git a/presentation/src/main/res/drawable/img_ratio_2_3.xml b/presentation/src/main/res/drawable/img_ratio_2_3.xml index a5f0065c..e500087b 100644 --- a/presentation/src/main/res/drawable/img_ratio_2_3.xml +++ b/presentation/src/main/res/drawable/img_ratio_2_3.xml @@ -1,237 +1,20 @@ + android:width="61dp" + android:height="61dp" + android:viewportWidth="61" + android:viewportHeight="61"> - - - - - - - - - - - - - - - - - - - - - - - - - + android:pathData="M4.618,10.723L56.618,10.723A4,4 0,0 1,60.618 14.723L60.618,46.723A4,4 0,0 1,56.618 50.723L4.618,50.723A4,4 0,0 1,0.618 46.723L0.618,14.723A4,4 0,0 1,4.618 10.723z" + android:fillColor="#ffffff" + android:fillAlpha="0.1"/> + android:strokeColor="#ffffff"/> - - - - - - - - - - - - + android:pathData="M30.618,30.222C29.518,30.222 28.576,29.83 27.793,29.047C27.01,28.264 26.618,27.322 26.618,26.222C26.618,25.122 27.01,24.181 27.793,23.397C28.576,22.614 29.518,22.222 30.618,22.222C31.718,22.222 32.66,22.614 33.443,23.397C34.227,24.181 34.618,25.122 34.618,26.222C34.618,27.322 34.227,28.264 33.443,29.047C32.66,29.83 31.718,30.222 30.618,30.222ZM22.618,38.222V35.422C22.618,34.855 22.764,34.335 23.056,33.86C23.347,33.385 23.735,33.022 24.218,32.772C25.251,32.256 26.302,31.868 27.368,31.61C28.435,31.351 29.518,31.222 30.618,31.222C31.718,31.222 32.801,31.351 33.868,31.61C34.935,31.868 35.985,32.256 37.018,32.772C37.501,33.022 37.889,33.385 38.181,33.86C38.472,34.335 38.618,34.855 38.618,35.422V38.222H22.618Z" + android:fillColor="#ffffff" + android:fillAlpha="0.8"/> diff --git a/presentation/src/main/res/drawable/img_ratio_3_2.xml b/presentation/src/main/res/drawable/img_ratio_3_2.xml index c51006ad..11fc4425 100644 --- a/presentation/src/main/res/drawable/img_ratio_3_2.xml +++ b/presentation/src/main/res/drawable/img_ratio_3_2.xml @@ -1,237 +1,20 @@ + android:width="61dp" + android:height="61dp" + android:viewportWidth="61" + android:viewportHeight="61"> - - - - - - - - - - - - - - - - - - - - - - - - - + android:pathData="M14.618,0.723L46.618,0.723A4,4 0,0 1,50.618 4.723L50.618,56.723A4,4 0,0 1,46.618 60.723L14.618,60.723A4,4 0,0 1,10.618 56.723L10.618,4.723A4,4 0,0 1,14.618 0.723z" + android:fillColor="#ffffff" + android:fillAlpha="0.1"/> + android:strokeColor="#ffffff"/> - - - - - - - - - - - - + android:pathData="M30.69,30.222C29.59,30.222 28.648,29.83 27.865,29.047C27.081,28.264 26.69,27.322 26.69,26.222C26.69,25.122 27.081,24.181 27.865,23.397C28.648,22.614 29.59,22.222 30.69,22.222C31.789,22.222 32.731,22.614 33.514,23.397C34.298,24.181 34.689,25.122 34.689,26.222C34.689,27.322 34.298,28.264 33.514,29.047C32.731,29.83 31.789,30.222 30.69,30.222ZM22.69,38.222V35.422C22.69,34.855 22.835,34.335 23.127,33.86C23.419,33.385 23.806,33.022 24.289,32.772C25.323,32.256 26.373,31.868 27.44,31.61C28.506,31.351 29.59,31.222 30.69,31.222C31.789,31.222 32.873,31.351 33.939,31.61C35.006,31.868 36.056,32.256 37.09,32.772C37.573,33.022 37.96,33.385 38.252,33.86C38.544,34.335 38.689,34.855 38.689,35.422V38.222H22.69Z" + android:fillColor="#ffffff" + android:fillAlpha="0.8"/> diff --git a/presentation/src/main/res/drawable/img_selfie_ai_one.png b/presentation/src/main/res/drawable/img_selfie_ai_one.png deleted file mode 100644 index be442773..00000000 Binary files a/presentation/src/main/res/drawable/img_selfie_ai_one.png and /dev/null differ diff --git a/presentation/src/main/res/drawable/img_selfie_ai_two.png b/presentation/src/main/res/drawable/img_selfie_ai_two.png deleted file mode 100644 index 0d2449d7..00000000 Binary files a/presentation/src/main/res/drawable/img_selfie_ai_two.png and /dev/null differ diff --git a/presentation/src/main/res/drawable/img_selfie_ex_one.png b/presentation/src/main/res/drawable/img_selfie_ex_one.png deleted file mode 100644 index f264ffdd..00000000 Binary files a/presentation/src/main/res/drawable/img_selfie_ex_one.png and /dev/null differ diff --git a/presentation/src/main/res/drawable/img_selfie_ex_two.png b/presentation/src/main/res/drawable/img_selfie_ex_two.png deleted file mode 100644 index 5d6c6401..00000000 Binary files a/presentation/src/main/res/drawable/img_selfie_ex_two.png and /dev/null differ diff --git a/presentation/src/main/res/drawable/img_selfie_one.png b/presentation/src/main/res/drawable/img_selfie_one.png index fe8da81d..e795d233 100644 Binary files a/presentation/src/main/res/drawable/img_selfie_one.png and b/presentation/src/main/res/drawable/img_selfie_one.png differ diff --git a/presentation/src/main/res/drawable/img_selfie_three.png b/presentation/src/main/res/drawable/img_selfie_three.png index 23e3fae8..1b3e3dc8 100644 Binary files a/presentation/src/main/res/drawable/img_selfie_three.png and b/presentation/src/main/res/drawable/img_selfie_three.png differ diff --git a/presentation/src/main/res/drawable/img_selfie_two.png b/presentation/src/main/res/drawable/img_selfie_two.png index b1900543..765945e0 100644 Binary files a/presentation/src/main/res/drawable/img_selfie_two.png and b/presentation/src/main/res/drawable/img_selfie_two.png differ diff --git a/presentation/src/main/res/drawable/img_shine.png b/presentation/src/main/res/drawable/img_shine.png new file mode 100644 index 00000000..05afe5d0 Binary files /dev/null and b/presentation/src/main/res/drawable/img_shine.png differ diff --git a/presentation/src/main/res/drawable/img_signup_bg_bottom.xml b/presentation/src/main/res/drawable/img_signup_bg_bottom.xml deleted file mode 100644 index 7165c30f..00000000 --- a/presentation/src/main/res/drawable/img_signup_bg_bottom.xml +++ /dev/null @@ -1,18 +0,0 @@ - - - - - - - diff --git a/presentation/src/main/res/drawable/img_signup_bg_top.xml b/presentation/src/main/res/drawable/img_signup_bg_top.xml deleted file mode 100644 index fec3c94c..00000000 --- a/presentation/src/main/res/drawable/img_signup_bg_top.xml +++ /dev/null @@ -1,18 +0,0 @@ - - - - - - - diff --git a/presentation/src/main/res/drawable/img_tooltip_feed.png b/presentation/src/main/res/drawable/img_tooltip_feed.png new file mode 100644 index 00000000..b8fa3963 Binary files /dev/null and b/presentation/src/main/res/drawable/img_tooltip_feed.png differ diff --git a/presentation/src/main/res/drawable/img_tooltip_finished.png b/presentation/src/main/res/drawable/img_tooltip_finished.png new file mode 100644 index 00000000..29349821 Binary files /dev/null and b/presentation/src/main/res/drawable/img_tooltip_finished.png differ diff --git a/presentation/src/main/res/drawable/ic_verify_tooltip.png b/presentation/src/main/res/drawable/img_tooltip_verify.png similarity index 100% rename from presentation/src/main/res/drawable/ic_verify_tooltip.png rename to presentation/src/main/res/drawable/img_tooltip_verify.png diff --git a/presentation/src/main/res/drawable/img_wait_background.png b/presentation/src/main/res/drawable/img_wait_background.png deleted file mode 100644 index 9d2c5c88..00000000 Binary files a/presentation/src/main/res/drawable/img_wait_background.png and /dev/null differ diff --git a/presentation/src/main/res/drawable/img_wait_character.png b/presentation/src/main/res/drawable/img_wait_character.png deleted file mode 100644 index a3cebdb9..00000000 Binary files a/presentation/src/main/res/drawable/img_wait_character.png and /dev/null differ diff --git a/presentation/src/main/res/drawable/img_waiting_title.xml b/presentation/src/main/res/drawable/img_waiting_title.xml deleted file mode 100644 index a46d96e7..00000000 --- a/presentation/src/main/res/drawable/img_waiting_title.xml +++ /dev/null @@ -1,24 +0,0 @@ - - - - - - - - diff --git a/presentation/src/main/res/drawable/layerlist_create_progressbar.xml b/presentation/src/main/res/drawable/layerlist_create_progressbar.xml index ea392dc1..53713225 100644 --- a/presentation/src/main/res/drawable/layerlist_create_progressbar.xml +++ b/presentation/src/main/res/drawable/layerlist_create_progressbar.xml @@ -1,16 +1,22 @@ + - - + + - + - - + + - + \ No newline at end of file diff --git a/presentation/src/main/res/drawable/menu_create.png b/presentation/src/main/res/drawable/menu_create.png new file mode 100644 index 00000000..975d98f6 Binary files /dev/null and b/presentation/src/main/res/drawable/menu_create.png differ diff --git a/presentation/src/main/res/drawable/menu_create_selected.xml b/presentation/src/main/res/drawable/menu_create_selected.xml deleted file mode 100644 index 14578ff9..00000000 --- a/presentation/src/main/res/drawable/menu_create_selected.xml +++ /dev/null @@ -1,34 +0,0 @@ - - - - - - - diff --git a/presentation/src/main/res/drawable/menu_create_unselected.xml b/presentation/src/main/res/drawable/menu_create_unselected.xml deleted file mode 100644 index 88536d9d..00000000 --- a/presentation/src/main/res/drawable/menu_create_unselected.xml +++ /dev/null @@ -1,57 +0,0 @@ - - - - - - - - - - - - - - - - - diff --git a/presentation/src/main/res/drawable/menu_feed_selected.xml b/presentation/src/main/res/drawable/menu_feed_selected.xml index 1ff2a581..8bc045e9 100644 --- a/presentation/src/main/res/drawable/menu_feed_selected.xml +++ b/presentation/src/main/res/drawable/menu_feed_selected.xml @@ -1,18 +1,9 @@ + android:width="24dp" + android:height="25dp" + android:viewportWidth="24" + android:viewportHeight="25"> - - - + android:pathData="M13,9.5V3.5H21V9.5H13ZM3,13.5V3.5H11V13.5H3ZM13,21.5V11.5H21V21.5H13ZM3,21.5V15.5H11V21.5H3Z" + android:fillColor="#ffffff"/> diff --git a/presentation/src/main/res/drawable/menu_feed_unselected.xml b/presentation/src/main/res/drawable/menu_feed_unselected.xml index 14470f7e..1e8d1036 100644 --- a/presentation/src/main/res/drawable/menu_feed_unselected.xml +++ b/presentation/src/main/res/drawable/menu_feed_unselected.xml @@ -1,30 +1,9 @@ + android:width="24dp" + android:height="25dp" + android:viewportWidth="24" + android:viewportHeight="25"> - - - + android:pathData="M13,9.5V3.5H21V9.5H13ZM3,13.5V3.5H11V13.5H3ZM13,21.5V11.5H21V21.5H13ZM3,21.5V15.5H11V21.5H3Z" + android:fillColor="#80FFFFFF"/> diff --git a/presentation/src/main/res/drawable/menu_profile_selected.xml b/presentation/src/main/res/drawable/menu_profile_selected.xml index 3fff99e5..44e8fcd6 100644 --- a/presentation/src/main/res/drawable/menu_profile_selected.xml +++ b/presentation/src/main/res/drawable/menu_profile_selected.xml @@ -1,30 +1,9 @@ + android:width="24dp" + android:height="25dp" + android:viewportWidth="24" + android:viewportHeight="25"> - - - - + android:pathData="M12,12.5C10.9,12.5 9.958,12.108 9.175,11.325C8.392,10.542 8,9.6 8,8.5C8,7.4 8.392,6.458 9.175,5.675C9.958,4.892 10.9,4.5 12,4.5C13.1,4.5 14.042,4.892 14.825,5.675C15.608,6.458 16,7.4 16,8.5C16,9.6 15.608,10.542 14.825,11.325C14.042,12.108 13.1,12.5 12,12.5ZM4,20.5V17.7C4,17.133 4.146,16.612 4.438,16.138C4.729,15.663 5.117,15.3 5.6,15.05C6.633,14.533 7.683,14.146 8.75,13.887C9.817,13.629 10.9,13.5 12,13.5C13.1,13.5 14.183,13.629 15.25,13.887C16.317,14.146 17.367,14.533 18.4,15.05C18.883,15.3 19.271,15.663 19.563,16.138C19.854,16.612 20,17.133 20,17.7V20.5H4Z" + android:fillColor="#ffffff"/> diff --git a/presentation/src/main/res/drawable/menu_profile_unselected.xml b/presentation/src/main/res/drawable/menu_profile_unselected.xml index 3cf35b67..cbf48d5a 100644 --- a/presentation/src/main/res/drawable/menu_profile_unselected.xml +++ b/presentation/src/main/res/drawable/menu_profile_unselected.xml @@ -1,29 +1,9 @@ + android:width="24dp" + android:height="25dp" + android:viewportWidth="24" + android:viewportHeight="25"> - - - + android:pathData="M12,12.5C10.9,12.5 9.958,12.108 9.175,11.325C8.392,10.542 8,9.6 8,8.5C8,7.4 8.392,6.458 9.175,5.675C9.958,4.892 10.9,4.5 12,4.5C13.1,4.5 14.042,4.892 14.825,5.675C15.608,6.458 16,7.4 16,8.5C16,9.6 15.608,10.542 14.825,11.325C14.042,12.108 13.1,12.5 12,12.5ZM4,20.5V17.7C4,17.133 4.146,16.612 4.438,16.138C4.729,15.663 5.117,15.3 5.6,15.05C6.633,14.533 7.683,14.146 8.75,13.887C9.817,13.629 10.9,13.5 12,13.5C13.1,13.5 14.183,13.629 15.25,13.887C16.317,14.146 17.367,14.533 18.4,15.05C18.883,15.3 19.271,15.663 19.563,16.138C19.854,16.612 20,17.133 20,17.7V20.5H4Z" + android:fillColor="#80FFFFFF"/> diff --git a/presentation/src/main/res/drawable/mock_img_2_3.png b/presentation/src/main/res/drawable/mock_img_2_3.png index db70d7b1..a84fffe1 100644 Binary files a/presentation/src/main/res/drawable/mock_img_2_3.png and b/presentation/src/main/res/drawable/mock_img_2_3.png differ diff --git a/presentation/src/main/res/drawable/mock_img_3_2.png b/presentation/src/main/res/drawable/mock_img_3_2.png index 8c13bb06..50bb30f8 100644 Binary files a/presentation/src/main/res/drawable/mock_img_3_2.png and b/presentation/src/main/res/drawable/mock_img_3_2.png differ diff --git a/presentation/src/main/res/drawable/sel_btn_gradation.xml b/presentation/src/main/res/drawable/sel_btn_gradation.xml new file mode 100644 index 00000000..919e18f2 --- /dev/null +++ b/presentation/src/main/res/drawable/sel_btn_gradation.xml @@ -0,0 +1,22 @@ + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/presentation/src/main/res/drawable/sel_btn.xml b/presentation/src/main/res/drawable/sel_btn_green.xml similarity index 65% rename from presentation/src/main/res/drawable/sel_btn.xml rename to presentation/src/main/res/drawable/sel_btn_green.xml index aed2344e..c06655ab 100644 --- a/presentation/src/main/res/drawable/sel_btn.xml +++ b/presentation/src/main/res/drawable/sel_btn_green.xml @@ -1,10 +1,9 @@ - + - + @@ -12,7 +11,7 @@ - + diff --git a/presentation/src/main/res/drawable/shape_bg_fill_green1_line_10_rect.xml b/presentation/src/main/res/drawable/shape_bg_fill_green1_line_10_rect.xml deleted file mode 100644 index 8a4b164b..00000000 --- a/presentation/src/main/res/drawable/shape_bg_fill_green1_line_10_rect.xml +++ /dev/null @@ -1,7 +0,0 @@ - - - - - - - \ No newline at end of file diff --git a/presentation/src/main/res/drawable/shape_bg_fill_green_line_10_rect.xml b/presentation/src/main/res/drawable/shape_bg_fill_green_line_10_rect.xml deleted file mode 100644 index fd805e37..00000000 --- a/presentation/src/main/res/drawable/shape_bg_fill_green_line_10_rect.xml +++ /dev/null @@ -1,7 +0,0 @@ - - - - - - - \ No newline at end of file diff --git a/presentation/src/main/res/drawable/shape_bg_fill_grey7_line_rect.xml b/presentation/src/main/res/drawable/shape_bg_fill_grey7_line_rect.xml deleted file mode 100644 index 737d4775..00000000 --- a/presentation/src/main/res/drawable/shape_bg_fill_grey7_line_rect.xml +++ /dev/null @@ -1,6 +0,0 @@ - - - - - - \ No newline at end of file diff --git a/presentation/src/main/res/drawable/shape_green_new_fill_10_rect.xml b/presentation/src/main/res/drawable/shape_black_fill_10_rect.xml similarity index 76% rename from presentation/src/main/res/drawable/shape_green_new_fill_10_rect.xml rename to presentation/src/main/res/drawable/shape_black_fill_10_rect.xml index 9b248cf7..f134f84d 100644 --- a/presentation/src/main/res/drawable/shape_green_new_fill_10_rect.xml +++ b/presentation/src/main/res/drawable/shape_black_fill_10_rect.xml @@ -1,5 +1,5 @@ - + \ No newline at end of file diff --git a/presentation/src/main/res/drawable/shape_green1_fill_6_rect.xml b/presentation/src/main/res/drawable/shape_black_fill_16_rect.xml similarity index 58% rename from presentation/src/main/res/drawable/shape_green1_fill_6_rect.xml rename to presentation/src/main/res/drawable/shape_black_fill_16_rect.xml index 4e7ba345..e4998c72 100644 --- a/presentation/src/main/res/drawable/shape_green1_fill_6_rect.xml +++ b/presentation/src/main/res/drawable/shape_black_fill_16_rect.xml @@ -1,5 +1,5 @@ - - + + \ No newline at end of file diff --git a/presentation/src/main/res/drawable/shape_white_fill_14_rect.xml b/presentation/src/main/res/drawable/shape_gray_fill_10_rect.xml similarity index 55% rename from presentation/src/main/res/drawable/shape_white_fill_14_rect.xml rename to presentation/src/main/res/drawable/shape_gray_fill_10_rect.xml index fae12551..c2409ef1 100644 --- a/presentation/src/main/res/drawable/shape_white_fill_14_rect.xml +++ b/presentation/src/main/res/drawable/shape_gray_fill_10_rect.xml @@ -1,5 +1,5 @@ - - + + \ No newline at end of file diff --git a/presentation/src/main/res/drawable/shape_gray_fill_16_rect.xml b/presentation/src/main/res/drawable/shape_gray_fill_16_rect.xml new file mode 100644 index 00000000..0a3552c3 --- /dev/null +++ b/presentation/src/main/res/drawable/shape_gray_fill_16_rect.xml @@ -0,0 +1,5 @@ + + + + + \ No newline at end of file diff --git a/presentation/src/main/res/drawable/shape_green4_fill_6_rect.xml b/presentation/src/main/res/drawable/shape_gray_fill_6_rect.xml similarity index 76% rename from presentation/src/main/res/drawable/shape_green4_fill_6_rect.xml rename to presentation/src/main/res/drawable/shape_gray_fill_6_rect.xml index e8fe68cb..08d00065 100644 --- a/presentation/src/main/res/drawable/shape_green4_fill_6_rect.xml +++ b/presentation/src/main/res/drawable/shape_gray_fill_6_rect.xml @@ -1,5 +1,5 @@ - + \ No newline at end of file diff --git a/presentation/src/main/res/drawable/shape_gray_fill_green_line_rect.xml b/presentation/src/main/res/drawable/shape_gray_fill_green_line_rect.xml new file mode 100644 index 00000000..606ea5b4 --- /dev/null +++ b/presentation/src/main/res/drawable/shape_gray_fill_green_line_rect.xml @@ -0,0 +1,7 @@ + + + + + \ No newline at end of file diff --git a/presentation/src/main/res/drawable/shape_gray_fill_top16_rect.xml b/presentation/src/main/res/drawable/shape_gray_fill_top16_rect.xml new file mode 100644 index 00000000..f80b6d9d --- /dev/null +++ b/presentation/src/main/res/drawable/shape_gray_fill_top16_rect.xml @@ -0,0 +1,7 @@ + + + + + \ No newline at end of file diff --git a/presentation/src/main/res/drawable/shape_gray_green_fill_6_rect.xml b/presentation/src/main/res/drawable/shape_gray_green_fill_6_rect.xml new file mode 100644 index 00000000..23c8f531 --- /dev/null +++ b/presentation/src/main/res/drawable/shape_gray_green_fill_6_rect.xml @@ -0,0 +1,8 @@ + + + + + \ No newline at end of file diff --git a/presentation/src/main/res/drawable/shape_green1_line_8_rect.xml b/presentation/src/main/res/drawable/shape_green1_line_8_rect.xml deleted file mode 100644 index e6cda0fd..00000000 --- a/presentation/src/main/res/drawable/shape_green1_line_8_rect.xml +++ /dev/null @@ -1,6 +0,0 @@ - - - - - - \ No newline at end of file diff --git a/presentation/src/main/res/drawable/shape_green3_fill_15_bottom_rect.xml b/presentation/src/main/res/drawable/shape_green3_fill_15_bottom_rect.xml deleted file mode 100644 index a1af81df..00000000 --- a/presentation/src/main/res/drawable/shape_green3_fill_15_bottom_rect.xml +++ /dev/null @@ -1,7 +0,0 @@ - - - - - diff --git a/presentation/src/main/res/drawable/shape_green4_fill_green_line_15_rect.xml b/presentation/src/main/res/drawable/shape_green4_fill_green_line_15_rect.xml deleted file mode 100644 index 430aeb66..00000000 --- a/presentation/src/main/res/drawable/shape_green4_fill_green_line_15_rect.xml +++ /dev/null @@ -1,7 +0,0 @@ - - - - - - - \ No newline at end of file diff --git a/presentation/src/main/res/drawable/shape_green_fill_10_rect.xml b/presentation/src/main/res/drawable/shape_green_fill_10_rect.xml new file mode 100644 index 00000000..d16b0749 --- /dev/null +++ b/presentation/src/main/res/drawable/shape_green_fill_10_rect.xml @@ -0,0 +1,5 @@ + + + + + \ No newline at end of file diff --git a/presentation/src/main/res/drawable/shape_green_gradation_6_rect.xml b/presentation/src/main/res/drawable/shape_green_gradation_6_rect.xml new file mode 100644 index 00000000..971d644b --- /dev/null +++ b/presentation/src/main/res/drawable/shape_green_gradation_6_rect.xml @@ -0,0 +1,8 @@ + + + + + \ No newline at end of file diff --git a/presentation/src/main/res/drawable/shape_green_gradation_fill_100_rect.xml b/presentation/src/main/res/drawable/shape_green_gradation_fill_100_rect.xml new file mode 100644 index 00000000..ae59a45a --- /dev/null +++ b/presentation/src/main/res/drawable/shape_green_gradation_fill_100_rect.xml @@ -0,0 +1,8 @@ + + + + + \ No newline at end of file diff --git a/presentation/src/main/res/drawable/shape_green_gradation_fill_10_rect.xml b/presentation/src/main/res/drawable/shape_green_gradation_fill_10_rect.xml new file mode 100644 index 00000000..ec4b4275 --- /dev/null +++ b/presentation/src/main/res/drawable/shape_green_gradation_fill_10_rect.xml @@ -0,0 +1,8 @@ + + + + + \ No newline at end of file diff --git a/presentation/src/main/res/drawable/shape_grey1_50_fill_green_line_rect.xml b/presentation/src/main/res/drawable/shape_grey1_50_fill_green_line_rect.xml deleted file mode 100644 index 009b03b7..00000000 --- a/presentation/src/main/res/drawable/shape_grey1_50_fill_green_line_rect.xml +++ /dev/null @@ -1,6 +0,0 @@ - - - - - - \ No newline at end of file diff --git a/presentation/src/main/res/drawable/shape_grey2_line_8_rect.xml b/presentation/src/main/res/drawable/shape_grey2_line_8_rect.xml deleted file mode 100644 index 61e5311f..00000000 --- a/presentation/src/main/res/drawable/shape_grey2_line_8_rect.xml +++ /dev/null @@ -1,6 +0,0 @@ - - - - - - \ No newline at end of file diff --git a/presentation/src/main/res/drawable/shape_grey5_fill_6_rect.xml b/presentation/src/main/res/drawable/shape_grey5_fill_6_rect.xml deleted file mode 100644 index 76cac5d3..00000000 --- a/presentation/src/main/res/drawable/shape_grey5_fill_6_rect.xml +++ /dev/null @@ -1,5 +0,0 @@ - - - - - diff --git a/presentation/src/main/res/drawable/shape_grey6_fill_15_rect.xml b/presentation/src/main/res/drawable/shape_grey6_fill_15_rect.xml deleted file mode 100644 index 9da36288..00000000 --- a/presentation/src/main/res/drawable/shape_grey6_fill_15_rect.xml +++ /dev/null @@ -1,5 +0,0 @@ - - - - - diff --git a/presentation/src/main/res/drawable/shape_grey6_fill_grey5_line_10_rect.xml b/presentation/src/main/res/drawable/shape_grey6_fill_grey5_line_10_rect.xml deleted file mode 100644 index 2f34f023..00000000 --- a/presentation/src/main/res/drawable/shape_grey6_fill_grey5_line_10_rect.xml +++ /dev/null @@ -1,8 +0,0 @@ - - - - - - - \ No newline at end of file diff --git a/presentation/src/main/res/drawable/shape_kakao_fill_8_rect.xml b/presentation/src/main/res/drawable/shape_kakao_fill_10_rect.xml similarity index 83% rename from presentation/src/main/res/drawable/shape_kakao_fill_8_rect.xml rename to presentation/src/main/res/drawable/shape_kakao_fill_10_rect.xml index fe492bfa..e3367d75 100644 --- a/presentation/src/main/res/drawable/shape_kakao_fill_8_rect.xml +++ b/presentation/src/main/res/drawable/shape_kakao_fill_10_rect.xml @@ -2,5 +2,5 @@ - + \ No newline at end of file diff --git a/presentation/src/main/res/drawable/shape_red_fill_10_rect.xml b/presentation/src/main/res/drawable/shape_red_fill_10_rect.xml new file mode 100644 index 00000000..03ae0b23 --- /dev/null +++ b/presentation/src/main/res/drawable/shape_red_fill_10_rect.xml @@ -0,0 +1,5 @@ + + + + + \ No newline at end of file diff --git a/presentation/src/main/res/drawable/shape_white10_fill_100_rect.xml b/presentation/src/main/res/drawable/shape_white10_fill_100_rect.xml new file mode 100644 index 00000000..613f4d0d --- /dev/null +++ b/presentation/src/main/res/drawable/shape_white10_fill_100_rect.xml @@ -0,0 +1,5 @@ + + + + + \ No newline at end of file diff --git a/presentation/src/main/res/drawable/shape_white40_fill_100_rect.xml b/presentation/src/main/res/drawable/shape_white40_fill_100_rect.xml new file mode 100644 index 00000000..46759ef0 --- /dev/null +++ b/presentation/src/main/res/drawable/shape_white40_fill_100_rect.xml @@ -0,0 +1,5 @@ + + + + + \ No newline at end of file diff --git a/presentation/src/main/res/drawable/shape_white_fill_grey7_line_rect.xml b/presentation/src/main/res/drawable/shape_white_fill_grey7_line_rect.xml deleted file mode 100644 index 837bd02f..00000000 --- a/presentation/src/main/res/drawable/shape_white_fill_grey7_line_rect.xml +++ /dev/null @@ -1,6 +0,0 @@ - - - - - - \ No newline at end of file diff --git a/presentation/src/main/res/layout/fragment_create.xml b/presentation/src/main/res/layout/activity_create.xml similarity index 64% rename from presentation/src/main/res/layout/fragment_create.xml rename to presentation/src/main/res/layout/activity_create.xml index 755c7244..97d5af3e 100644 --- a/presentation/src/main/res/layout/fragment_create.xml +++ b/presentation/src/main/res/layout/activity_create.xml @@ -9,21 +9,21 @@ + type="kr.genti.presentation.create.CreateViewModel" /> + android:background="@color/black"> - @@ -32,34 +32,43 @@ android:id="@+id/btn_back" android:layout_width="wrap_content" android:layout_height="wrap_content" - android:layout_marginStart="28dp" - android:layout_marginTop="9dp" + android:layout_marginStart="16dp" + android:layout_marginTop="5dp" android:src="@drawable/ic_back" - android:visibility="gone" + app:layout_constraintBottom_toBottomOf="@id/tv_create_title" app:layout_constraintStart_toStartOf="parent" - app:layout_constraintTop_toTopOf="@id/iv_create_logo" - tools:visibility="visible" /> + app:layout_constraintTop_toTopOf="@id/tv_create_title" /> + app:layout_constraintEnd_toEndOf="parent" + app:layout_constraintStart_toStartOf="parent" + app:layout_constraintTop_toBottomOf="@id/tv_create_title" /> + + + android:layout_height="match_parent" + android:background="@color/black"> - - - - - - - - - - - - - + app:layout_constraintTop_toTopOf="parent" + tools:src="@drawable/mock_img_3_2" + tools:visibility="gone" /> - - - - - + + app:layout_constraintTop_toBottomOf="@id/btn_close"> - - - + app:layout_constraintStart_toStartOf="parent" /> + app:layout_constraintStart_toStartOf="parent" /> - - - + app:layout_constraintStart_toStartOf="parent"> - + + + + app:layout_constraintBottom_toBottomOf="@id/cv_finished_image" + app:layout_constraintEnd_toEndOf="parent" /> + app:layout_constraintStart_toStartOf="parent" /> - + + - + android:layout_gravity="center" + android:layout_marginTop="0.5dp" + android:src="@drawable/ic_info_white_40" /> - + android:layout_marginStart="8dp" + android:text="@string/finished_btn_unwanted" + android:textColor="@color/white_60" /> - + - - - + app:layout_constraintStart_toStartOf="parent" + app:layout_constraintTop_toTopOf="parent" + tools:visibility="gone"> + + + + \ No newline at end of file diff --git a/presentation/src/main/res/layout/activity_login.xml b/presentation/src/main/res/layout/activity_login.xml index be7b4ee2..7363fd94 100644 --- a/presentation/src/main/res/layout/activity_login.xml +++ b/presentation/src/main/res/layout/activity_login.xml @@ -1,6 +1,7 @@ + xmlns:app="http://schemas.android.com/apk/res-auto" + xmlns:tools="http://schemas.android.com/tools"> @@ -9,94 +10,83 @@ + android:background="@color/black"> - - + app:layout_constraintStart_toStartOf="parent" + app:layout_constraintTop_toTopOf="parent" + tools:scaleType="centerCrop" /> - + app:layout_constraintTop_toTopOf="parent" /> + app:layout_constraintTop_toBottomOf="@id/tv_login_title" />" + app:layout_constraintStart_toStartOf="parent"> diff --git a/presentation/src/main/res/layout/activity_main.xml b/presentation/src/main/res/layout/activity_main.xml index 11c6a04b..808a3892 100644 --- a/presentation/src/main/res/layout/activity_main.xml +++ b/presentation/src/main/res/layout/activity_main.xml @@ -23,8 +23,9 @@ android:id="@+id/bnv_main" android:layout_width="0dp" android:layout_height="64dp" - android:background="@color/background_white" + android:background="@color/black" android:elevation="0dp" + android:paddingHorizontal="30dp" android:paddingTop="8dp" android:paddingBottom="10dp" app:itemRippleColor="@null" @@ -35,16 +36,28 @@ + + \ No newline at end of file diff --git a/presentation/src/main/res/layout/activity_onboarding.xml b/presentation/src/main/res/layout/activity_onboarding.xml index e61e9e7d..78d73dae 100644 --- a/presentation/src/main/res/layout/activity_onboarding.xml +++ b/presentation/src/main/res/layout/activity_onboarding.xml @@ -8,48 +8,27 @@ + android:layout_height="match_parent" + android:background="@color/black"> - - - - @@ -58,47 +37,61 @@ android:id="@+id/dot_indicator" android:layout_width="wrap_content" android:layout_height="16dp" - android:layout_marginBottom="98dp" + android:layout_marginBottom="48dp" app:ci_drawable="@drawable/ic_indicator_green" - app:ci_drawable_unselected="@drawable/ic_indicator_grey5" - app:layout_constraintBottom_toBottomOf="parent" + app:ci_drawable_unselected="@drawable/ic_indicator_white_30" + app:layout_constraintBottom_toTopOf="@id/layout_bottom_btn" app:layout_constraintEnd_toEndOf="parent" app:layout_constraintStart_toStartOf="parent" /> - + app:layout_constraintTop_toTopOf="parent" /> - + app:layout_constraintStart_toStartOf="parent"> + + + + + + \ No newline at end of file diff --git a/presentation/src/main/res/layout/activity_openchat.xml b/presentation/src/main/res/layout/activity_openchat.xml deleted file mode 100644 index 9b55891e..00000000 --- a/presentation/src/main/res/layout/activity_openchat.xml +++ /dev/null @@ -1,147 +0,0 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - \ No newline at end of file diff --git a/presentation/src/main/res/layout/activity_setting.xml b/presentation/src/main/res/layout/activity_setting.xml index 3ac795fc..e6efee61 100644 --- a/presentation/src/main/res/layout/activity_setting.xml +++ b/presentation/src/main/res/layout/activity_setting.xml @@ -9,159 +9,176 @@ - - + android:layout_height="match_parent" + android:background="@color/black"> + app:layout_constraintTop_toTopOf="@id/tv_setting_title" /> - + + + + - - - - - - - - - - - - - - + + + + - - - - - - + app:layout_constraintTop_toBottomOf="@id/btn_privacy_policy" /> + app:layout_constraintTop_toTopOf="@id/btn_company_info" /> + + + + + + + + + + + + + + \ No newline at end of file diff --git a/presentation/src/main/res/layout/activity_signup.xml b/presentation/src/main/res/layout/activity_signup.xml index a875e765..92a48539 100644 --- a/presentation/src/main/res/layout/activity_signup.xml +++ b/presentation/src/main/res/layout/activity_signup.xml @@ -19,203 +19,206 @@ + android:background="@color/black"> - - - - - + app:layout_constraintTop_toBottomOf="@id/tv_signup_title" /> + + app:layout_constraintTop_toBottomOf="@id/tv_signup_subtitle"> - - + + + + + + + + - - - + + + + + + + + app:layout_constraintTop_toBottomOf="@id/btn_gender_woman" /> - - - + app:layout_constraintTop_toBottomOf="@id/tv_signup_birth_title" /> - - + app:layout_constraintTop_toBottomOf="@id/et_signup_birth_year" + tools:background="@drawable/shape_green_gradation_fill_100_rect" /> diff --git a/presentation/src/main/res/layout/activity_verify.xml b/presentation/src/main/res/layout/activity_verify.xml index 09273f6f..a4dbdc86 100644 --- a/presentation/src/main/res/layout/activity_verify.xml +++ b/presentation/src/main/res/layout/activity_verify.xml @@ -9,7 +9,7 @@ + android:background="@color/black"> + android:layout_height="match_parent" + android:background="@color/black"> - - - - + app:layout_constraintTop_toBottomOf="@id/tv_wait_header" /> - + - + - + - + - + - - + diff --git a/presentation/src/main/res/layout/bottom_sheet_feed_info.xml b/presentation/src/main/res/layout/bottom_sheet_feed_info.xml new file mode 100644 index 00000000..acd8b794 --- /dev/null +++ b/presentation/src/main/res/layout/bottom_sheet_feed_info.xml @@ -0,0 +1,69 @@ + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/presentation/src/main/res/layout/dialog_create_error.xml b/presentation/src/main/res/layout/dialog_create_error.xml index 9f6adcce..4882771e 100644 --- a/presentation/src/main/res/layout/dialog_create_error.xml +++ b/presentation/src/main/res/layout/dialog_create_error.xml @@ -7,101 +7,91 @@ - - - + app:layout_constraintTop_toTopOf="parent"> - + - + - + - + - + - + \ No newline at end of file diff --git a/presentation/src/main/res/layout/dialog_create_finished.xml b/presentation/src/main/res/layout/dialog_create_finished.xml index feba82b1..7486540a 100644 --- a/presentation/src/main/res/layout/dialog_create_finished.xml +++ b/presentation/src/main/res/layout/dialog_create_finished.xml @@ -7,102 +7,93 @@ - - - + app:layout_constraintTop_toTopOf="parent"> - + - + - + - + - + - + + \ No newline at end of file diff --git a/presentation/src/main/res/layout/dialog_create_guide.xml b/presentation/src/main/res/layout/dialog_create_guide.xml deleted file mode 100644 index 8dde8fa4..00000000 --- a/presentation/src/main/res/layout/dialog_create_guide.xml +++ /dev/null @@ -1,119 +0,0 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - - - - \ No newline at end of file diff --git a/presentation/src/main/res/layout/dialog_create_select.xml b/presentation/src/main/res/layout/dialog_create_select.xml new file mode 100644 index 00000000..a92d1ff1 --- /dev/null +++ b/presentation/src/main/res/layout/dialog_create_select.xml @@ -0,0 +1,160 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/presentation/src/main/res/layout/dialog_create_unable.xml b/presentation/src/main/res/layout/dialog_create_unable.xml index 743a60fd..55414c0e 100644 --- a/presentation/src/main/res/layout/dialog_create_unable.xml +++ b/presentation/src/main/res/layout/dialog_create_unable.xml @@ -8,102 +8,74 @@ - + app:layout_constraintTop_toTopOf="parent"> - + - + - + - + - - - - - + \ No newline at end of file diff --git a/presentation/src/main/res/layout/dialog_finished_image.xml b/presentation/src/main/res/layout/dialog_finished_image.xml index 9f0e5ff3..4aab5787 100644 --- a/presentation/src/main/res/layout/dialog_finished_image.xml +++ b/presentation/src/main/res/layout/dialog_finished_image.xml @@ -1,6 +1,7 @@ + xmlns:app="http://schemas.android.com/apk/res-auto" + xmlns:tools="http://schemas.android.com/tools"> @@ -8,60 +9,78 @@ + android:layout_height="match_parent" + tools:background="@color/black"> - + + + app:layout_constraintTop_toTopOf="parent" + tools:src="@drawable/mock_img_2_3" /> - + app:layout_constraintStart_toStartOf="parent"> - + + + + + \ No newline at end of file diff --git a/presentation/src/main/res/layout/dialog_finished_rating.xml b/presentation/src/main/res/layout/dialog_finished_rating.xml index a046edbe..c02c204f 100644 --- a/presentation/src/main/res/layout/dialog_finished_rating.xml +++ b/presentation/src/main/res/layout/dialog_finished_rating.xml @@ -7,89 +7,98 @@ + android:layout_width="match_parent" + android:layout_height="wrap_content"> - + app:layout_constraintTop_toTopOf="parent"> - + - + - + - + + + - + \ No newline at end of file diff --git a/presentation/src/main/res/layout/dialog_finished_report.xml b/presentation/src/main/res/layout/dialog_finished_report.xml index df8daf4f..3e5ecdc9 100644 --- a/presentation/src/main/res/layout/dialog_finished_report.xml +++ b/presentation/src/main/res/layout/dialog_finished_report.xml @@ -1,6 +1,7 @@ + xmlns:app="http://schemas.android.com/apk/res-auto" + xmlns:tools="http://schemas.android.com/tools"> @@ -13,14 +14,14 @@ + android:layout_width="match_parent" + android:layout_height="match_parent"> - - - - - - - - + android:layout_width="match_parent" + android:layout_height="wrap_content" + android:layout_marginHorizontal="24dp" + android:background="@drawable/shape_gray_fill_16_rect" + android:paddingBottom="16dp" + tools:visibility="gone"> - - - - - - - - + android:layout_marginHorizontal="24dp" + android:background="@drawable/shape_gray_fill_16_rect" + android:paddingBottom="16dp" + android:visibility="gone" + tools:visibility="visible"> - - + app:layout_constraintTop_toBottomOf="@id/tv_error_output_title" /> @@ -254,16 +174,6 @@ - - \ No newline at end of file diff --git a/presentation/src/main/res/layout/dialog_profile_image.xml b/presentation/src/main/res/layout/dialog_profile_image.xml index b3877a7e..6c39359c 100644 --- a/presentation/src/main/res/layout/dialog_profile_image.xml +++ b/presentation/src/main/res/layout/dialog_profile_image.xml @@ -9,87 +9,83 @@ + android:layout_height="match_parent" + tools:background="@color/black"> + + + + - + app:layout_constraintEnd_toStartOf="@id/btn_download" + app:layout_constraintStart_toStartOf="parent" /> - - - - - - - - - - + app:layout_constraintStart_toEndOf="@id/btn_share" /> \ No newline at end of file diff --git a/presentation/src/main/res/layout/dialog_push.xml b/presentation/src/main/res/layout/dialog_push.xml index 22422f49..f860da7b 100644 --- a/presentation/src/main/res/layout/dialog_push.xml +++ b/presentation/src/main/res/layout/dialog_push.xml @@ -8,103 +8,92 @@ - + app:layout_constraintTop_toTopOf="parent"> - + - + - - - + - + - + - + diff --git a/presentation/src/main/res/layout/dialog_setting_logout.xml b/presentation/src/main/res/layout/dialog_setting_logout.xml index 058eb0c8..71736865 100644 --- a/presentation/src/main/res/layout/dialog_setting_logout.xml +++ b/presentation/src/main/res/layout/dialog_setting_logout.xml @@ -7,111 +7,91 @@ - + app:layout_constraintTop_toTopOf="parent"> - + - + - + - + - - - + - + \ No newline at end of file diff --git a/presentation/src/main/res/layout/dialog_setting_quit.xml b/presentation/src/main/res/layout/dialog_setting_quit.xml index 1601f7ae..ca8c176f 100644 --- a/presentation/src/main/res/layout/dialog_setting_quit.xml +++ b/presentation/src/main/res/layout/dialog_setting_quit.xml @@ -7,112 +7,91 @@ - + app:layout_constraintTop_toTopOf="parent"> - + - + - + - + - - - + - + \ No newline at end of file diff --git a/presentation/src/main/res/layout/dialog_verify_exit.xml b/presentation/src/main/res/layout/dialog_verify_exit.xml index 7d1ef310..ea96fa33 100644 --- a/presentation/src/main/res/layout/dialog_verify_exit.xml +++ b/presentation/src/main/res/layout/dialog_verify_exit.xml @@ -7,91 +7,91 @@ - + app:layout_constraintTop_toTopOf="parent"> - + - + - + - + - + - + \ No newline at end of file diff --git a/presentation/src/main/res/layout/fragment_define.xml b/presentation/src/main/res/layout/fragment_define.xml index c9e57450..3308ee32 100644 --- a/presentation/src/main/res/layout/fragment_define.xml +++ b/presentation/src/main/res/layout/fragment_define.xml @@ -1,7 +1,6 @@ + xmlns:app="http://schemas.android.com/apk/res-auto"> @@ -9,235 +8,151 @@ + type="kr.genti.presentation.create.CreateViewModel" /> + android:background="@color/black"> - + + + + + + + + + + + + + + + + + app:layout_constraintStart_toStartOf="parent"> - - - - - + android:layout_width="0dp" + android:layout_height="wrap_content" + android:layout_margin="16dp" + android:background="@drawable/shape_black_fill_10_rect" + android:padding="8dp" + app:layout_constraintEnd_toEndOf="parent" + app:layout_constraintStart_toStartOf="parent" + app:layout_constraintTop_toTopOf="parent"> - - - - + app:layout_constraintTop_toTopOf="parent" /> - - - - - - - - - - - - - - - - - - - - + android:background="@drawable/sel_btn_gradation" + android:enabled="@{vm.isWritten}" + android:padding="10dp" + android:src="@drawable/ic_play" + app:layout_constraintBottom_toBottomOf="parent" + app:layout_constraintEnd_toEndOf="parent" /> - - + diff --git a/presentation/src/main/res/layout/fragment_feed.xml b/presentation/src/main/res/layout/fragment_feed.xml index f4bfa55d..021ea7eb 100644 --- a/presentation/src/main/res/layout/fragment_feed.xml +++ b/presentation/src/main/res/layout/fragment_feed.xml @@ -10,31 +10,7 @@ - - - - + android:background="@color/black"> @@ -76,5 +52,38 @@ + + + + + + \ No newline at end of file diff --git a/presentation/src/main/res/layout/fragment_number.xml b/presentation/src/main/res/layout/fragment_number.xml new file mode 100644 index 00000000..25ddf5ba --- /dev/null +++ b/presentation/src/main/res/layout/fragment_number.xml @@ -0,0 +1,196 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/presentation/src/main/res/layout/fragment_pose.xml b/presentation/src/main/res/layout/fragment_pose.xml index d8478064..b7a560da 100644 --- a/presentation/src/main/res/layout/fragment_pose.xml +++ b/presentation/src/main/res/layout/fragment_pose.xml @@ -1,6 +1,7 @@ + xmlns:app="http://schemas.android.com/apk/res-auto" + xmlns:tools="http://schemas.android.com/tools"> @@ -8,391 +9,182 @@ + type="kr.genti.presentation.create.CreateViewModel" /> - - - - + android:background="@color/black"> - + + + + + android:id="@+id/btn_ratio_3_2" + android:layout_width="0dp" + android:layout_height="0dp" + android:layout_margin="1dp" + android:background="@{vm.selectedRatio == pictureRatio.RATIO_SERO ? @drawable/shape_gray_green_fill_6_rect : @drawable/shape_gray_fill_6_rect}" + android:onClick="@{() -> vm.selectRatio(pictureRatio.RATIO_SERO)}" + app:layout_constraintBottom_toBottomOf="@id/bg_ratio_3_2" + app:layout_constraintEnd_toEndOf="@id/bg_ratio_3_2" + app:layout_constraintStart_toStartOf="@id/bg_ratio_3_2" + app:layout_constraintTop_toTopOf="@id/bg_ratio_3_2" + tools:background="@drawable/shape_gray_green_fill_6_rect"> - - - - - - - - - + app:layout_constraintTop_toBottomOf="@id/iv_ratio_3_2" + app:layout_constraintVertical_chainStyle="packed" /> - - - + app:layout_constraintTop_toBottomOf="@id/tv_ratio_3_2_title" /> - - - - - + - + - - - + - - - - + app:layout_constraintTop_toTopOf="parent" /> - - - - - - - - - - - - - + app:layout_constraintTop_toBottomOf="@id/iv_ratio_2_3" + app:layout_constraintVertical_chainStyle="packed" /> - - - - + app:layout_constraintStart_toStartOf="parent" + app:layout_constraintTop_toBottomOf="@id/tv_ratio_2_3_title" /> - + diff --git a/presentation/src/main/res/layout/fragment_profile.xml b/presentation/src/main/res/layout/fragment_profile.xml index 6e93b27a..fb102072 100644 --- a/presentation/src/main/res/layout/fragment_profile.xml +++ b/presentation/src/main/res/layout/fragment_profile.xml @@ -10,122 +10,15 @@ - - - - - - - - - - - - - - - - - - - - - - - - - - + android:background="@color/black"> @@ -134,24 +27,38 @@ android:id="@+id/btn_setting" android:layout_width="wrap_content" android:layout_height="wrap_content" - android:layout_marginEnd="16dp" + android:layout_marginEnd="4dp" android:padding="12dp" android:src="@drawable/ic_setting" app:layout_constraintBottom_toBottomOf="@id/tv_profile_title" app:layout_constraintEnd_toEndOf="parent" app:layout_constraintTop_toTopOf="@id/tv_profile_title" /> + + @@ -163,22 +70,14 @@ app:layout_constraintBottom_toBottomOf="parent" app:layout_constraintEnd_toEndOf="parent" app:layout_constraintStart_toStartOf="parent" - app:layout_constraintTop_toBottomOf="@id/layout_profile_header"> - - + app:layout_constraintTop_toBottomOf="@id/iv_profile_making"> + + + + \ No newline at end of file diff --git a/presentation/src/main/res/layout/fragment_selfie.xml b/presentation/src/main/res/layout/fragment_selfie.xml index 46a87513..b829554c 100644 --- a/presentation/src/main/res/layout/fragment_selfie.xml +++ b/presentation/src/main/res/layout/fragment_selfie.xml @@ -9,360 +9,514 @@ + type="kr.genti.presentation.create.CreateViewModel" /> - - - - + android:background="@color/black"> + app:layout_constraintTop_toTopOf="parent"> - - - - - - + - - - + app:layout_constraintStart_toStartOf="parent" + app:layout_constraintTop_toBottomOf="@id/tv_selfie_guide_1" /> - + - + - - + android:layout_marginHorizontal="16dp" + android:layout_marginTop="34dp" + android:elevation="2dp" + android:visibility="gone" + app:layout_constraintEnd_toEndOf="parent" + app:layout_constraintStart_toStartOf="parent" + app:layout_constraintTop_toBottomOf="@id/tv_selfie_warning"> - + android:layout_height="0dp" + android:layout_margin="1dp" + android:elevation="1dp" + android:scaleType="centerCrop" + app:layout_constraintBottom_toBottomOf="@id/iv_added_image_bg_1" + app:layout_constraintDimensionRatio="1" + app:layout_constraintEnd_toEndOf="@id/iv_added_image_bg_1" + app:layout_constraintStart_toStartOf="@id/iv_added_image_bg_1" + app:layout_constraintTop_toTopOf="@id/iv_added_image_bg_1" + tools:src="@drawable/mock_img_3_2" /> - + app:layout_constraintTop_toTopOf="parent" /> + android:layout_height="0dp" + android:layout_margin="1dp" + android:elevation="1dp" + android:scaleType="centerCrop" + app:layout_constraintBottom_toBottomOf="@id/iv_added_image_bg_2" + app:layout_constraintDimensionRatio="1" + app:layout_constraintEnd_toEndOf="@id/iv_added_image_bg_2" + app:layout_constraintStart_toStartOf="@id/iv_added_image_bg_2" + app:layout_constraintTop_toTopOf="@id/iv_added_image_bg_2" + tools:src="@drawable/mock_img_3_2" /> + android:background="@drawable/shape_gray_fill_green_line_rect" + app:layout_constraintBottom_toBottomOf="parent" + app:layout_constraintDimensionRatio="1" + app:layout_constraintEnd_toStartOf="@id/iv_added_image_bg_3" + app:layout_constraintStart_toEndOf="@id/iv_added_image_bg_1" + app:layout_constraintTop_toTopOf="parent" /> + android:layout_height="0dp" + android:layout_margin="1dp" + android:elevation="1dp" + android:scaleType="centerCrop" + app:layout_constraintBottom_toBottomOf="@id/iv_added_image_bg_3" + app:layout_constraintDimensionRatio="1" + app:layout_constraintEnd_toEndOf="@id/iv_added_image_bg_3" + app:layout_constraintStart_toStartOf="@id/iv_added_image_bg_3" + app:layout_constraintTop_toTopOf="@id/iv_added_image_bg_3" + tools:src="@drawable/mock_img_3_2" /> - + app:layout_constraintStart_toEndOf="@id/iv_added_image_bg_2" + app:layout_constraintTop_toTopOf="parent" /> + + + + + app:layout_constraintTop_toTopOf="parent" /> + app:layout_constraintTop_toBottomOf="@id/tv_selfie_input_title" /> - - + app:layout_constraintEnd_toStartOf="@id/iv_ex_image_3" + app:layout_constraintStart_toEndOf="@id/iv_ex_image_1" + app:layout_constraintTop_toBottomOf="@id/tv_selfie_input_title" /> + app:layout_constraintEnd_toEndOf="parent" + app:layout_constraintStart_toEndOf="@id/iv_ex_image_2" + app:layout_constraintTop_toBottomOf="@id/tv_selfie_input_title" /> - + - + + + app:layout_constraintStart_toStartOf="parent" + app:layout_constraintTop_toTopOf="parent"> + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + - - - + - diff --git a/presentation/src/main/res/layout/item_define_example.xml b/presentation/src/main/res/layout/item_define_example.xml new file mode 100644 index 00000000..39f9edbf --- /dev/null +++ b/presentation/src/main/res/layout/item_define_example.xml @@ -0,0 +1,61 @@ + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/presentation/src/main/res/layout/item_feed_info.xml b/presentation/src/main/res/layout/item_feed_info.xml index 5b3bc227..f5396189 100644 --- a/presentation/src/main/res/layout/item_feed_info.xml +++ b/presentation/src/main/res/layout/item_feed_info.xml @@ -8,55 +8,55 @@ + android:layout_height="wrap_content" + android:background="@color/black" + android:paddingBottom="12dp"> - - - + app:layout_constraintTop_toBottomOf="@id/iv_feed_logo" /> + app:layout_constraintTop_toBottomOf="@id/tv_feed_title" /> + + \ No newline at end of file diff --git a/presentation/src/main/res/layout/item_feed_item.xml b/presentation/src/main/res/layout/item_feed_item.xml index 40a3084c..72b18eb8 100644 --- a/presentation/src/main/res/layout/item_feed_item.xml +++ b/presentation/src/main/res/layout/item_feed_item.xml @@ -11,76 +11,81 @@ android:layout_width="match_parent" android:layout_height="wrap_content" android:layout_marginHorizontal="16dp" - android:layout_marginTop="20dp" - android:layout_marginBottom="4dp" - android:background="@drawable/shape_green4_fill_green_line_15_rect" - android:paddingBottom="12dp"> + android:layout_marginTop="16dp"> - - - + android:paddingBottom="16dp"> - + - + - + - + + + + + \ No newline at end of file diff --git a/presentation/src/main/res/layout/item_onboarding.xml b/presentation/src/main/res/layout/item_onboarding.xml index 1265ccfb..c5d2c846 100644 --- a/presentation/src/main/res/layout/item_onboarding.xml +++ b/presentation/src/main/res/layout/item_onboarding.xml @@ -8,62 +8,76 @@ + android:layout_height="match_parent" + android:background="@color/black"> - - - - - - - - - + app:layout_constraintTop_toBottomOf="@id/tv_onboarding_subtitle" /> + + + + + + \ No newline at end of file diff --git a/presentation/src/main/res/layout/item_profile_image.xml b/presentation/src/main/res/layout/item_profile_image.xml index 49d62c5b..0c8d37b5 100644 --- a/presentation/src/main/res/layout/item_profile_image.xml +++ b/presentation/src/main/res/layout/item_profile_image.xml @@ -15,15 +15,15 @@ android:id="@+id/iv_profile_item_image" android:layout_width="0dp" android:layout_height="0dp" - android:background="@color/grey_6" - android:scaleType="centerCrop" android:layout_margin="1dp" - app:layout_constraintDimensionRatio="1" + android:background="@color/gray" + android:scaleType="centerCrop" app:layout_constraintBottom_toBottomOf="parent" + app:layout_constraintDimensionRatio="1" app:layout_constraintEnd_toEndOf="parent" app:layout_constraintStart_toStartOf="parent" app:layout_constraintTop_toTopOf="parent" - tools:src="@drawable/img_selfie_ex_one"/> + tools:src="@drawable/img_selfie_one" /> \ No newline at end of file diff --git a/presentation/src/main/res/menu/menu_main.xml b/presentation/src/main/res/menu/menu_main.xml index 6bbbabb4..1b537db6 100644 --- a/presentation/src/main/res/menu/menu_main.xml +++ b/presentation/src/main/res/menu/menu_main.xml @@ -6,7 +6,7 @@ android:title="@string/menu_feed" /> + app:startDestination="@id/numberFragment"> + + + + + + app:popExitAnim="@anim/slide_right_exit" /> + app:popExitAnim="@anim/slide_right_exit" /> \ No newline at end of file diff --git a/presentation/src/main/res/raw/lottie_waiting.json b/presentation/src/main/res/raw/lottie_waiting.json new file mode 100644 index 00000000..7ed5a420 --- /dev/null +++ b/presentation/src/main/res/raw/lottie_waiting.json @@ -0,0 +1 @@ +{"v":"4.8.0","meta":{"g":"LottieFiles AE 3.5.9","a":"","k":"","d":"","tc":""},"fr":59.4385375976562,"ip":0,"op":140.998903098463,"w":700,"h":700,"nm":"사진 생성 대기 motion","ddd":0,"assets":[{"id":"image_0","w":688,"h":688,"u":"/images/","p":"","e":0},{"id":"image_1","w":688,"h":688,"u":"/images/","p":"","e":0},{"id":"comp_0","layers":[{"ddd":0,"ind":1,"ty":4,"nm":"Shape Layer 1","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":1,"k":[{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.702],"y":[0.001]},"t":0,"s":[0]},{"t":140.671205647786,"s":[360]}],"ix":10},"p":{"a":0,"k":[1120.496,1120.353,0],"ix":2},"a":{"a":0,"k":[0,0,0],"ix":1},"s":{"a":0,"k":[360,360,100],"ix":6}},"ao":0,"shapes":[{"ty":"gr","it":[{"d":1,"ty":"el","s":{"a":0,"k":[137.21,137.21],"ix":2},"p":{"a":0,"k":[0,0],"ix":3},"nm":"Ellipse Path 1","mn":"ADBE Vector Shape - Ellipse","hd":false},{"ty":"st","c":{"a":0,"k":[0.866666965859,0.909803981407,0.984314024682,1],"ix":3},"o":{"a":0,"k":100,"ix":4},"w":{"a":0,"k":8,"ix":5},"lc":2,"lj":1,"ml":4,"bm":0,"nm":"Stroke 1","mn":"ADBE Vector Graphic - Stroke","hd":false},{"ty":"tr","p":{"a":0,"k":[-2.294,-0.338],"ix":2},"a":{"a":0,"k":[0,0],"ix":1},"s":{"a":0,"k":[105.659,105.659],"ix":3},"r":{"a":0,"k":0,"ix":6},"o":{"a":0,"k":100,"ix":7},"sk":{"a":0,"k":0,"ix":4},"sa":{"a":0,"k":0,"ix":5},"nm":"Transform"}],"nm":"Ellipse 1","np":3,"cix":2,"bm":0,"ix":1,"mn":"ADBE Vector Group","hd":false},{"ty":"tm","s":{"a":0,"k":0,"ix":1},"e":{"a":0,"k":2,"ix":2},"o":{"a":0,"k":0,"ix":3},"m":1,"ix":2,"nm":"Trim Paths 1","mn":"ADBE Vector Filter - Trim","hd":false}],"ip":0,"op":4180.51047770182,"st":0,"bm":0}]}],"layers":[{"ddd":0,"ind":1,"ty":0,"nm":"Comp 1","refId":"comp_0","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[350,350,0],"ix":2},"a":{"a":0,"k":[1120,1120,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6}},"ao":0,"w":2240,"h":2240,"ip":0,"op":4180.51047770182,"st":0,"bm":0},{"ddd":0,"ind":2,"ty":2,"nm":"motion_loading_on.png","cl":"png","refId":"image_0","sr":1,"ks":{"o":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":0,"s":[0]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":88.999,"s":[100]},{"t":139.998910877906,"s":[0]}],"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[350,350,0],"ix":2},"a":{"a":0,"k":[344,344,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6}},"ao":0,"ip":0,"op":140.998903098463,"st":0,"bm":0},{"ddd":0,"ind":3,"ty":2,"nm":"motion_loading_off.png","cl":"png","refId":"image_1","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[350,350,0],"ix":2},"a":{"a":0,"k":[344,344,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6}},"ao":0,"ip":0,"op":140.998903098463,"st":0,"bm":0}],"markers":[]} \ No newline at end of file diff --git a/presentation/src/main/res/values/appearances.xml b/presentation/src/main/res/values/appearances.xml index f024a423..5d429dad 100644 --- a/presentation/src/main/res/values/appearances.xml +++ b/presentation/src/main/res/values/appearances.xml @@ -3,75 +3,52 @@ - - - - - - - - - - - - - - - diff --git a/presentation/src/main/res/values/colors.xml b/presentation/src/main/res/values/colors.xml index e9f7c599..a90a5205 100644 --- a/presentation/src/main/res/values/colors.xml +++ b/presentation/src/main/res/values/colors.xml @@ -1,41 +1,29 @@ - - #000000 - #121212 - #6F6F6F - #949494 - #B6B6B6 - #CFCFCF - #F3F3F3 - #4D131326 + + #49F155 + #6CEE2A + #1CF48B - #B3121212 - #80121212 - - #FFFFFF - #F9F9F9 - #7D7D7D - - - #00FF20 - #1CE718 - #83FF92 - #C7FACE - #EDF9EE - #2600FF20 - - - #9966FF - #732DFF - #CDB4FF - #E6DCFA - #F4F2F9 + + #030F0F + #192222 + #0D2D2B + #495959 - #FE0D0D #FFFEE500 + #F1532F + + + #FFFFFF + #CCFFFFFF + #99FFFFFF + #66FFFFFF + #4DFFFFFF + #33FFFFFF + #1AFFFFFF #00000000 @@ -44,8 +32,4 @@ #80000000 #4D000000 - - #030F0F - #49F155 - \ No newline at end of file diff --git a/presentation/src/main/res/values/strings.xml b/presentation/src/main/res/values/strings.xml index ca9439f6..226ccc3d 100644 --- a/presentation/src/main/res/values/strings.xml +++ b/presentation/src/main/res/values/strings.xml @@ -16,51 +16,82 @@ 다음으로 사진 생성하기 + 공유하기 + 저장하기 + 피드 생성 프로필 - 카카오로 계속하기 - 내 마음대로 표현하는\n하나뿐인 AI 사진 + 카카오 로그인 + 내 마음대로 세상에 하나뿐인\n나의 AI 사진을 찍어 보세요. + 5초만에 가입해서 바로 시작할 수 있어요. - 사진을 더 나처럼 표현하기 위해\n아래의 정보가 필요해요! - 성별을 알려주세요! - 남자 - 여자 - 태어난 연도를 알려주세요! - 눌러서 태어난 연도 선택하기 - 입력 완료 + 성별과 나이를 알려주세요. + 사진을 더 잘 표현하기 위해 활용됩니다. + 성별 + 남성 + 여성 + 출생년도 + YYYY + 완료 - 젠티가 처음인 당신을 위해! 다음으로 젠티하러 가기 - - 만들고 싶은 사진을 설명해주세요✏️ - 성적이거나 폭력적인 사진은 만들어지지 않아요 - 의상과 배경을 포함해서 설명해 주세요!\n\n*헤어스타일을 변경하는 기능은 준비 중이에요\n*너무 특정한 배경과 의상은 어려울 수 있어요\n(반포 한강 공원, 나이키 티셔츠 등) - 참고할 구도의 사진이 있다면 추가해주세요 (선택) - (선택) - (참고사진은 최대 1장 업로드 할 수 있어요) + 젠티, 어떻게 사용하나요? + 원하는 사진을 자유롭게 설명해요. + 의상, 배경 등 어떤 사진을 만들고 싶은지 입력해주세요. + + 내 사진 만들기 + 찍고 싶은 내 사진을 AI로 마음껏 만들어 보세요! + 일상적인 사진부터 판타지한 장면까지 모두 가능해요. + 내 사진 세 장만 있으면 만들 수 있어요. + 부모님 사진 만들기 + 사랑하는 부모님의 사진을 만들어 보세요! + 부모님의 독사진, 부모님과 함께 찍은 사진을 만들 수 있어요. + 흑백 사진이거나 화질이 안좋은 사진도 괜찮아요. + + 사진 만들기 + %1$d/3 + + 부모님의 어떤 모습을 담고 싶나요? + 부모님과 함께 찍고 싶은 사진이나 보고 싶은 부모님의 모습을 설명해주세요. + 어떤 사진을 만들고 싶나요? + AI가 이해하지 못하는 장소나 의상은 생성이 어려울 수 있어요. + 성적이거나 폭력적인 사진은 생성되지 않아요. + 만들고 싶은 사진을 설명해주세요. 이런 사진은 어때요? 프랑스 야경을 즐기는 모습을 그려주세요. 항공점퍼를 입고 테라스에 서 있는 모습이에요. - 사진의 비율을 선택해주세요 📷 - 비율은 자유롭게 맡길래요 - 카메라 앵글을 선택해주세요 📷 - 앵글은 자유롭게 맡길래요 - 원하는 프레임을 선택해주세요 📷 - 프레임은 자유롭게 맡길래요 - 3:2 비율\n(세로로 긴 사진) - 2:3 비율\n(가로로 긴 사진) - - 사진 생성에 이용할 얼굴 3장을 업로드해주세요 - 💡얼굴은 이런 사진을 사용해주세요! - 모자, 악세사리 등으로 얼굴이 잘 안보이는 사진을 사용하면 얼굴이 잘 안나올 수 있어요 - 사진은 정면사진을 포함하여 오른쪽, 왼쪽 등 다양한 각도일수록 얼굴 생성에 도움이 돼요 - 만들고자 하는 사진의 포즈와 비슷한 각도의 얼굴 사진을 사용하면 얼굴이 더욱 정확하게 반영돼요 - 본인의 사진만 업로드해주세요. 타인의 사진을 이용할 경우 서비스 이용이 차단되며 법적인 제재를 받을 수도 있어요! - ]]> - 💡\n사진의 각도가 다양할수록 생성된 얼굴이 자연스러워져요! + 부모님 사진 만들기 + + 어떤 사진을 찍고 싶나요? + 1인 사진 + 부모님 단독 + 2인 사진 + 부모님 / 부모님과 나 + + 사진의 비율을 선택해주세요. + 세로 사진 + 3:2 비율 + 가로 사진 + 2:3 비율 + + 사진 생성에 사용할\n얼굴 사진 3장을 선택해주세요. + 정면을 포함하여 사진의 각도가 다양할수록 좋아요. + 모자, 악세사리 등으로 얼굴이 가려진 사진은 안돼요. + 얼굴 표정이 다양할 수록 더 실감나는 사진을 만들 수 있어요. + 흑백 사진이거나, 화질이 좋지 않아도 괜찮아요. + 타인의 사진을 도용하면 서비스 이용이 정지되고,\n법적인 처벌을 받을 수 있습니다. + 사진 생성에 사용할 부모님 얼굴 사진\n3장을 선택해주세요 + 사진 생성에 사용할 두 분의 사진을\n각각 3장씩 선택해주세요 + 이런 사진이 좋아요 + 첫번째 인물 + 두번째 인물 + 사진 선택하기 + 다시 선택하기 + 사진 생성하기 + 3장의 이미지를 선택해 주세요 갤러리 이미지를 가져오는 과정에서 오류가 발생했습니다. @@ -71,72 +102,83 @@ 을 선택해야 해요! 예시 - 세상에 없던 나만의\n 사진을 찍고있어요! - • • • - 예상 소요시간은 2시간입니다. - 미리 만들어놓은 컨셉에 얼굴만 바꾸는게 아니라\n’나만의 하나뿐인 사진’을 찍어드리기 위해 \n배경부터 의상, 구도까지 꼼꼼하게 준비하고 있어요 피드로 돌아가기 - - 정말 죄송합니다. - 서버에 오류가 발생해서\n사진이 만들어지지 않았어요. - 사진 다시 생성하기 + 사진 생성 중 + 세상에 없던\n나만의 사진을 찍고 있어요! + 소중한 사람과의 추억이 담긴\n사진을 찍고 있어요! + 사진이 완성되면, 완료 알림을 보내드려요. + 예상 소요 시간 + 2시간 정도 걸릴 예정이에요. + 세상에 하나뿐인 사진을 만들어 드리기 위해\n배경부터 의상까지 꼼꼼하게 준비하고 있어요. + + 사진 생성에 실패했습니다. + 죄송합니다. 서버에 오류가 발생하였습니다.\n사진 생성을 다시 시도할까요? + 닫기 + 사진 다시 생성 사진이 완성되었어요! - 다른 사진을 만들기 전에\n완성된 사진을 확인할까요? - 사진 확인하기 + 다른 사진을 만들기 전에\n완성된 사진을 보러갈까요? + 사진 보기 - 하나뿐인 나만의 사진이\n완성되었어요! - 사진이 마이페이지에 저장되었어요! + 하나뿐인 나만의 사진 완성! + 마이페이지에서 사진을 확인할 수 있습니다. 공유하기 - 메인으로 이동하기 - 혹시 만들려고 했던 사진과 전혀 다른 사진이 나왔나요? + 사진이 잘못 나왔나요? + 소중한 사진이 완성되었어요! + 사진 저장하기 - 어떤 오류사항이 있었나요? - 구체적으로 작성해 주실수록 오류 확인이 \n빠르게 진행됩니다! - 제출하기 - 취소 + 사진에 어떤 문제가 있나요? + 구체적으로 작성 해주실수록\n오류 확인이 빠르게 진행됩니다. + 오류 사항을 입력해주세요. + 제출 + 닫기 의견 감사합니다! - 작성해주신 내용 잘 확인하여 더 좋은\n서비스를 제공하는 젠티가 되겠습니다. - 확인했습니다! + 작성해주신 내용을 잘 확인하여\n더 좋은 서비스를 제공하는 젠티가 되겠습니다. + 닫기 사진은 어떠셨나요? 별점을 남겨주시면 앞으로 더 마음에 드는\n사진을 만들어 드릴 수 있어요! - 제출하기 - 건너뛰기 + 건너뛰기 - 젠플루언서들이 만든 사진이에요 + 젠플루언서들이 만든 사진들을 둘러보세요. 젠플루언서란? - *여러분이 만든 사진은 자동으로 업로드 되지 않아요️ + 여러분이 만든 사진은 자동으로 업로드되지 않습니다. 사진 설명 + 젠플루언서란? + 젠티로 자신만의 독창적인 사진을 만드는 사람들로,\n다른 사용자들이 자신만의 사진을 잘 만들 수 있도록\n영감을 주는 역할을 합니다. + 어떤 혜택이 있나요? 마이페이지 세상에 없던 나만의 사진 찍는 중 ⋯ 내가 만든 사진 - 아직 내가 만든 사진이 없어요 + 아직 내가 만든 사진이 없어요. + 세상에 하나뿐인 나만의 사진을 만들어보세요! 사진이 저장되었어요\n친구들에게 공유해보세요! - 설정 및 개인정보 + 설정 이용 약관 개인정보처리방침 앱 버전 v%s 사업자 정보 + 문의하기 로그아웃 회원 탈퇴 정말 로그아웃 하시겠어요? - 사진 생성 중에 로그아웃 하시면\n오류가 발생할 수 있습니다. 주의해주세요! - 취소하기 + 사진 생성 중에 로그아웃 하시면\n오류가 발생할 수 있으니 주의해주세요. + 취소 로그아웃 - 정말 탈퇴 하시겠어요? - 생성한 사진 내역이 모두 사라집니다.\n주의해주세요! + 정말 탈퇴하시겠어요? + 탈퇴 시, 생성한 사진 내역이 모두 사라집니다.\n그래도 탈퇴하시겠습니까? 탈퇴하기 사진 완성 알림을 받아보세요! - 알림에 동의해 주시면\n사진이 완성되었을 때 알려드릴게요 - 사진 완성 알림 받기 + 알림에 동의해 주시면\n사진이 완성되었을 때 알려드릴게요. + 다음에 + 알림 받기 내가 만든 특별한 사진, 다른 사람에게 보여주고 싶다면? @@ -160,4 +202,7 @@ 메인 화면으로 나가면 \n촬영한 사진은 자동으로 삭제됩니다. 나가기 + 현재 점검 중입니다! + 닫기 + \ No newline at end of file diff --git a/presentation/src/main/res/values/styles.xml b/presentation/src/main/res/values/styles.xml new file mode 100644 index 00000000..07ee5fc7 --- /dev/null +++ b/presentation/src/main/res/values/styles.xml @@ -0,0 +1,11 @@ + + + + + + + \ No newline at end of file diff --git a/presentation/src/main/res/values/themes.xml b/presentation/src/main/res/values/themes.xml index 2b811f81..64e3906f 100644 --- a/presentation/src/main/res/values/themes.xml +++ b/presentation/src/main/res/values/themes.xml @@ -3,11 +3,20 @@ diff --git a/settings.gradle.kts b/settings.gradle.kts index 4f982844..9af2d230 100644 --- a/settings.gradle.kts +++ b/settings.gradle.kts @@ -1,4 +1,9 @@ +enableFeaturePreview("TYPESAFE_PROJECT_ACCESSORS") + pluginManagement { + + includeBuild("build-logic") + repositories { google() mavenCentral()