diff --git a/data_local/src/main/java/com/example/util/simpletimetracker/data_local/di/DataLocalModuleBinds.kt b/data_local/src/main/java/com/example/util/simpletimetracker/data_local/di/DataLocalModuleBinds.kt index 9c88418c0..bc287fce5 100644 --- a/data_local/src/main/java/com/example/util/simpletimetracker/data_local/di/DataLocalModuleBinds.kt +++ b/data_local/src/main/java/com/example/util/simpletimetracker/data_local/di/DataLocalModuleBinds.kt @@ -21,6 +21,7 @@ import com.example.util.simpletimetracker.data_local.repo.RecordTypeToDefaultTag import com.example.util.simpletimetracker.data_local.repo.RecordTypeToTagRepoImpl import com.example.util.simpletimetracker.data_local.repo.RunningRecordRepoImpl import com.example.util.simpletimetracker.data_local.repo.RunningRecordToRecordTagRepoImpl +import com.example.util.simpletimetracker.data_local.resolver.BackupPartialRepoImpl import com.example.util.simpletimetracker.data_local.resolver.BackupRepoImpl import com.example.util.simpletimetracker.data_local.resolver.CsvRepoImpl import com.example.util.simpletimetracker.data_local.resolver.IcsRepoImpl @@ -42,6 +43,7 @@ import com.example.util.simpletimetracker.domain.repo.RecordTypeToDefaultTagRepo import com.example.util.simpletimetracker.domain.repo.RecordTypeToTagRepo import com.example.util.simpletimetracker.domain.repo.RunningRecordRepo import com.example.util.simpletimetracker.domain.repo.RunningRecordToRecordTagRepo +import com.example.util.simpletimetracker.domain.resolver.BackupPartialRepo import com.example.util.simpletimetracker.domain.resolver.BackupRepo import com.example.util.simpletimetracker.domain.resolver.CsvRepo import com.example.util.simpletimetracker.domain.resolver.IcsRepo @@ -76,6 +78,10 @@ interface DataLocalModuleBinds { @Singleton fun bindBackupRepo(impl: BackupRepoImpl): BackupRepo + @Binds + @Singleton + fun bindBackupPartialRepo(impl: BackupPartialRepoImpl): BackupPartialRepo + @Binds @Singleton fun bindCsvRepo(impl: CsvRepoImpl): CsvRepo diff --git a/data_local/src/main/java/com/example/util/simpletimetracker/data_local/resolver/BackupPartialRepoImpl.kt b/data_local/src/main/java/com/example/util/simpletimetracker/data_local/resolver/BackupPartialRepoImpl.kt new file mode 100644 index 000000000..9a6022920 --- /dev/null +++ b/data_local/src/main/java/com/example/util/simpletimetracker/data_local/resolver/BackupPartialRepoImpl.kt @@ -0,0 +1,499 @@ +package com.example.util.simpletimetracker.data_local.resolver + +import com.example.util.simpletimetracker.core.R +import com.example.util.simpletimetracker.core.repo.ResourceRepo +import com.example.util.simpletimetracker.data_local.resolver.BackupRepoImpl.DataHandler +import com.example.util.simpletimetracker.domain.extension.orEmpty +import com.example.util.simpletimetracker.domain.extension.orZero +import com.example.util.simpletimetracker.domain.model.ActivityFilter +import com.example.util.simpletimetracker.domain.model.BackupOptionsData +import com.example.util.simpletimetracker.domain.model.Category +import com.example.util.simpletimetracker.domain.model.ComplexRule +import com.example.util.simpletimetracker.domain.model.FavouriteColor +import com.example.util.simpletimetracker.domain.model.FavouriteComment +import com.example.util.simpletimetracker.domain.model.FavouriteIcon +import com.example.util.simpletimetracker.domain.model.PartialBackupRestoreData +import com.example.util.simpletimetracker.domain.model.Record +import com.example.util.simpletimetracker.domain.model.RecordTag +import com.example.util.simpletimetracker.domain.model.RecordToRecordTag +import com.example.util.simpletimetracker.domain.model.RecordType +import com.example.util.simpletimetracker.domain.model.RecordTypeCategory +import com.example.util.simpletimetracker.domain.model.RecordTypeGoal +import com.example.util.simpletimetracker.domain.model.RecordTypeToDefaultTag +import com.example.util.simpletimetracker.domain.model.RecordTypeToTag +import com.example.util.simpletimetracker.domain.model.getExistingValues +import com.example.util.simpletimetracker.domain.model.getNotExistingValues +import com.example.util.simpletimetracker.domain.repo.ActivityFilterRepo +import com.example.util.simpletimetracker.domain.repo.CategoryRepo +import com.example.util.simpletimetracker.domain.repo.ComplexRuleRepo +import com.example.util.simpletimetracker.domain.repo.FavouriteColorRepo +import com.example.util.simpletimetracker.domain.repo.FavouriteCommentRepo +import com.example.util.simpletimetracker.domain.repo.FavouriteIconRepo +import com.example.util.simpletimetracker.domain.repo.RecordRepo +import com.example.util.simpletimetracker.domain.repo.RecordTagRepo +import com.example.util.simpletimetracker.domain.repo.RecordToRecordTagRepo +import com.example.util.simpletimetracker.domain.repo.RecordTypeCategoryRepo +import com.example.util.simpletimetracker.domain.repo.RecordTypeGoalRepo +import com.example.util.simpletimetracker.domain.repo.RecordTypeRepo +import com.example.util.simpletimetracker.domain.repo.RecordTypeToDefaultTagRepo +import com.example.util.simpletimetracker.domain.repo.RecordTypeToTagRepo +import com.example.util.simpletimetracker.domain.resolver.BackupPartialRepo +import com.example.util.simpletimetracker.domain.resolver.ResultCode +import kotlinx.coroutines.Dispatchers +import kotlinx.coroutines.withContext +import javax.inject.Inject + +class BackupPartialRepoImpl @Inject constructor( + private val backupRepo: BackupRepoImpl, + private val recordTypeRepo: RecordTypeRepo, + private val recordRepo: RecordRepo, + private val categoryRepo: CategoryRepo, + private val recordTypeCategoryRepo: RecordTypeCategoryRepo, + private val recordTypeToTagRepo: RecordTypeToTagRepo, + private val recordTypeToDefaultTagRepo: RecordTypeToDefaultTagRepo, + private val recordToRecordTagRepo: RecordToRecordTagRepo, + private val recordTagRepo: RecordTagRepo, + private val activityFilterRepo: ActivityFilterRepo, + private val favouriteCommentRepo: FavouriteCommentRepo, + private val favouriteColorRepo: FavouriteColorRepo, + private val favouriteIconRepo: FavouriteIconRepo, + private val recordTypeGoalRepo: RecordTypeGoalRepo, + private val complexRuleRepo: ComplexRuleRepo, + private val resourceRepo: ResourceRepo, +) : BackupPartialRepo { + + // Replace original id with 0 to add instead of replacing. + // Replace original ids in other data with actual id after adding. + override suspend fun partialRestoreBackupFile( + params: BackupOptionsData.Custom, + ): ResultCode = withContext(Dispatchers.IO) { + val originalTypeIdToAddedId: MutableMap = params.data.types + .values.getExistingValues().associate { it.id to it.id }.toMutableMap() + val originalCategoryIdToAddedId: MutableMap = params.data.categories + .values.getExistingValues().associate { it.id to it.id }.toMutableMap() + val originalTagIdToAddedId: MutableMap = params.data.tags + .values.getExistingValues().associate { it.id to it.id }.toMutableMap() + val originalRecordIdToAddedId: MutableMap = params.data.records + .values.getExistingValues().associate { it.id to it.id }.toMutableMap() + + params.data.types.values.getNotExistingValues().forEach { type -> + val originalId = type.id + val addedId = type.copy( + id = 0, + ).let { recordTypeRepo.add(it) } + originalTypeIdToAddedId[originalId] = addedId + } + params.data.records.values.getNotExistingValues().forEach { record -> + val originalId = record.id + val newTypeId = originalTypeIdToAddedId[record.typeId] + ?: return@forEach + val addedId = record.copy( + id = 0, + typeId = newTypeId, + ).let { recordRepo.add(it) } + originalRecordIdToAddedId[originalId] = addedId + } + params.data.categories.values.getNotExistingValues().forEach { category -> + val originalId = category.id + val addedId = category.copy( + id = 0, + ).let { categoryRepo.add(it) } + originalCategoryIdToAddedId[originalId] = addedId + } + params.data.typeToCategory.getNotExistingValues().forEach { typeToCategory -> + val newTypeId = originalTypeIdToAddedId[typeToCategory.recordTypeId] + ?: return@forEach + val newCategoryId = originalCategoryIdToAddedId[typeToCategory.categoryId] + ?: return@forEach + typeToCategory.copy( + recordTypeId = newTypeId, + categoryId = newCategoryId, + ).let { recordTypeCategoryRepo.add(it) } + } + params.data.tags.values.getNotExistingValues().forEach { tag -> + val originalId = tag.id + val newColorSource = originalTypeIdToAddedId[tag.iconColorSource].orZero() + val addedId = tag.copy( + id = 0, + iconColorSource = newColorSource, + ).let { recordTagRepo.add(it) } + originalTagIdToAddedId[originalId] = addedId + } + params.data.recordToTag.getNotExistingValues().forEach { recordToTag -> + val newRecordId = originalRecordIdToAddedId[recordToTag.recordId] + ?: return@forEach + val newTagId = originalTagIdToAddedId[recordToTag.recordTagId] + ?: return@forEach + recordToTag.copy( + recordId = newRecordId, + recordTagId = newTagId, + ).let { recordToRecordTagRepo.add(it) } + } + params.data.typeToTag.getNotExistingValues().forEach { typeToTag -> + val newTypeId = originalTypeIdToAddedId[typeToTag.recordTypeId] + ?: return@forEach + val newTagId = originalTagIdToAddedId[typeToTag.tagId] + ?: return@forEach + typeToTag.copy( + recordTypeId = newTypeId, + tagId = newTagId, + ).let { recordTypeToTagRepo.add(it) } + } + params.data.typeToDefaultTag.getNotExistingValues().forEach { typeToDefaultTag -> + val newTypeId = originalTypeIdToAddedId[typeToDefaultTag.recordTypeId] + ?: return@forEach + val newTagId = originalTagIdToAddedId[typeToDefaultTag.tagId] + ?: return@forEach + typeToDefaultTag.copy( + recordTypeId = newTypeId, + tagId = newTagId, + ).let { recordTypeToDefaultTagRepo.add(it) } + } + params.data.activityFilters.values.getNotExistingValues().forEach { activityFilter -> + val newTypeIds = activityFilter.selectedIds + .mapNotNull { originalTypeIdToAddedId[it] } + activityFilter.copy( + id = 0, + selectedIds = newTypeIds, + ).let { activityFilterRepo.add(it) } + } + params.data.favouriteComments.values.getNotExistingValues().forEach { favComment -> + favComment.copy( + id = 0, + ).let { favouriteCommentRepo.add(it) } + } + params.data.favouriteColors.values.getNotExistingValues().forEach { favColor -> + favColor.copy( + id = 0, + ).let { favouriteColorRepo.add(it) } + } + params.data.favouriteIcon.values.getNotExistingValues().forEach { favIcon -> + favIcon.copy( + id = 0, + ).let { favouriteIconRepo.add(it) } + } + params.data.goals.values.getNotExistingValues().forEach { goal -> + val newId = when (val idData = goal.idData) { + is RecordTypeGoal.IdData.Type -> originalTypeIdToAddedId[idData.value] + ?.let(RecordTypeGoal.IdData::Type) + is RecordTypeGoal.IdData.Category -> originalCategoryIdToAddedId[idData.value] + ?.let(RecordTypeGoal.IdData::Category) + } ?: return@forEach + goal.copy( + id = 0, + idData = newId, + ).let { recordTypeGoalRepo.add(it) } + } + params.data.rules.values.getNotExistingValues().forEach { rule -> + val newStartingTypeIds = rule.conditionStartingTypeIds + .mapNotNull { originalTypeIdToAddedId[it] }.toSet() + val newCurrentTypeIds = rule.conditionCurrentTypeIds + .mapNotNull { originalTypeIdToAddedId[it] }.toSet() + val newAssignTagIds = rule.actionAssignTagIds + .mapNotNull { originalTagIdToAddedId[it] }.toSet() + rule.copy( + id = 0, + actionAssignTagIds = newAssignTagIds, + conditionStartingTypeIds = newStartingTypeIds, + conditionCurrentTypeIds = newCurrentTypeIds, + ).takeIf { + it.hasActions && it.hasConditions + }?.let { complexRuleRepo.add(it) } + } + return@withContext ResultCode.Success( + resourceRepo.getString(R.string.message_import_complete), + ) + } + + /** + * Marks data from the backup file if it is already exist, + * to skip later on import. + */ + @Suppress("ComplexRedundantLet") + override suspend fun readBackupFile( + uriString: String, + ): Pair = withContext(Dispatchers.IO) { + // Result data + val types: MutableList = mutableListOf() + val typesCurrent: List = recordTypeRepo.getAll() + val records: MutableList = mutableListOf() + val recordsCurrent: List = recordRepo.getAll().map { it.copy(tagIds = emptyList()) } + val categories: MutableList = mutableListOf() + val categoriesCurrent: List = categoryRepo.getAll() + val typeToCategory: MutableList = mutableListOf() + val typeToCategoryCurrent: List = recordTypeCategoryRepo.getAll() + val tags: MutableList = mutableListOf() + val tagsCurrent: List = recordTagRepo.getAll() + val recordToTag: MutableList = mutableListOf() + val recordToTagCurrent: List = recordToRecordTagRepo.getAll() + val typeToTag: MutableList = mutableListOf() + val typeToTagCurrent: List = recordTypeToTagRepo.getAll() + val typeToDefaultTag: MutableList = mutableListOf() + val typeToDefaultTagCurrent: List = recordTypeToDefaultTagRepo.getAll() + val activityFilters: MutableList = mutableListOf() + val activityFiltersCurrent: List = activityFilterRepo.getAll() + val favouriteComments: MutableList = mutableListOf() + val favouriteCommentsCurrent: List = favouriteCommentRepo.getAll() + val favouriteColors: MutableList = mutableListOf() + val favouriteColorsCurrent: List = favouriteColorRepo.getAll() + val favouriteIcon: MutableList = mutableListOf() + val favouriteIconCurrent: List = favouriteIconRepo.getAll() + val goals: MutableList = mutableListOf() + val goalsCurrent: List = recordTypeGoalRepo.getAllTypeGoals() + val rules: MutableList = mutableListOf() + val rulesCurrent: List = complexRuleRepo.getAll() + val settings: MutableList> = mutableListOf() + + val result = backupRepo.readBackup( + uriString = uriString, + successCodeMessage = null, + errorCodeMessage = R.string.settings_file_open_error, + clearData = false, + migrateTags = { + tags += backupRepo.migrateTags( + types = types, + data = it, + ) + }, + dataHandler = DataHandler( + types = types::add, + records = records::add, + categories = categories::add, + typeToCategory = typeToCategory::add, + tags = tags::add, + recordToTag = recordToTag::add, + typeToTag = typeToTag::add, + typeToDefaultTag = typeToDefaultTag::add, + activityFilters = activityFilters::add, + favouriteComments = favouriteComments::add, + favouriteColors = favouriteColors::add, + favouriteIcon = favouriteIcon::add, + goals = goals::add, + rules = rules::add, + settings = settings::add, + ), + ) + + // If type already exist - need to replace typeId to existing typeId in records etc. + val (newTypes, originalTypeIdToExistingId) = types.let { + mapToHolder(it, typesCurrent) + } + + val (newRecords, originalRecordIdToExistingId) = records.mapNotNull { item -> + val newTypeId = originalTypeIdToExistingId[item.typeId] + ?: return@mapNotNull null + item.copy(typeId = newTypeId) + }.let { + mapToHolder(it, recordsCurrent) + } + + val (newCategories, originalCategoryIdToExistingId) = categories.let { + mapToHolder(it, categoriesCurrent) + } + + val newTypeToCategory = typeToCategory.mapNotNull { item -> + val newTypeId = originalTypeIdToExistingId[item.recordTypeId] + ?: return@mapNotNull null + val newCategoryId = originalCategoryIdToExistingId[item.categoryId] + ?: return@mapNotNull null + item.copy( + recordTypeId = newTypeId, + categoryId = newCategoryId, + ) + }.let { + mapToHolder(it, typeToCategoryCurrent) + }.list + + val (newTags, originalTagIdToExistingId) = tags.map { item -> + val newColorSource = originalTypeIdToExistingId[item.iconColorSource].orZero() + item.copy( + iconColorSource = newColorSource, + ) + }.let { + mapToHolder(it, tagsCurrent) + } + + val newRecordToTag = recordToTag.mapNotNull { item -> + val newRecordId = originalRecordIdToExistingId[item.recordId] + ?: return@mapNotNull null + val newTagId = originalTagIdToExistingId[item.recordTagId] + ?: return@mapNotNull null + item.copy( + recordId = newRecordId, + recordTagId = newTagId, + ) + }.let { + mapToHolder(it, recordToTagCurrent) + }.list + + val newTypeToTag = typeToTag.mapNotNull { item -> + val newTypeId = originalTypeIdToExistingId[item.recordTypeId] + ?: return@mapNotNull null + val newTagId = originalTagIdToExistingId[item.tagId] + ?: return@mapNotNull null + item.copy( + recordTypeId = newTypeId, + tagId = newTagId, + ) + }.let { + mapToHolder(it, typeToTagCurrent) + }.list + + val newTypeToDefaultTag = typeToDefaultTag.mapNotNull { item -> + val newTypeId = originalTypeIdToExistingId[item.recordTypeId] + ?: return@mapNotNull null + val newTagId = originalTagIdToExistingId[item.tagId] + ?: return@mapNotNull null + item.copy( + recordTypeId = newTypeId, + tagId = newTagId, + ) + }.let { + mapToHolder(it, typeToDefaultTagCurrent) + }.list + + val newActivityFilters = activityFilters.map { item -> + val newTypeIds = item.selectedIds + .mapNotNull { originalTypeIdToExistingId[it] } + item.copy( + selectedIds = newTypeIds, + ) + }.let { + mapToHolder(it, activityFiltersCurrent) + }.list + + val newFavouriteComments = favouriteComments.let { + mapToHolder(it, favouriteCommentsCurrent) + }.list + + val newFavouriteColors = favouriteColors.let { + mapToHolder(it, favouriteColorsCurrent) + }.list + + val newFavouriteIcon = favouriteIcon.let { + mapToHolder(it, favouriteIconCurrent) + }.list + + val newGoals = goals.mapNotNull { item -> + val newId = when (val idData = item.idData) { + is RecordTypeGoal.IdData.Type -> originalTypeIdToExistingId[idData.value] + ?.let(RecordTypeGoal.IdData::Type) + is RecordTypeGoal.IdData.Category -> originalCategoryIdToExistingId[idData.value] + ?.let(RecordTypeGoal.IdData::Category) + } ?: return@mapNotNull null + item.copy( + idData = newId, + ) + }.let { + mapToHolder(it, goalsCurrent) + }.list + + val newRules = rules.mapNotNull { item -> + val newStartingTypeIds = item.conditionStartingTypeIds + .mapNotNull { originalTypeIdToExistingId[it] }.toSet() + val newCurrentTypeIds = item.conditionCurrentTypeIds + .mapNotNull { originalTypeIdToExistingId[it] }.toSet() + val newAssignTagIds = item.actionAssignTagIds + .mapNotNull { originalTagIdToExistingId[it] }.toSet() + item.copy( + actionAssignTagIds = newAssignTagIds, + conditionStartingTypeIds = newStartingTypeIds, + conditionCurrentTypeIds = newCurrentTypeIds, + ).takeIf { + it.hasActions && it.hasConditions + } + }.let { + mapToHolder(it, rulesCurrent) + }.list + + // Fill tags after all data processed, with actual tagIds. + val newRecordToTagMap = newRecordToTag.groupBy { + it.data.recordId + } + val newRecordsWithTags = newRecords.map { record -> + val thisTags = newRecordToTagMap[record.data.id].orEmpty().map { it.data } + val newData = record.data.copy(tagIds = thisTags.map(RecordToRecordTag::recordTagId)) + record.copy(data = newData) + } + + result to PartialBackupRestoreData( + types = newTypes.associateBy { it.data.id }, + records = newRecordsWithTags.associateBy { it.data.id }, + categories = newCategories.associateBy { it.data.id }, + typeToCategory = newTypeToCategory, + tags = newTags.associateBy { it.data.id }, + recordToTag = newRecordToTag, + typeToTag = newTypeToTag, + typeToDefaultTag = newTypeToDefaultTag, + activityFilters = newActivityFilters.associateBy { it.data.id }, + favouriteComments = newFavouriteComments.associateBy { it.data.id }, + favouriteColors = newFavouriteColors.associateBy { it.data.id }, + favouriteIcon = newFavouriteIcon.associateBy { it.data.id }, + goals = newGoals.associateBy { it.data.id }, + rules = newRules.associateBy { it.data.id }, + ) + } + + @Suppress("UNCHECKED_CAST") + private fun getIdData( + value: T?, + ): IdData { + return when (value) { + is RecordType -> IdData({ copy(id = it) }, { id }) + is Record -> IdData({ copy(id = it) }, { id }) + is Category -> IdData({ copy(id = it) }, { id }) + is RecordTypeCategory -> IdData({ this }, { 0 }) + is RecordTag -> IdData({ copy(id = it) }, { id }) + is RecordToRecordTag -> IdData({ this }, { 0 }) + is RecordTypeToTag -> IdData({ this }, { 0 }) + is RecordTypeToDefaultTag -> IdData({ this }, { 0 }) + is ActivityFilter -> IdData({ copy(id = it) }, { id }) + is FavouriteComment -> IdData({ copy(id = it) }, { id }) + is FavouriteColor -> IdData({ copy(id = it) }, { id }) + is FavouriteIcon -> IdData({ copy(id = it) }, { id }) + is RecordTypeGoal -> IdData({ copy(id = it) }, { id }) + is ComplexRule -> IdData({ copy(id = it) }, { id }) + else -> IdData({ this }, { 0 }) + } as IdData + } + + private fun mapToHolder( + dataFromFile: List, + currentData: List, + ): ReadData { + if (dataFromFile.isEmpty()) return ReadData(emptyList(), emptyMap()) + val idData = getIdData(dataFromFile.firstOrNull()) + val replaceId: T.(Long) -> T = { id -> idData.idSetter(this, id) } + val id: T.() -> Long = { idData.idGetter(this) } + val currentDataClean: Map = currentData.associate { it.replaceId(0) to it.id() } + val originalIdsToExistingId = mutableMapOf() + val list = dataFromFile.map { item -> + val cleanItem = item.replaceId(0) + val existingId = currentDataClean[cleanItem] + val itemId = item.id() + if (itemId != 0L) { + originalIdsToExistingId[itemId] = existingId ?: itemId + } + val newItem = if (existingId != null) { + item.replaceId(existingId) + } else { + item + } + PartialBackupRestoreData.Holder( + exist = existingId != null, + data = newItem, + ) + } + return ReadData(list, originalIdsToExistingId) + } + + private data class ReadData( + val list: List>, + val originalIdsToExistingId: Map, + ) + + private data class IdData( + val idSetter: T.(Long) -> T, + val idGetter: T.() -> Long, + ) +} \ No newline at end of file diff --git a/data_local/src/main/java/com/example/util/simpletimetracker/data_local/resolver/BackupRepoImpl.kt b/data_local/src/main/java/com/example/util/simpletimetracker/data_local/resolver/BackupRepoImpl.kt index e35b26618..d830926cc 100644 --- a/data_local/src/main/java/com/example/util/simpletimetracker/data_local/resolver/BackupRepoImpl.kt +++ b/data_local/src/main/java/com/example/util/simpletimetracker/data_local/resolver/BackupRepoImpl.kt @@ -19,7 +19,6 @@ import com.example.util.simpletimetracker.domain.model.DayOfWeek import com.example.util.simpletimetracker.domain.model.FavouriteColor import com.example.util.simpletimetracker.domain.model.FavouriteComment import com.example.util.simpletimetracker.domain.model.FavouriteIcon -import com.example.util.simpletimetracker.domain.model.PartialBackupRestoreData import com.example.util.simpletimetracker.domain.model.Record import com.example.util.simpletimetracker.domain.model.RecordTag import com.example.util.simpletimetracker.domain.model.RecordToRecordTag @@ -216,218 +215,7 @@ class BackupRepoImpl @Inject constructor( ) } - // Replace original id with 0 to add instead of replacing. - // Replace original ids in other data with actual id after adding. - override suspend fun partialRestoreBackupFile( - params: BackupOptionsData.Custom, - ): ResultCode = withContext(Dispatchers.IO) { - val originalTypeIdToAddedId: MutableMap = mutableMapOf() - val originalCategoryIdToAddedId: MutableMap = mutableMapOf() - val originalTagIdToAddedId: MutableMap = mutableMapOf() - val originalRecordIdToAddedId: MutableMap = mutableMapOf() - - params.data.types.values.forEach { type -> - val originalId = type.id - val addedId = type.copy( - id = 0, - ).let { recordTypeRepo.add(it) } - originalTypeIdToAddedId[originalId] = addedId - } - params.data.records.values.forEach { record -> - val originalId = record.id - val newTypeId = originalTypeIdToAddedId[record.typeId] - ?: return@forEach - val addedId = record.copy( - id = 0, - typeId = newTypeId, - ).let { recordRepo.add(it) } - originalRecordIdToAddedId[originalId] = addedId - } - params.data.categories.values.forEach { - val originalId = it.id - val addedId = it.copy( - id = 0, - ).let { categoryRepo.add(it) } - originalCategoryIdToAddedId[originalId] = addedId - } - params.data.typeToCategory.forEach { typeToCategory -> - val newTypeId = originalTypeIdToAddedId[typeToCategory.recordTypeId] - ?: return@forEach - val newCategoryId = originalCategoryIdToAddedId[typeToCategory.categoryId] - ?: return@forEach - typeToCategory.copy( - recordTypeId = newTypeId, - categoryId = newCategoryId, - ).let { recordTypeCategoryRepo.add(it) } - } - params.data.tags.values.forEach { tag -> - val originalId = tag.id - val newColorSource = originalTypeIdToAddedId[tag.iconColorSource].orZero() - val addedId = tag.copy( - id = 0, - iconColorSource = newColorSource, - ).let { recordTagRepo.add(it) } - originalTagIdToAddedId[originalId] = addedId - } - params.data.recordToTag.forEach { recordToTag -> - val newRecordId = originalRecordIdToAddedId[recordToTag.recordId] - ?: return@forEach - val newTagId = originalTagIdToAddedId[recordToTag.recordTagId] - ?: return@forEach - recordToTag.copy( - recordId = newRecordId, - recordTagId = newTagId, - ).let { recordToRecordTagRepo.add(it) } - } - params.data.typeToTag.forEach { typeToTag -> - val newTypeId = originalTypeIdToAddedId[typeToTag.recordTypeId] - ?: return@forEach - val newTagId = originalTagIdToAddedId[typeToTag.tagId] - ?: return@forEach - typeToTag.copy( - recordTypeId = newTypeId, - tagId = newTagId, - ).let { recordTypeToTagRepo.add(it) } - } - params.data.typeToDefaultTag.forEach { typeToDefaultTag -> - val newTypeId = originalTypeIdToAddedId[typeToDefaultTag.recordTypeId] - ?: return@forEach - val newTagId = originalTagIdToAddedId[typeToDefaultTag.tagId] - ?: return@forEach - typeToDefaultTag.copy( - recordTypeId = newTypeId, - tagId = newTagId, - ).let { recordTypeToDefaultTagRepo.add(it) } - } - params.data.activityFilters.values.forEach { activityFilter -> - val newTypeIds = activityFilter.selectedIds - .mapNotNull { originalTypeIdToAddedId[it] } - activityFilter.copy( - id = 0, - selectedIds = newTypeIds, - ).let { activityFilterRepo.add(it) } - } - params.data.favouriteComments.values.forEach { favComment -> - favComment.copy( - id = 0, - ).let { favouriteCommentRepo.add(it) } - } - params.data.favouriteColors.values.forEach { favColor -> - favColor.copy( - id = 0, - ).let { favouriteColorRepo.add(it) } - } - params.data.favouriteIcon.values.forEach { favIcon -> - favIcon.copy( - id = 0, - ).let { favouriteIconRepo.add(it) } - } - params.data.goals.values.forEach { goal -> - val newId = when (val idData = goal.idData) { - is RecordTypeGoal.IdData.Type -> originalTypeIdToAddedId[idData.value] - ?.let(RecordTypeGoal.IdData::Type) - is RecordTypeGoal.IdData.Category -> originalCategoryIdToAddedId[idData.value] - ?.let(RecordTypeGoal.IdData::Category) - } ?: return@forEach - goal.copy( - id = 0, - idData = newId, - ).let { recordTypeGoalRepo.add(it) } - } - params.data.rules.values.forEach { rule -> - val newStartingTypeIds = rule.conditionStartingTypeIds - .mapNotNull { originalTypeIdToAddedId[it] }.toSet() - val newCurrentTypeIds = rule.conditionCurrentTypeIds - .mapNotNull { originalTypeIdToAddedId[it] }.toSet() - val newAssignTagIds = rule.actionAssignTagIds - .mapNotNull { originalTagIdToAddedId[it] }.toSet() - rule.copy( - id = 0, - actionAssignTagIds = newAssignTagIds, - conditionStartingTypeIds = newStartingTypeIds, - conditionCurrentTypeIds = newCurrentTypeIds, - ).let { complexRuleRepo.add(it) } - } - return@withContext ResultCode.Success( - resourceRepo.getString(R.string.message_import_complete), - ) - } - - override suspend fun readBackupFile( - uriString: String, - ): Pair = withContext(Dispatchers.IO) { - // Result data - val types: MutableList = mutableListOf() - val records: MutableList = mutableListOf() - val categories: MutableList = mutableListOf() - val typeToCategory: MutableList = mutableListOf() - val tags: MutableList = mutableListOf() - val recordToTag: MutableList = mutableListOf() - val typeToTag: MutableList = mutableListOf() - val typeToDefaultTag: MutableList = mutableListOf() - val activityFilters: MutableList = mutableListOf() - val favouriteComments: MutableList = mutableListOf() - val favouriteColors: MutableList = mutableListOf() - val favouriteIcon: MutableList = mutableListOf() - val goals: MutableList = mutableListOf() - val rules: MutableList = mutableListOf() - val settings: MutableList> = mutableListOf() - - val result = readBackup( - uriString = uriString, - successCodeMessage = null, - errorCodeMessage = R.string.settings_file_open_error, - clearData = false, - migrateTags = { - tags += migrateTags( - types = types, - data = it, - ) - }, - dataHandler = DataHandler( - types = types::add, - records = records::add, - categories = categories::add, - typeToCategory = typeToCategory::add, - tags = tags::add, - recordToTag = recordToTag::add, - typeToTag = typeToTag::add, - typeToDefaultTag = typeToDefaultTag::add, - activityFilters = activityFilters::add, - favouriteComments = favouriteComments::add, - favouriteColors = favouriteColors::add, - favouriteIcon = favouriteIcon::add, - goals = goals::add, - rules = rules::add, - settings = settings::add, - ), - ) - - val recordToTagIds = recordToTag.groupBy { it.recordId } - val processedRecords = records.map { - val thisTags = recordToTagIds[it.id].orEmpty() - it.copy(tagIds = thisTags.map(RecordToRecordTag::recordTagId)) - } - - result to PartialBackupRestoreData( - types = types.associateBy { it.id }, - records = processedRecords.associateBy { it.id }, - categories = categories.associateBy { it.id }, - typeToCategory = typeToCategory, - tags = tags.associateBy { it.id }, - recordToTag = recordToTag, - typeToTag = typeToTag, - typeToDefaultTag = typeToDefaultTag, - activityFilters = activityFilters.associateBy { it.id }, - favouriteComments = favouriteComments.associateBy { it.id }, - favouriteColors = favouriteColors.associateBy { it.id }, - favouriteIcon = favouriteIcon.associateBy { it.id }, - goals = goals.associateBy { it.id }, - rules = rules.associateBy { it.id }, - ) - } - - private suspend fun readBackup( + suspend fun readBackup( uriString: String, @StringRes successCodeMessage: Int?, @StringRes errorCodeMessage: Int, @@ -993,7 +781,7 @@ class BackupRepoImpl @Inject constructor( ) } - private fun migrateTags( + fun migrateTags( types: List, data: List>, ): List { @@ -1025,7 +813,7 @@ class BackupRepoImpl @Inject constructor( private fun String.restoreNewline() = replace("␤", "\n") - private data class DataHandler( + data class DataHandler( val types: suspend (RecordType) -> Unit, val records: suspend (Record) -> Unit, val categories: suspend (Category) -> Unit, diff --git a/domain/src/main/java/com/example/util/simpletimetracker/domain/interactor/BackupInteractor.kt b/domain/src/main/java/com/example/util/simpletimetracker/domain/interactor/BackupInteractor.kt index 203706250..85ee15a74 100644 --- a/domain/src/main/java/com/example/util/simpletimetracker/domain/interactor/BackupInteractor.kt +++ b/domain/src/main/java/com/example/util/simpletimetracker/domain/interactor/BackupInteractor.kt @@ -2,12 +2,14 @@ package com.example.util.simpletimetracker.domain.interactor import com.example.util.simpletimetracker.domain.model.BackupOptionsData import com.example.util.simpletimetracker.domain.model.PartialBackupRestoreData +import com.example.util.simpletimetracker.domain.resolver.BackupPartialRepo import com.example.util.simpletimetracker.domain.resolver.BackupRepo import com.example.util.simpletimetracker.domain.resolver.ResultCode import javax.inject.Inject class BackupInteractor @Inject constructor( private val backupRepo: BackupRepo, + private val backupPartialRepo: BackupPartialRepo, private val externalViewsInteractor: UpdateExternalViewsInteractor, ) { @@ -30,7 +32,7 @@ class BackupInteractor @Inject constructor( suspend fun partialRestoreBackupFile( params: BackupOptionsData.Custom, ): ResultCode { - val resultCode = backupRepo.partialRestoreBackupFile(params) + val resultCode = backupPartialRepo.partialRestoreBackupFile(params) doAfterRestore() return resultCode } @@ -38,7 +40,7 @@ class BackupInteractor @Inject constructor( suspend fun readBackupFileContent( uriString: String, ): Pair { - return backupRepo.readBackupFile(uriString) + return backupPartialRepo.readBackupFile(uriString) } suspend fun doAfterRestore() { diff --git a/domain/src/main/java/com/example/util/simpletimetracker/domain/model/PartialBackupRestoreData.kt b/domain/src/main/java/com/example/util/simpletimetracker/domain/model/PartialBackupRestoreData.kt index ffc8d3f60..424216881 100644 --- a/domain/src/main/java/com/example/util/simpletimetracker/domain/model/PartialBackupRestoreData.kt +++ b/domain/src/main/java/com/example/util/simpletimetracker/domain/model/PartialBackupRestoreData.kt @@ -2,18 +2,32 @@ package com.example.util.simpletimetracker.domain.model // TODO switch to LongObjectMap from androidx.collections data class PartialBackupRestoreData( - val types: Map, - val records: Map, - val categories: Map, - val typeToCategory: List, - val tags: Map, - val recordToTag: List, - val typeToTag: List, - val typeToDefaultTag: List, - val activityFilters: Map, - val favouriteComments: Map, - val favouriteColors: Map, - val favouriteIcon: Map, - val goals: Map, - val rules: Map, -) \ No newline at end of file + val types: Map>, + val records: Map>, + val categories: Map>, + val typeToCategory: List>, + val tags: Map>, + val recordToTag: List>, + val typeToTag: List>, + val typeToDefaultTag: List>, + val activityFilters: Map>, + val favouriteComments: Map>, + val favouriteColors: Map>, + val favouriteIcon: Map>, + val goals: Map>, + val rules: Map>, +) { + + data class Holder( + val exist: Boolean, + val data: T, + ) +} + +fun Collection>.getNotExistingValues(): List { + return this.mapNotNull { if (!it.exist) it.data else null } +} + +fun Collection>.getExistingValues(): List { + return this.mapNotNull { if (it.exist) it.data else null } +} \ No newline at end of file diff --git a/domain/src/main/java/com/example/util/simpletimetracker/domain/resolver/BackupPartialRepo.kt b/domain/src/main/java/com/example/util/simpletimetracker/domain/resolver/BackupPartialRepo.kt new file mode 100644 index 000000000..0c51b1f52 --- /dev/null +++ b/domain/src/main/java/com/example/util/simpletimetracker/domain/resolver/BackupPartialRepo.kt @@ -0,0 +1,15 @@ +package com.example.util.simpletimetracker.domain.resolver + +import com.example.util.simpletimetracker.domain.model.BackupOptionsData +import com.example.util.simpletimetracker.domain.model.PartialBackupRestoreData + +interface BackupPartialRepo { + + suspend fun partialRestoreBackupFile( + params: BackupOptionsData.Custom, + ): ResultCode + + suspend fun readBackupFile( + uriString: String, + ): Pair +} \ No newline at end of file diff --git a/domain/src/main/java/com/example/util/simpletimetracker/domain/resolver/BackupRepo.kt b/domain/src/main/java/com/example/util/simpletimetracker/domain/resolver/BackupRepo.kt index b7baa3501..65f81e961 100644 --- a/domain/src/main/java/com/example/util/simpletimetracker/domain/resolver/BackupRepo.kt +++ b/domain/src/main/java/com/example/util/simpletimetracker/domain/resolver/BackupRepo.kt @@ -1,7 +1,6 @@ package com.example.util.simpletimetracker.domain.resolver import com.example.util.simpletimetracker.domain.model.BackupOptionsData -import com.example.util.simpletimetracker.domain.model.PartialBackupRestoreData interface BackupRepo { @@ -14,12 +13,4 @@ interface BackupRepo { uriString: String, params: BackupOptionsData.Restore, ): ResultCode - - suspend fun partialRestoreBackupFile( - params: BackupOptionsData.Custom, - ): ResultCode - - suspend fun readBackupFile( - uriString: String, - ): Pair } \ No newline at end of file diff --git a/features/feature_settings/src/main/java/com/example/util/simpletimetracker/feature_settings/partialRestore/interactor/PartialRestoreViewDataInteractor.kt b/features/feature_settings/src/main/java/com/example/util/simpletimetracker/feature_settings/partialRestore/interactor/PartialRestoreViewDataInteractor.kt index 1f17fc370..cab3b84be 100644 --- a/features/feature_settings/src/main/java/com/example/util/simpletimetracker/feature_settings/partialRestore/interactor/PartialRestoreViewDataInteractor.kt +++ b/features/feature_settings/src/main/java/com/example/util/simpletimetracker/feature_settings/partialRestore/interactor/PartialRestoreViewDataInteractor.kt @@ -1,10 +1,13 @@ package com.example.util.simpletimetracker.feature_settings.partialRestore.interactor import com.example.util.simpletimetracker.core.mapper.ColorMapper +import com.example.util.simpletimetracker.core.repo.ResourceRepo import com.example.util.simpletimetracker.domain.interactor.PrefsInteractor import com.example.util.simpletimetracker.domain.model.PartialBackupRestoreData import com.example.util.simpletimetracker.feature_base_adapter.ViewHolderType +import com.example.util.simpletimetracker.feature_base_adapter.hint.HintViewData import com.example.util.simpletimetracker.feature_base_adapter.recordFilter.FilterViewData +import com.example.util.simpletimetracker.feature_settings.R import com.example.util.simpletimetracker.feature_settings.partialRestore.mapper.PartialRestoreViewDataMapper import com.example.util.simpletimetracker.feature_settings.partialRestore.model.PartialRestoreFilterType import com.example.util.simpletimetracker.feature_settings.partialRestore.utils.getIds @@ -14,13 +17,14 @@ class PartialRestoreViewDataInteractor @Inject constructor( private val prefsInteractor: PrefsInteractor, private val mapper: PartialRestoreViewDataMapper, private val colorMapper: ColorMapper, + private val resourceRepo: ResourceRepo, ) { fun getInitialFilters( data: PartialBackupRestoreData, ): Map> { return availableFilters.filter { - data.getIds(it).isNotEmpty() + data.getIds(it, existing = false).isNotEmpty() }.associateWith { emptySet() } @@ -33,7 +37,7 @@ class PartialRestoreViewDataInteractor @Inject constructor( val isDarkTheme = prefsInteractor.getDarkMode() return filters.toList().mapIndexed { index, (type, ids) -> - val allIds = data.getIds(type) + val allIds = data.getIds(type, existing = false) val selectedIds = allIds.filter { it !in ids } val selected = selectedIds.isNotEmpty() val name = mapper.mapFilterName( @@ -53,6 +57,10 @@ class PartialRestoreViewDataInteractor @Inject constructor( removeBtnVisible = selected, selected = selected, ) + }.ifEmpty { + HintViewData( + resourceRepo.getString(R.string.no_data), + ).let(::listOf) } } diff --git a/features/feature_settings/src/main/java/com/example/util/simpletimetracker/feature_settings/partialRestore/utils/PartialRestoreExtensions.kt b/features/feature_settings/src/main/java/com/example/util/simpletimetracker/feature_settings/partialRestore/utils/PartialRestoreExtensions.kt index 9990d4acc..f5d5ee543 100644 --- a/features/feature_settings/src/main/java/com/example/util/simpletimetracker/feature_settings/partialRestore/utils/PartialRestoreExtensions.kt +++ b/features/feature_settings/src/main/java/com/example/util/simpletimetracker/feature_settings/partialRestore/utils/PartialRestoreExtensions.kt @@ -5,6 +5,7 @@ import com.example.util.simpletimetracker.feature_settings.partialRestore.model. fun PartialBackupRestoreData.getIds( filter: PartialRestoreFilterType, + existing: Boolean, ): Set { return when (filter) { is PartialRestoreFilterType.Activities -> types @@ -16,5 +17,7 @@ fun PartialBackupRestoreData.getIds( is PartialRestoreFilterType.FavouriteColors -> favouriteColors is PartialRestoreFilterType.FavouriteIcons -> favouriteIcon is PartialRestoreFilterType.ComplexRules -> rules + }.filter { + if (existing) it.value.exist else !it.value.exist }.keys -} \ No newline at end of file +} diff --git a/features/feature_settings/src/main/java/com/example/util/simpletimetracker/feature_settings/partialRestore/view/PartialRestoreFragment.kt b/features/feature_settings/src/main/java/com/example/util/simpletimetracker/feature_settings/partialRestore/view/PartialRestoreFragment.kt index 82debc284..91a90a8fe 100644 --- a/features/feature_settings/src/main/java/com/example/util/simpletimetracker/feature_settings/partialRestore/view/PartialRestoreFragment.kt +++ b/features/feature_settings/src/main/java/com/example/util/simpletimetracker/feature_settings/partialRestore/view/PartialRestoreFragment.kt @@ -4,8 +4,10 @@ import android.view.LayoutInflater import android.view.ViewGroup import androidx.fragment.app.viewModels import com.example.util.simpletimetracker.core.base.BaseBottomSheetFragment +import com.example.util.simpletimetracker.core.dialog.StandardDialogListener import com.example.util.simpletimetracker.core.extension.setSkipCollapsed import com.example.util.simpletimetracker.feature_base_adapter.BaseRecyclerAdapter +import com.example.util.simpletimetracker.feature_base_adapter.hint.createHintAdapterDelegate import com.example.util.simpletimetracker.feature_base_adapter.recordFilter.createFilterAdapterDelegate import com.example.util.simpletimetracker.feature_settings.partialRestore.model.PartialRestoreFilterType import com.example.util.simpletimetracker.feature_settings.partialRestore.viewModel.PartialRestoreViewModel @@ -21,7 +23,8 @@ import com.example.util.simpletimetracker.feature_settings.databinding.SettingsP @AndroidEntryPoint class PartialRestoreFragment : BaseBottomSheetFragment(), - PartialRestoreSelectionDialogListener { + PartialRestoreSelectionDialogListener, + StandardDialogListener { override val inflater: (LayoutInflater, ViewGroup?, Boolean) -> Binding = Binding::inflate @@ -30,6 +33,7 @@ class PartialRestoreFragment : private val filtersAdapter: BaseRecyclerAdapter by lazy { BaseRecyclerAdapter( + createHintAdapterDelegate(), createFilterAdapterDelegate( onClick = viewModel::onFilterClick, onRemoveClick = { @@ -70,4 +74,8 @@ class PartialRestoreFragment : ) { viewModel.onDataSelected(type, dataIds, tag) } + + override fun onPositiveClick(tag: String?, data: Any?) { + viewModel.onPositiveDialogClick(tag) + } } diff --git a/features/feature_settings/src/main/java/com/example/util/simpletimetracker/feature_settings/partialRestore/viewModel/PartialRestoreViewModel.kt b/features/feature_settings/src/main/java/com/example/util/simpletimetracker/feature_settings/partialRestore/viewModel/PartialRestoreViewModel.kt index ceee9bf09..e7cc6bf4d 100644 --- a/features/feature_settings/src/main/java/com/example/util/simpletimetracker/feature_settings/partialRestore/viewModel/PartialRestoreViewModel.kt +++ b/features/feature_settings/src/main/java/com/example/util/simpletimetracker/feature_settings/partialRestore/viewModel/PartialRestoreViewModel.kt @@ -6,11 +6,13 @@ import com.example.util.simpletimetracker.core.base.BaseViewModel import com.example.util.simpletimetracker.core.base.SingleLiveEvent import com.example.util.simpletimetracker.core.extension.lazySuspend import com.example.util.simpletimetracker.core.extension.set +import com.example.util.simpletimetracker.core.repo.ResourceRepo import com.example.util.simpletimetracker.domain.model.ActivityFilter import com.example.util.simpletimetracker.domain.model.PartialBackupRestoreData import com.example.util.simpletimetracker.domain.model.RecordTypeGoal import com.example.util.simpletimetracker.feature_base_adapter.ViewHolderType import com.example.util.simpletimetracker.feature_base_adapter.recordFilter.FilterViewData +import com.example.util.simpletimetracker.feature_settings.R import com.example.util.simpletimetracker.feature_settings.partialRestore.interactor.PartialRestoreViewDataInteractor import com.example.util.simpletimetracker.feature_settings.partialRestore.mapper.PartialRestoreViewDataMapper import com.example.util.simpletimetracker.feature_settings.partialRestore.model.PartialRestoreFilterType @@ -18,6 +20,7 @@ import com.example.util.simpletimetracker.feature_settings.partialRestore.utils. import com.example.util.simpletimetracker.feature_settings.partialRestoreSelection.model.PartialRestoreSelectionDialogParams import com.example.util.simpletimetracker.feature_settings.viewModel.delegate.SettingsFileWorkDelegate import com.example.util.simpletimetracker.navigation.Router +import com.example.util.simpletimetracker.navigation.params.screen.StandardDialogParams import dagger.hilt.android.lifecycle.HiltViewModel import kotlinx.coroutines.Dispatchers import kotlinx.coroutines.launch @@ -27,6 +30,7 @@ import javax.inject.Inject @HiltViewModel class PartialRestoreViewModel @Inject constructor( private val router: Router, + private val resourceRepo: ResourceRepo, private val partialRestoreViewDataInteractor: PartialRestoreViewDataInteractor, private val settingsFileWorkDelegate: SettingsFileWorkDelegate, private val partialRestoreViewDataMapper: PartialRestoreViewDataMapper, @@ -60,7 +64,9 @@ class PartialRestoreViewModel @Inject constructor( fun onFilterRemoveClick(data: FilterViewData) { val itemType = data.type as? PartialRestoreFilterType ?: return val currentData = settingsFileWorkDelegate.partialBackupRestoreData ?: return - filters = filters.toMutableMap().apply { put(itemType, currentData.getIds(itemType)) } + filters = filters.toMutableMap().apply { + put(itemType, currentData.getIds(itemType, existing = false)) + } updateFilters() } @@ -75,6 +81,23 @@ class PartialRestoreViewModel @Inject constructor( } fun onRestoreClick() { + router.navigate( + StandardDialogParams( + tag = PARTIAL_RESTORE_ALERT_DIALOG_TAG, + message = resourceRepo.getString(R.string.archive_deletion_alert), + btnPositive = resourceRepo.getString(R.string.ok), + btnNegative = resourceRepo.getString(R.string.cancel), + ), + ) + } + + fun onPositiveDialogClick(tag: String?) { + when (tag) { + PARTIAL_RESTORE_ALERT_DIALOG_TAG -> onRestore() + } + } + + private fun onRestore() { val data = settingsFileWorkDelegate.partialBackupRestoreDataSelectable ?: return val filteredData = partialRestoreViewDataMapper.mapFilteredData(filters, data) settingsFileWorkDelegate.onPartialRestoreConfirmed(filteredData) @@ -90,73 +113,82 @@ class PartialRestoreViewModel @Inject constructor( val categoriesIds = categories.keys // Check tags - val tags = data.tags.mapValues { - if (it.value.iconColorSource != 0L && it.value.iconColorSource !in typesIds) { - it.value.copy( - icon = types[it.value.iconColorSource]?.icon ?: it.value.icon, - color = types[it.value.iconColorSource]?.color ?: it.value.color, + val tags = data.tags.mapValues { (_, item) -> + if (item.data.iconColorSource != 0L && + item.data.iconColorSource !in typesIds + ) { + val newData = item.data.copy( + icon = types[item.data.iconColorSource]?.data?.icon + ?: item.data.icon, + color = types[item.data.iconColorSource]?.data?.color + ?: item.data.color, ) + item.copy(data = newData) } else { - it.value + item } } val typeToTag = data.typeToTag.filter { - it.recordTypeId in typesIds && it.tagId in tags + it.data.recordTypeId in typesIds && it.data.tagId in tags } val typeToDefaultTag = data.typeToDefaultTag.filter { - it.recordTypeId in typesIds && it.tagId in tags + it.data.recordTypeId in typesIds && it.data.tagId in tags } // Check records val records = data.records.filter { - it.value.typeId in typesIds - }.mapValues { record -> - record.value.copy(tagIds = record.value.tagIds.filter { it in tags }) + it.value.data.typeId in typesIds + }.mapValues { (_, item) -> + val newData = item.data.copy( + tagIds = item.data.tagIds.filter { it in tags }, + ) + item.copy(data = newData) } val recordsIds = records.keys // Check record to tag relation val recordToTag = data.recordToTag.filter { - it.recordId in recordsIds && it.recordTagId in tags + it.data.recordId in recordsIds && it.data.recordTagId in tags } // Check type to category relation val typeToCategory = data.typeToCategory.filter { - it.recordTypeId in typesIds && it.categoryId in categoriesIds + it.data.recordTypeId in typesIds && it.data.categoryId in categoriesIds } // Check filters - val activityFilters = data.activityFilters.mapValues { filter -> - val newIds = filter.value.selectedIds.filter { - when (filter.value.type) { + val activityFilters = data.activityFilters.mapValues { (_, item) -> + val newIds = item.data.selectedIds.filter { + when (item.data.type) { is ActivityFilter.Type.Activity -> it in typesIds is ActivityFilter.Type.Category -> it in categoriesIds } } - filter.value.copy(selectedIds = newIds) + val newData = item.data.copy(selectedIds = newIds) + item.copy(data = newData) } // Check goals val goals = data.goals.filter { - when (val id = it.value.idData) { + when (val id = it.value.data.idData) { is RecordTypeGoal.IdData.Type -> id.value in typesIds is RecordTypeGoal.IdData.Category -> id.value in categoriesIds } } // Check rules - val rules = data.rules.mapNotNull { rule -> - val new = rule.value.copy( - actionAssignTagIds = rule.value.actionAssignTagIds + val rules = data.rules.mapNotNull { (id, item) -> + val newData = item.data.copy( + actionAssignTagIds = item.data.actionAssignTagIds .filter { it in tags }.toSet(), - conditionStartingTypeIds = rule.value.conditionStartingTypeIds + conditionStartingTypeIds = item.data.conditionStartingTypeIds .filter { it in typesIds }.toSet(), - conditionCurrentTypeIds = rule.value.conditionCurrentTypeIds + conditionCurrentTypeIds = item.data.conditionCurrentTypeIds .filter { it in typesIds }.toSet(), ).takeIf { it.hasActions && it.hasConditions } ?: return@mapNotNull null - rule.key to new + id to item.copy(data = newData) }.toMap() return@withContext PartialBackupRestoreData( @@ -199,7 +231,9 @@ class PartialRestoreViewModel @Inject constructor( ) settingsFileWorkDelegate.partialBackupRestoreDataSelectable = newSelectableData filters = filters.mapValues { (filter, ids) -> - ids.filter { it in newSelectableData.getIds(filter) }.toSet() + ids.filter { + it in newSelectableData.getIds(filter, existing = false) + }.toSet() } } @@ -229,5 +263,6 @@ class PartialRestoreViewModel @Inject constructor( companion object { private const val PARTIAL_RESTORE_SELECTION_TAG = "PARTIAL_RESTORE_SELECTION_TAG" + private const val PARTIAL_RESTORE_ALERT_DIALOG_TAG = "PARTIAL_RESTORE_ALERT_DIALOG_TAG" } } diff --git a/features/feature_settings/src/main/java/com/example/util/simpletimetracker/feature_settings/partialRestoreSelection/PartialRestoreSelectionViewDataInteractor.kt b/features/feature_settings/src/main/java/com/example/util/simpletimetracker/feature_settings/partialRestoreSelection/PartialRestoreSelectionViewDataInteractor.kt index 81330180c..84576ebc3 100644 --- a/features/feature_settings/src/main/java/com/example/util/simpletimetracker/feature_settings/partialRestoreSelection/PartialRestoreSelectionViewDataInteractor.kt +++ b/features/feature_settings/src/main/java/com/example/util/simpletimetracker/feature_settings/partialRestoreSelection/PartialRestoreSelectionViewDataInteractor.kt @@ -19,6 +19,7 @@ import com.example.util.simpletimetracker.domain.interactor.SortCardsInteractor import com.example.util.simpletimetracker.domain.model.CardOrder import com.example.util.simpletimetracker.domain.model.CardTagOrder import com.example.util.simpletimetracker.domain.model.PartialBackupRestoreData +import com.example.util.simpletimetracker.domain.model.getNotExistingValues import com.example.util.simpletimetracker.feature_base_adapter.ViewHolderType import com.example.util.simpletimetracker.feature_base_adapter.category.CategoryViewData import com.example.util.simpletimetracker.feature_base_adapter.color.ColorViewData @@ -58,7 +59,7 @@ class PartialRestoreSelectionViewDataInteractor @Inject constructor( return when (extra.type) { is PartialRestoreFilterType.Activities -> { - data.types.values.let { + data.types.values.getNotExistingValues().let { sortCardsInteractor.sort( cardOrder = CardOrder.NAME, manualOrderProvider = { emptyMap() }, @@ -76,7 +77,7 @@ class PartialRestoreSelectionViewDataInteractor @Inject constructor( } } PartialRestoreFilterType.Categories -> { - data.categories.values.let { + data.categories.values.getNotExistingValues().let { sortCardsInteractor.sort( cardOrder = CardOrder.NAME, manualOrderProvider = { emptyMap() }, @@ -91,15 +92,15 @@ class PartialRestoreSelectionViewDataInteractor @Inject constructor( } } PartialRestoreFilterType.Tags -> { - val types = data.types + val types = data.types.mapValues { it.value.data } val activityOrderProvider = { recordTagInteractor.getActivityOrderProvider( - tags = data.tags.values.toList(), + tags = data.tags.values.getNotExistingValues(), typesMap = types, - typesToTags = data.typeToTag, + typesToTags = data.typeToTag.map { it.data }, ) } - data.tags.values.let { + data.tags.values.getNotExistingValues().let { sortCardsInteractor.sortTags( cardTagOrder = CardTagOrder.ACTIVITY, manualOrderProvider = { emptyMap() }, @@ -121,9 +122,9 @@ class PartialRestoreSelectionViewDataInteractor @Inject constructor( } } PartialRestoreFilterType.Records -> { - val typesMap = data.types - val tags = data.tags.values.toList() - data.records.values.mapNotNull { + val typesMap = data.types.mapValues { it.value.data } + val tags = data.tags.values.map { it.data } + data.records.values.getNotExistingValues().mapNotNull { val viewData = recordViewDataMapper.mapFilteredRecord( record = it, recordTypes = typesMap, @@ -140,7 +141,7 @@ class PartialRestoreSelectionViewDataInteractor @Inject constructor( }.let(dateDividerViewDataMapper::addDateViewData) } PartialRestoreFilterType.ActivityFilters -> { - data.activityFilters.values.toList().let { + data.activityFilters.values.getNotExistingValues().let { activityFilterInteractor.sort(it) }.map { activityFilterViewDataMapper.mapFiltered( @@ -151,7 +152,7 @@ class PartialRestoreSelectionViewDataInteractor @Inject constructor( } } PartialRestoreFilterType.FavouriteComments -> { - data.favouriteComments.values.toList().let { + data.favouriteComments.values.getNotExistingValues().let { favouriteCommentInteractor.sort(it) }.map { val filtered = it.id in dataIdsFiltered @@ -170,7 +171,7 @@ class PartialRestoreSelectionViewDataInteractor @Inject constructor( } } PartialRestoreFilterType.FavouriteColors -> { - data.favouriteColors.values.toList().let { + data.favouriteColors.values.getNotExistingValues().let { favouriteColorInteractor.sort(it) }.map { val filtered = it.id in dataIdsFiltered @@ -184,11 +185,11 @@ class PartialRestoreSelectionViewDataInteractor @Inject constructor( } } PartialRestoreFilterType.FavouriteIcons -> { - val icons = data.favouriteIcon + val icons = data.favouriteIcon.values.getNotExistingValues() val iconImages = iconSelectionMapper.mapFavouriteIconImages( - icons.values.toList(), + icons, ).mapNotNull { icon -> - val favIcon = icons.values.firstOrNull { it.icon == icon.iconName } + val favIcon = icons.firstOrNull { it.icon == icon.iconName } ?: return@mapNotNull null iconSelectionMapper.mapImageViewData( iconName = icon.iconName, @@ -200,9 +201,9 @@ class PartialRestoreSelectionViewDataInteractor @Inject constructor( ) } val iconEmojis = iconSelectionMapper.mapFavouriteIconEmojis( - icons.values.toList(), + icons, ).mapNotNull { icon -> - val favIcon = icons.values.firstOrNull { it.icon == icon.emojiCode } + val favIcon = icons.firstOrNull { it.icon == icon.emojiCode } ?: return@mapNotNull null iconSelectionMapper.mapEmojiViewData( codes = icon.emojiCode, @@ -215,19 +216,19 @@ class PartialRestoreSelectionViewDataInteractor @Inject constructor( iconImages + iconEmojis } PartialRestoreFilterType.ComplexRules -> { - val typesMap = data.types + val typesMap = data.types.mapValues { it.value.data } val typesOrder = data.types.keys.toList() - val tagsMap = data.tags + val tagsMap = data.tags.mapValues { it.value.data } val tagsOrder = data.tags.keys.toList() - data.rules.values.map { + data.rules.values.getNotExistingValues().map { rule -> complexRulesViewDataMapper.mapRuleFiltered( - rule = it, + rule = rule, isDarkTheme = isDarkTheme, typesMap = typesMap, tagsMap = tagsMap, typesOrder = typesOrder, tagsOrder = tagsOrder, - isFiltered = it.id in dataIdsFiltered, + isFiltered = rule.id in dataIdsFiltered, disableButtonVisible = false, ) } diff --git a/features/feature_settings/src/main/java/com/example/util/simpletimetracker/feature_settings/partialRestoreSelection/PartialRestoreSelectionViewModel.kt b/features/feature_settings/src/main/java/com/example/util/simpletimetracker/feature_settings/partialRestoreSelection/PartialRestoreSelectionViewModel.kt index 1eeee2323..ed829ec5a 100644 --- a/features/feature_settings/src/main/java/com/example/util/simpletimetracker/feature_settings/partialRestoreSelection/PartialRestoreSelectionViewModel.kt +++ b/features/feature_settings/src/main/java/com/example/util/simpletimetracker/feature_settings/partialRestoreSelection/PartialRestoreSelectionViewModel.kt @@ -65,15 +65,15 @@ class PartialRestoreSelectionViewModel @Inject constructor( fun onIconClick(item: IconSelectionViewData) { val iconId = getData()?.favouriteIcon?.values?.firstOrNull { - it.icon == item.iconName - }?.id ?: return + it.data.icon == item.iconName + }?.data?.id ?: return addOrRemoveId(iconId) } fun onEmojiClick(item: EmojiViewData) { val iconId = getData()?.favouriteIcon?.values?.firstOrNull { - it.icon == item.emojiCodes - }?.id ?: return + it.data.icon == item.emojiCodes + }?.data?.id ?: return addOrRemoveId(iconId) } @@ -93,7 +93,7 @@ class PartialRestoreSelectionViewModel @Inject constructor( } fun onHideAllClick() { - val allIds = getData()?.getIds(extra.type).orEmpty() + val allIds = getData()?.getIds(extra.type, existing = false).orEmpty() dataIdsFiltered.addAll(allIds) updateViewData() } diff --git a/features/feature_settings/src/main/res/layout/settings_partial_restore_fragment.xml b/features/feature_settings/src/main/res/layout/settings_partial_restore_fragment.xml index b3261d6b0..88a141806 100644 --- a/features/feature_settings/src/main/res/layout/settings_partial_restore_fragment.xml +++ b/features/feature_settings/src/main/res/layout/settings_partial_restore_fragment.xml @@ -30,7 +30,7 @@ style="@style/SettingsSecondaryText" android:layout_width="match_parent" android:layout_height="wrap_content" - android:layout_marginHorizontal="4dp" + android:layout_marginHorizontal="8dp" android:gravity="center" android:paddingBottom="0dp" android:text="@string/partial_restore_filter_hint" /> @@ -51,6 +51,16 @@ style="@style/EditScreenDivider" android:layout_width="match_parent" /> + +