android 在Jetpack Compose中通过同时触摸惰性列项来下载文件

bgibtngc  于 2024-01-04  发布在  Android
关注(0)|答案(1)|浏览(156)

我有一个列表,当我触摸一个项目时,我希望下载与该项目相关的文件。我编写了下载文件的代码,但我有一个问题。用户可能同时触摸几个项目,并希望同时下载它们。我的代码的问题是,当我同时触摸几个项目时,下载操作相互干扰。
ViewModel类:

  1. class BusinessViewModel(private val userRepository: UserRepository) : ViewModel() {
  2. val downloadSqliteDatabaseData: MutableState<State<String>> = mutableStateOf(State.Empty)
  3. fun downloadSqliteDatabase(file: File?, businessId: Long) {
  4. viewModelScope.launch {
  5. userRepository.downloadSqliteDatabase(businessId).onStart {
  6. downloadSqliteDatabaseData.value = State.Loading
  7. }.catch {
  8. downloadSqliteDatabaseData.value = State.Error("ERROR", MyHttpStatusCode.NetworkError)
  9. }.collect {
  10. downloadSqliteDatabaseData.value = when (it.status) {
  11. HttpStatusCode.OK -> {
  12. try {
  13. val data: ByteArray = it.body()
  14. file?.writeBytes(data)
  15. State.Success("", it.status)
  16. } catch (ex: Exception) {
  17. State.Error("ERROR", MyHttpStatusCode.ConvertDataError)
  18. }
  19. }
  20. MyHttpStatusCode.InvalidUser -> {
  21. State.Error("ERROR", it.status)
  22. }
  23. else -> {
  24. State.Error("ERROR", it.status)
  25. }
  26. }
  27. }
  28. }
  29. }
  30. }

字符串
合成功能:

  1. val downloadSqliteDatabaseData by remember { viewModel.downloadSqliteDatabaseData }
  2. DisposableEffect(downloadSqliteDatabaseData) {
  3. when (downloadSqliteDatabaseData) {
  4. is State.Empty -> {}
  5. is State.Loading -> {
  6. //downloadDatabaseLoading.value = true
  7. }
  8. is State.Success -> {
  9. //downloadDatabaseLoading.value = false
  10. }
  11. is State.Error -> {
  12. println("Error")
  13. //downloadDatabaseLoading.value = false
  14. }
  15. }
  16. onDispose {
  17. }
  18. }
  19. LazyColumn(
  20. state = lazyListState,
  21. modifier = Modifier.fillMaxSize(),
  22. horizontalAlignment = Alignment.CenterHorizontally
  23. ) {
  24. itemsIndexed(
  25. items = businessList,
  26. key = { _, item -> item.id },
  27. ) { index, item ->
  28. ClickBox(
  29. modifier = Modifier
  30. .padding(start = 15.dp, end = 15.dp)
  31. .fillMaxWidth()
  32. .height(120.dp),
  33. onClick = {
  34. val dbPath = File("test.txt")
  35. viewModel.downloadSqliteDatabase(dbPath, item.id)
  36. }
  37. ) {
  38. //....
  39. }
  40. }
  41. }

44u64gxh

44u64gxh1#

你使当前文件的下载成为整个UI状态的焦点。

  1. val downloadSqliteDatabaseData: MutableState<State<String>>

字符串
如果要处理多个,则需要向下传递到列表中项目的状态。

  1. sealed class BusinessState {
  2. object Loading : BusinessState()
  3. data class Loaded(val items: List<DatabaseUiItem>) : BusinessState()
  4. }
  5. data class DatabaseUiItem(
  6. val id: Long,
  7. val path: String,
  8. val state: BusinessItemState = BusinessItemState.None
  9. )
  10. sealed class BusinessItemState {
  11. object None : BusinessItemState()
  12. object Loading : BusinessItemState()
  13. object Loaded : BusinessItemState()
  14. object Error : BusinessItemState()
  15. }


下面是工作示例的其余部分

  1. class BusinessViewModel() : ViewModel() {
  2. private val failed: MutableStateFlow<List<String>> = MutableStateFlow(emptyList())
  3. private val inProgress: MutableStateFlow<List<String>> = MutableStateFlow(emptyList())
  4. private val downloaded: MutableStateFlow<List<String>> = MutableStateFlow(emptyList())
  5. private val _state: MutableStateFlow<BusinessState> = MutableStateFlow(BusinessState.Loading)
  6. val state = _state.asStateFlow()
  7. init {
  8. val items = flowOf(listOf(0L to "test_0.txt", 1L to "test_1.txt", 2L to "test_2.txt", 3L to "test_3.txt" )
  9. )
  10. // Here I am going to assume that you are fetching the initial list from somewhere
  11. viewModelScope.launch(Dispatchers.IO) {
  12. delay(200)
  13. combine(items, inProgress, failed, downloaded) { items, progress, errors, loaded ->
  14. BusinessState.Loaded(items = items.map { (id: Long, path: String) ->
  15. DatabaseUiItem(
  16. id = id,
  17. path = path,
  18. state = when {
  19. loaded.contains(path) -> BusinessItemState.Loaded
  20. progress.contains(path) -> BusinessItemState.Loading
  21. errors.contains(path) -> BusinessItemState.Error
  22. else -> BusinessItemState.None
  23. }
  24. )
  25. })
  26. }.collect { newState ->
  27. _state.update { newState }
  28. }
  29. }
  30. }
  31. fun downloadSqliteDatabase(file: String?, businessId: Long) {
  32. if (file == null) return
  33. inProgress.update { oldList -> mutableListOf<String>().apply { addAll(oldList) } + file }
  34. viewModelScope.launch(Dispatchers.IO) {
  35. downloadDatabase(file).onSuccess {
  36. inProgress.update { oldList -> mutableListOf<String>().apply { remove(file) } }
  37. downloaded.update { oldList -> mutableListOf<String>().apply { addAll(oldList) } + file }
  38. }.onFailure {
  39. inProgress.update { oldList -> mutableListOf<String>().apply { remove(file) } }
  40. failed.update { oldList -> mutableListOf<String>().apply { addAll(oldList) } + file }
  41. }
  42. }
  43. }
  44. private suspend fun downloadDatabase(file: String): Result<String> = withContext(Dispatchers.IO) {
  45. try {
  46. delay(3000L)
  47. if (file == "test_1.txt") {
  48. throw Exception("")
  49. }
  50. Result.success(file)
  51. } catch (ex: Exception) {
  52. Result.failure(ex)
  53. }
  54. }
  55. }
  56. @Composable
  57. fun BusinessList(
  58. items: List<DatabaseUiItem>,
  59. onItemClick: (String, Long) -> Unit
  60. ) {
  61. LazyColumn() {
  62. itemsIndexed() { index, item ->
  63. Button(
  64. modifier = Modifier.fillMaxWidth(),
  65. onClick = { onItemClick(item.path, item.id) }
  66. ) {
  67. Text(text = item.path)
  68. when (item.state) {
  69. BusinessItemState.Error -> Icon(imageVector = Icons.Default.Error, contentDescription = null)
  70. BusinessItemState.Loaded -> Icon(imageVector = Icons.Default.Check, contentDescription = null)
  71. BusinessItemState.Loading -> CircularProgressIndicator()
  72. BusinessItemState.None -> Unit
  73. }
  74. }
  75. }
  76. }
  77. }

展开查看全部

相关问题