diff --git a/app/build.gradle.kts b/app/build.gradle.kts index fb00091e..a1a2231c 100644 --- a/app/build.gradle.kts +++ b/app/build.gradle.kts @@ -21,8 +21,8 @@ android { defaultConfig { applicationId = "com.joeloewi.croissant" - versionCode = 61 - versionName = "1.3.0" + versionCode = 62 + versionName = "1.3.1" targetSdk = 34 testInstrumentationRunner = "androidx.test.runner.AndroidJUnitRunner" diff --git a/app/src/main/kotlin/com/joeloewi/croissant/ui/navigation/main/attendances/screen/createattendance/composable/SelectGames.kt b/app/src/main/kotlin/com/joeloewi/croissant/ui/navigation/main/attendances/screen/createattendance/composable/SelectGames.kt index eb29be3f..7c376ec6 100644 --- a/app/src/main/kotlin/com/joeloewi/croissant/ui/navigation/main/attendances/screen/createattendance/composable/SelectGames.kt +++ b/app/src/main/kotlin/com/joeloewi/croissant/ui/navigation/main/attendances/screen/createattendance/composable/SelectGames.kt @@ -93,7 +93,8 @@ fun SelectGames( HoYoLABGame.HonkaiImpact3rd, HoYoLABGame.GenshinImpact, HoYoLABGame.TearsOfThemis, - HoYoLABGame.HonkaiStarRail + HoYoLABGame.HonkaiStarRail, + HoYoLABGame.ZenlessZoneZero ).toImmutableList() } val containsNotSupportedGame = stringResource(id = R.string.contains_not_supported_game) @@ -424,7 +425,7 @@ fun ConnectedGamesContentListItem( val enabled by remember(hoYoLABGame, gameRecord) { derivedStateOf { - hoYoLABGame == HoYoLABGame.TearsOfThemis || hoYoLABGame == HoYoLABGame.HonkaiStarRail || currentGameRecord.value.gameId != GameRecord.INVALID_GAME_ID + hoYoLABGame == HoYoLABGame.TearsOfThemis || currentGameRecord.value.gameId != GameRecord.INVALID_GAME_ID } } diff --git a/app/src/main/kotlin/com/joeloewi/croissant/ui/navigation/main/settings/screen/SettingsScreen.kt b/app/src/main/kotlin/com/joeloewi/croissant/ui/navigation/main/settings/screen/SettingsScreen.kt index f3c6337d..42977b3f 100644 --- a/app/src/main/kotlin/com/joeloewi/croissant/ui/navigation/main/settings/screen/SettingsScreen.kt +++ b/app/src/main/kotlin/com/joeloewi/croissant/ui/navigation/main/settings/screen/SettingsScreen.kt @@ -197,12 +197,14 @@ fun SettingsContent( value = !isUnusedAppRestrictionEnabled().getOrDefault(false), role = Role.Switch, onValueChange = { - activityResult.launch( - IntentCompat.createManageUnusedAppRestrictionsIntent( - activity, - activity.packageName + runCatching { + activityResult.launch( + IntentCompat.createManageUnusedAppRestrictionsIntent( + activity, + activity.packageName + ) ) - ) + } } ), leadingContent = { diff --git a/app/src/main/kotlin/com/joeloewi/croissant/util/HoYoLABGameNames.kt b/app/src/main/kotlin/com/joeloewi/croissant/util/HoYoLABGameNames.kt index d3443b41..114b3923 100644 --- a/app/src/main/kotlin/com/joeloewi/croissant/util/HoYoLABGameNames.kt +++ b/app/src/main/kotlin/com/joeloewi/croissant/util/HoYoLABGameNames.kt @@ -22,6 +22,10 @@ fun HoYoLABGame.gameNameStringResId(): Int = when (this) { R.string.honkai_star_rail_game_name } + HoYoLABGame.ZenlessZoneZero -> { + R.string.zenless_zone_zero_game_name + } + HoYoLABGame.Unknown -> { R.string.unknown_game_name } diff --git a/app/src/main/kotlin/com/joeloewi/croissant/worker/AttendCheckInEventWorker.kt b/app/src/main/kotlin/com/joeloewi/croissant/worker/AttendCheckInEventWorker.kt index 4ed3f145..2adfafa2 100644 --- a/app/src/main/kotlin/com/joeloewi/croissant/worker/AttendCheckInEventWorker.kt +++ b/app/src/main/kotlin/com/joeloewi/croissant/worker/AttendCheckInEventWorker.kt @@ -47,7 +47,8 @@ class AttendCheckInEventWorker @AssistedInject constructor( private val attendCheckInGenshinImpactUseCase: CheckInUseCase.AttendCheckInGenshinImpact, private val attendCheckInHonkaiImpact3rdUseCase: CheckInUseCase.AttendCheckInHonkaiImpact3rd, private val attendCheckInTearsOfThemisUseCase: CheckInUseCase.AttendCheckInTearsOfThemis, - private val attendCheckInHonkaiStarRail: CheckInUseCase.AttendCheckInHonkaiStarRail, + private val attendCheckInHonkaiStarRailUseCase: CheckInUseCase.AttendCheckInHonkaiStarRailUseCase, + private val attendCheckInZenlessZoneZeroUseCase: CheckInUseCase.AttendCheckInZenlessZoneZeroUseCase, private val insertWorkerExecutionLogUseCase: WorkerExecutionLogUseCase.Insert, private val hasExecutedAtLeastOnce: WorkerExecutionLogUseCase.HasExecutedAtLeastOnce, private val insertSuccessLogUseCase: SuccessLogUseCase.Insert, @@ -159,7 +160,11 @@ class AttendCheckInEventWorker @AssistedInject constructor( } HoYoLABGame.HonkaiStarRail -> { - attendCheckInHonkaiStarRail(cookie = cookie) + attendCheckInHonkaiStarRailUseCase(cookie = cookie) + } + + HoYoLABGame.ZenlessZoneZero -> { + attendCheckInZenlessZoneZeroUseCase(cookie = cookie) } HoYoLABGame.Unknown -> { diff --git a/app/src/main/res/values-en-rUS/strings.xml b/app/src/main/res/values-en-rUS/strings.xml index 1efc437c..9223a0b2 100644 --- a/app/src/main/res/values-en-rUS/strings.xml +++ b/app/src/main/res/values-en-rUS/strings.xml @@ -119,4 +119,5 @@ To ensure accuracy, the scheduled task will be executed 30 seconds after the specified time. For example, if you schedule a task for 12:34 PM, it will actually be executed at 12:34:30 PM. Attendance failed due to unknown error. Retry has scheduled. Attendance failed due to too many requests. Retry has scheduled. + Zenless Zone Zero \ No newline at end of file diff --git a/app/src/main/res/values-ko-rKR/strings.xml b/app/src/main/res/values-ko-rKR/strings.xml index 2d03e692..480c22bf 100644 --- a/app/src/main/res/values-ko-rKR/strings.xml +++ b/app/src/main/res/values-ko-rKR/strings.xml @@ -118,4 +118,5 @@ 작업을 정확하게 실행하기 위해, 지정된 시간보다 30초 늦게 실행됩니다. 예를 들어, 12시 34분에 작업을 예약하면 실제로는 12시 34분 30초에 실행됩니다. 알 수 없는 문제로 인해 출석하지 못했습니다. 재시도를 예약합니다. 요청이 빈번하여 출석하지 못 했습니다. 재시도를 예약합니다. + 젠레스 존 제로 \ No newline at end of file diff --git a/app/src/main/res/values/strings.xml b/app/src/main/res/values/strings.xml index c4493d1f..49741014 100644 --- a/app/src/main/res/values/strings.xml +++ b/app/src/main/res/values/strings.xml @@ -123,4 +123,5 @@ To ensure accuracy, the scheduled task will be executed 30 seconds after the specified time. For example, if you schedule a task for 12:34 PM, it will actually be executed at 12:34:30 PM. Attendance failed due to unknown error. Retry has scheduled. Attendance failed due to too many requests. Retry has scheduled. + Zenless Zone Zero \ No newline at end of file diff --git a/data/src/main/kotlin/com/joeloewi/croissant/data/api/dao/ArcaLiveAppService.kt b/data/src/main/kotlin/com/joeloewi/croissant/data/api/dao/ArcaLiveAppService.kt index ba122f87..16649d25 100644 --- a/data/src/main/kotlin/com/joeloewi/croissant/data/api/dao/ArcaLiveAppService.kt +++ b/data/src/main/kotlin/com/joeloewi/croissant/data/api/dao/ArcaLiveAppService.kt @@ -26,7 +26,7 @@ import retrofit2.http.Query interface ArcaLiveAppService { @GET("view/article/{slug}/{articleId}") suspend fun getArticle( - @Header("User-Agent") userAgent: String = "live.arca.android.playstore/0.8.331-playstore", + @Header("User-Agent") userAgent: String = "net.umanle.arca.android.playstore/0.9.57", @Path("slug") slug: String, @Path("articleId") articleId: Long, @Query("viewCount") viewCount: Boolean = false diff --git a/data/src/main/kotlin/com/joeloewi/croissant/data/api/dao/GenshinImpactCheckInService.kt b/data/src/main/kotlin/com/joeloewi/croissant/data/api/dao/GenshinImpactCheckInService.kt index 6cec80c4..cd953ecf 100644 --- a/data/src/main/kotlin/com/joeloewi/croissant/data/api/dao/GenshinImpactCheckInService.kt +++ b/data/src/main/kotlin/com/joeloewi/croissant/data/api/dao/GenshinImpactCheckInService.kt @@ -5,15 +5,16 @@ import com.skydoves.sandwich.ApiResponse import retrofit2.http.Body import retrofit2.http.Header import retrofit2.http.POST -import retrofit2.http.Query import java.util.Locale interface GenshinImpactCheckInService { @POST("event/sol/sign") suspend fun attend( - @Query("lang") language: String = Locale.getDefault().toLanguageTag().lowercase(), @Header("Cookie") cookie: String, - @Body params: Map = mapOf("act_id" to "e202102251931481") + @Body params: Map = mapOf( + "act_id" to "e202102251931481", + "lang" to Locale.getDefault().toLanguageTag().lowercase() + ) ): ApiResponse } \ No newline at end of file diff --git a/data/src/main/kotlin/com/joeloewi/croissant/data/api/dao/ZenlessZoneZeroCheckInService.kt b/data/src/main/kotlin/com/joeloewi/croissant/data/api/dao/ZenlessZoneZeroCheckInService.kt new file mode 100644 index 00000000..94c34e30 --- /dev/null +++ b/data/src/main/kotlin/com/joeloewi/croissant/data/api/dao/ZenlessZoneZeroCheckInService.kt @@ -0,0 +1,35 @@ +package com.joeloewi.croissant.data.api.dao + +import com.joeloewi.croissant.data.api.model.response.AttendanceResponse +import com.skydoves.sandwich.ApiResponse +import retrofit2.http.Body +import retrofit2.http.Header +import retrofit2.http.POST +import java.util.Locale + +/* + * Copyright (C) 2024 joeloewi + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +interface ZenlessZoneZeroCheckInService { + + @POST("event/luna/zzz/os/sign") + suspend fun attend( + @Header("Cookie") cookie: String, + @Body params: Map = mapOf( + "act_id" to "e202406031448091", + "lang" to Locale.getDefault().toLanguageTag().lowercase() + ) + ): ApiResponse +} \ No newline at end of file diff --git a/data/src/main/kotlin/com/joeloewi/croissant/data/common/GameIntentGenerator.kt b/data/src/main/kotlin/com/joeloewi/croissant/data/common/GameIntentGenerator.kt index 0ed4ba8c..c6fcc067 100644 --- a/data/src/main/kotlin/com/joeloewi/croissant/data/common/GameIntentGenerator.kt +++ b/data/src/main/kotlin/com/joeloewi/croissant/data/common/GameIntentGenerator.kt @@ -35,6 +35,12 @@ fun generateGameIntent( } } + HoYoLABGame.ZenlessZoneZero -> { + with("com.HoYoverse.Nap") { + this to "market://details?id=${this}".toUri() + } + } + HoYoLABGame.Unknown -> { "" to Uri.EMPTY } diff --git a/data/src/main/kotlin/com/joeloewi/croissant/data/di/ApiModule.kt b/data/src/main/kotlin/com/joeloewi/croissant/data/di/ApiModule.kt index 99b585ff..7fc92912 100644 --- a/data/src/main/kotlin/com/joeloewi/croissant/data/di/ApiModule.kt +++ b/data/src/main/kotlin/com/joeloewi/croissant/data/di/ApiModule.kt @@ -21,6 +21,7 @@ import com.joeloewi.croissant.data.api.dao.ArcaLiveAppService import com.joeloewi.croissant.data.api.dao.CheckInService import com.joeloewi.croissant.data.api.dao.GenshinImpactCheckInService import com.joeloewi.croissant.data.api.dao.HoYoLABService +import com.joeloewi.croissant.data.api.dao.ZenlessZoneZeroCheckInService import com.joeloewi.croissant.data.api.model.response.AttendanceResponse import com.joeloewi.croissant.data.api.model.response.ChangeDataSwitchResponse import com.joeloewi.croissant.data.api.model.response.GameRecordCardResponse @@ -145,4 +146,12 @@ object ApiModule { .baseUrl("https://arca.live/api/app/") .build() .create() + + @Singleton + @Provides + fun providesZenlessZoneZeroCheckInService(retrofitBuilder: Retrofit.Builder): ZenlessZoneZeroCheckInService = + retrofitBuilder + .baseUrl("https://sg-act-nap-api.hoyolab.com") + .build() + .create() } \ No newline at end of file diff --git a/data/src/main/kotlin/com/joeloewi/croissant/data/repository/CheckInRepositoryImpl.kt b/data/src/main/kotlin/com/joeloewi/croissant/data/repository/CheckInRepositoryImpl.kt index 94d25d03..520e394d 100644 --- a/data/src/main/kotlin/com/joeloewi/croissant/data/repository/CheckInRepositoryImpl.kt +++ b/data/src/main/kotlin/com/joeloewi/croissant/data/repository/CheckInRepositoryImpl.kt @@ -43,4 +43,10 @@ class CheckInRepositoryImpl @Inject constructor( ): Result = checkInDataSource.runCatching { attendCheckInHonkaiImpact3rd(cookie).getOrThrow().throwIfNotOk() } + + override suspend fun attendCheckInZenlessZoneZero( + cookie: String + ): Result = checkInDataSource.runCatching { + attendCheckInZenlessZoneZero(cookie).getOrThrow().throwIfNotOk() + } } \ No newline at end of file diff --git a/data/src/main/kotlin/com/joeloewi/croissant/data/repository/remote/CheckInDataSource.kt b/data/src/main/kotlin/com/joeloewi/croissant/data/repository/remote/CheckInDataSource.kt index d24dd85e..af0d9579 100644 --- a/data/src/main/kotlin/com/joeloewi/croissant/data/repository/remote/CheckInDataSource.kt +++ b/data/src/main/kotlin/com/joeloewi/croissant/data/repository/remote/CheckInDataSource.kt @@ -26,4 +26,6 @@ interface CheckInDataSource { suspend fun attendCheckInGenshinImpact(cookie: String): ApiResponse suspend fun attendCheckInHonkaiImpact3rd(cookie: String): ApiResponse + + suspend fun attendCheckInZenlessZoneZero(cookie: String): ApiResponse } \ No newline at end of file diff --git a/data/src/main/kotlin/com/joeloewi/croissant/data/repository/remote/impl/ArcaLiveAppDataSourceImpl.kt b/data/src/main/kotlin/com/joeloewi/croissant/data/repository/remote/impl/ArcaLiveAppDataSourceImpl.kt index 5fb8a164..97d17c7e 100644 --- a/data/src/main/kotlin/com/joeloewi/croissant/data/repository/remote/impl/ArcaLiveAppDataSourceImpl.kt +++ b/data/src/main/kotlin/com/joeloewi/croissant/data/repository/remote/impl/ArcaLiveAppDataSourceImpl.kt @@ -76,6 +76,18 @@ class ArcaLiveAppDataSourceImpl @Inject constructor( } } + HoYoLABGame.ZenlessZoneZero -> { + arcaLiveAppService.getArticle( + slug = "zenlesszonezero", + articleId = 109976603 + ).mapSuccess { + Jsoup.parse(content) + .apply { select("img").remove() } + .html() + .replace("https://oo.pe/", "") + } + } + HoYoLABGame.TearsOfThemis, HoYoLABGame.Unknown -> throw IllegalStateException() } } diff --git a/data/src/main/kotlin/com/joeloewi/croissant/data/repository/remote/impl/CheckInDataSourceImpl.kt b/data/src/main/kotlin/com/joeloewi/croissant/data/repository/remote/impl/CheckInDataSourceImpl.kt index 1201f55c..1b4dfb93 100644 --- a/data/src/main/kotlin/com/joeloewi/croissant/data/repository/remote/impl/CheckInDataSourceImpl.kt +++ b/data/src/main/kotlin/com/joeloewi/croissant/data/repository/remote/impl/CheckInDataSourceImpl.kt @@ -18,6 +18,7 @@ package com.joeloewi.croissant.data.repository.remote.impl import com.joeloewi.croissant.data.api.dao.CheckInService import com.joeloewi.croissant.data.api.dao.GenshinImpactCheckInService +import com.joeloewi.croissant.data.api.dao.ZenlessZoneZeroCheckInService import com.joeloewi.croissant.data.api.model.response.AttendanceResponse import com.joeloewi.croissant.data.repository.remote.CheckInDataSource import com.joeloewi.croissant.data.util.runAndRetryWithExponentialBackOff @@ -28,7 +29,8 @@ import javax.inject.Inject class CheckInDataSourceImpl @Inject constructor( private val checkInService: CheckInService, - private val genshinImpactCheckInService: GenshinImpactCheckInService + private val genshinImpactCheckInService: GenshinImpactCheckInService, + private val zenlessZoneZeroCheckInService: ZenlessZoneZeroCheckInService ) : CheckInDataSource { override suspend fun attend(actId: String, cookie: String): ApiResponse = @@ -53,4 +55,12 @@ class CheckInDataSourceImpl @Inject constructor( checkInService.attendCheckInHonkaiImpact3rd(cookie = cookie) } } + + override suspend fun attendCheckInZenlessZoneZero( + cookie: String + ): ApiResponse = withContext(Dispatchers.IO) { + runAndRetryWithExponentialBackOff { + zenlessZoneZeroCheckInService.attend(cookie = cookie) + } + } } \ No newline at end of file diff --git a/domain/src/main/kotlin/com/joeloewi/croissant/domain/common/HoYoLABGame.kt b/domain/src/main/kotlin/com/joeloewi/croissant/domain/common/HoYoLABGame.kt index f9ceee51..542c40ff 100644 --- a/domain/src/main/kotlin/com/joeloewi/croissant/domain/common/HoYoLABGame.kt +++ b/domain/src/main/kotlin/com/joeloewi/croissant/domain/common/HoYoLABGame.kt @@ -23,19 +23,23 @@ enum class HoYoLABGame( ) { HonkaiImpact3rd( gameId = 1, - gameIconUrl = "https://webstatic-sea.hoyolab.com/communityweb/business/bh3_hoyoverse.png", + gameIconUrl = "https://hyl-static-res-prod.hoyolab.com/communityweb/business/bh3_hoyoverse.png", ), GenshinImpact( gameId = 2, - gameIconUrl = "https://webstatic-sea.hoyolab.com/communityweb/business/ys_hoyoverse.png", + gameIconUrl = "https://hyl-static-res-prod.hoyolab.com/communityweb/business/ys_hoyoverse.png", ), TearsOfThemis( gameId = 4, - gameIconUrl = "https://webstatic-sea.hoyolab.com/communityweb/business/nxx_hoyoverse.png", + gameIconUrl = "https://hyl-static-res-prod.hoyolab.com/communityweb/business/nxx_hoyoverse.png", ), HonkaiStarRail( gameId = 6, - gameIconUrl = "https://webstatic-sea.hoyolab.com/communityweb/business/starrail_hoyoverse.png" + gameIconUrl = "https://hyl-static-res-prod.hoyolab.com/communityweb/business/starrail_hoyoverse.png" + ), + ZenlessZoneZero( + gameId = 8, + gameIconUrl = "https://hyl-static-res-prod.hoyolab.com/communityweb/business/nap.png" ), Unknown( gameId = -1, diff --git a/domain/src/main/kotlin/com/joeloewi/croissant/domain/repository/CheckInRepository.kt b/domain/src/main/kotlin/com/joeloewi/croissant/domain/repository/CheckInRepository.kt index 5e0f0f3b..1b597d1c 100644 --- a/domain/src/main/kotlin/com/joeloewi/croissant/domain/repository/CheckInRepository.kt +++ b/domain/src/main/kotlin/com/joeloewi/croissant/domain/repository/CheckInRepository.kt @@ -25,4 +25,6 @@ interface CheckInRepository { suspend fun attendCheckInGenshinImpact(cookie: String): Result suspend fun attendCheckInHonkaiImpact3rd(cookie: String): Result + + suspend fun attendCheckInZenlessZoneZero(cookie: String): Result } \ No newline at end of file diff --git a/domain/src/main/kotlin/com/joeloewi/croissant/domain/usecase/CheckInUseCase.kt b/domain/src/main/kotlin/com/joeloewi/croissant/domain/usecase/CheckInUseCase.kt index 9ed60d1f..bacec219 100644 --- a/domain/src/main/kotlin/com/joeloewi/croissant/domain/usecase/CheckInUseCase.kt +++ b/domain/src/main/kotlin/com/joeloewi/croissant/domain/usecase/CheckInUseCase.kt @@ -30,7 +30,7 @@ sealed class CheckInUseCase { ) = checkInRepository.attend(actId, cookie) } - class AttendCheckInHonkaiStarRail @Inject constructor( + class AttendCheckInHonkaiStarRailUseCase @Inject constructor( private val checkInRepository: CheckInRepository ) : CheckInUseCase() { suspend operator fun invoke( @@ -54,4 +54,12 @@ sealed class CheckInUseCase { cookie: String ) = checkInRepository.attendCheckInGenshinImpact(cookie) } + + class AttendCheckInZenlessZoneZeroUseCase @Inject constructor( + private val checkInRepository: CheckInRepository + ) : CheckInUseCase() { + suspend operator fun invoke( + cookie: String + ) = checkInRepository.attendCheckInZenlessZoneZero(cookie) + } } \ No newline at end of file