mirror of
https://github.com/tachiyomiorg/tachiyomi.git
synced 2024-06-28 02:06:06 +02:00
3d178737b1
Also retain ordering of added repos.
112 lines
3.3 KiB
Kotlin
112 lines
3.3 KiB
Kotlin
package eu.kanade.presentation.more.settings.screen.browse
|
|
|
|
import androidx.compose.runtime.Immutable
|
|
import cafe.adriel.voyager.core.model.StateScreenModel
|
|
import cafe.adriel.voyager.core.model.screenModelScope
|
|
import dev.icerock.moko.resources.StringResource
|
|
import eu.kanade.domain.extension.interactor.CreateExtensionRepo
|
|
import eu.kanade.domain.extension.interactor.DeleteExtensionRepo
|
|
import eu.kanade.domain.extension.interactor.GetExtensionRepos
|
|
import kotlinx.collections.immutable.ImmutableSet
|
|
import kotlinx.collections.immutable.toImmutableSet
|
|
import kotlinx.coroutines.channels.Channel
|
|
import kotlinx.coroutines.flow.collectLatest
|
|
import kotlinx.coroutines.flow.receiveAsFlow
|
|
import kotlinx.coroutines.flow.update
|
|
import tachiyomi.core.util.lang.launchIO
|
|
import tachiyomi.i18n.MR
|
|
import uy.kohesive.injekt.Injekt
|
|
import uy.kohesive.injekt.api.get
|
|
|
|
class ExtensionReposScreenModel(
|
|
private val getExtensionRepos: GetExtensionRepos = Injekt.get(),
|
|
private val createExtensionRepo: CreateExtensionRepo = Injekt.get(),
|
|
private val deleteExtensionRepo: DeleteExtensionRepo = Injekt.get(),
|
|
) : StateScreenModel<RepoScreenState>(RepoScreenState.Loading) {
|
|
|
|
private val _events: Channel<RepoEvent> = Channel(Int.MAX_VALUE)
|
|
val events = _events.receiveAsFlow()
|
|
|
|
init {
|
|
screenModelScope.launchIO {
|
|
getExtensionRepos.subscribe()
|
|
.collectLatest { repos ->
|
|
mutableState.update {
|
|
RepoScreenState.Success(
|
|
repos = repos.toImmutableSet(),
|
|
)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Creates and adds a new repo to the database.
|
|
*
|
|
* @param name The name of the repo to create.
|
|
*/
|
|
fun createRepo(name: String) {
|
|
screenModelScope.launchIO {
|
|
when (createExtensionRepo.await(name)) {
|
|
is CreateExtensionRepo.Result.InvalidUrl -> _events.send(RepoEvent.InvalidUrl)
|
|
else -> {}
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Deletes the given repo from the database.
|
|
*
|
|
* @param repo The repo to delete.
|
|
*/
|
|
fun deleteRepo(repo: String) {
|
|
screenModelScope.launchIO {
|
|
deleteExtensionRepo.await(repo)
|
|
}
|
|
}
|
|
|
|
fun showDialog(dialog: RepoDialog) {
|
|
mutableState.update {
|
|
when (it) {
|
|
RepoScreenState.Loading -> it
|
|
is RepoScreenState.Success -> it.copy(dialog = dialog)
|
|
}
|
|
}
|
|
}
|
|
|
|
fun dismissDialog() {
|
|
mutableState.update {
|
|
when (it) {
|
|
RepoScreenState.Loading -> it
|
|
is RepoScreenState.Success -> it.copy(dialog = null)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
sealed class RepoEvent {
|
|
sealed class LocalizedMessage(val stringRes: StringResource) : RepoEvent()
|
|
data object InvalidUrl : LocalizedMessage(MR.strings.invalid_repo_name)
|
|
}
|
|
|
|
sealed class RepoDialog {
|
|
data object Create : RepoDialog()
|
|
data class Delete(val repo: String) : RepoDialog()
|
|
}
|
|
|
|
sealed class RepoScreenState {
|
|
|
|
@Immutable
|
|
data object Loading : RepoScreenState()
|
|
|
|
@Immutable
|
|
data class Success(
|
|
val repos: ImmutableSet<String>,
|
|
val dialog: RepoDialog? = null,
|
|
) : RepoScreenState() {
|
|
|
|
val isEmpty: Boolean
|
|
get() = repos.isEmpty()
|
|
}
|
|
}
|