anytype-kotlin-wild/app/src/androidTest/java/com/anytypeio/anytype/features/relations/EditRelationTagValueTest.kt
Evgenii Kozlov 36fe52e5ad
DROID-607 Relations & types and options | Refactoring | Everything-is-an-object refactoring (#2720)
* DROID-439 App | Relations refactoring, use SearchObjects for object types (#2592)
* DROID-446 Objects | Enhancement | Store relation links and process its updates (#2597)
* DROID-456 Tech | Remove deprecated API for creating object types (#2601)
* DROID-455 Object types | Enhancement Get object type list from ObjectSearch on Global search screen (#2602)
* DROID-459 Sets | Refactoring | Use relations links instead of relations for building search params to get data view data (#2603)
* DROID-458 Objects | Refactoring | Integrate new api for creating objets (#2604)
* DROID-464 Relations | Refactoring | New API for creating a relation from scratch - for data view or for an object (#2605)
* DROID-467 Relations | Refactoring | New API for creating options - for tags and statuses (#2607)
* DROID-470 Relations | Refactoring | New API for removing any relation from an object or from a data view (#2608)
* DROID-460 Tech | Object relations list (#2610)
* DROID-486 Sets | Refactoring | Displaying and controlling visibility of data view relations (#2611)
* DROID-500 Relations | Refactoring | Extend API of ObjectStore to be able to get a relation by its id (#2619)
* DROID-459 Sets | Fix | Get relations from store by id and not by key (#2618)
* DROID-489 Tech | Relations as object, add relationKey to relation connected screens (#2622)
* DROID-490 Sets | Refactoring | Displaying relation values for current object in bottom sheet cells (#2629)
* DROID-505 Relations | Refactoring | New interface for relations store (#2633)
* DROID-509 Relations | Refactoring | Bind new relations store with subscription container (#2635)
* DROID-507 Tech | MW , migration + relation links (#2636)
* DROID-507 Tech | MW , migration + relation links, fixes (#2637)
* DROID-517 Sets | Fix | Provide correct keys for data view search-and-subscribe query (#2641)
* DROID-409 Relations | Refactoring | Use relation key instead of id when creating new relation (#2642)
* DROID-521 Sets & Objects | Refactoring | Add relation to a data view or to an object from existing relations (#2644)
* DROID-522 Relations | Refactoring | Add objects to relations with object format (#2645)
* DROID-523 Object types | Refactoring | Implement global store for object types (#2646)
* DROID-527 Object types | Refactoring | Integrate global store for object types (#2647)
* DROID-531 Relations | Refactoring | Parse tag and status relations values (#2649)
* DROID-535 Tech | Integrate new MW lib with migration fixes (#2653)
* DROID-535 Tech | MW integration fixes (#2660)
* DROID-559 Relations | Refactoring | Parse tag and status values in editor (#2662)
* DROID-560 Relations | Refactoring | Integrate new lib with fixes (#2663)
* DROID-561 Relations | Refactoring | Parsing tag and status values in dv (#2665)
* DROID-562 Dashboard | Refactoring | Use store of object types as object type provider for favorites tab on dashboard (#2667)
* DROID-567 Relations | Refactoring | Suggest available options to populate a relation (#2671)
* DROID-604 Relations | Refactoring | Use details from Object.CreateRelation.Response to populate relation store (#2705)
* DROID-603 Relations | Refactoring | Creating relation options + Deleting relation from object (#2706)
* DROID-619 Relations | Refactoring | Migrate data view sorts and filters to the new relation-as-object paradigm (#2711)
* DROID-622 Relations | Tech | Update MW to 0.24.0-rc1 (#2714)
* DROID-598 Sets | Refactoring | Provide relation format for date filters (#2715)
* DROID-625 Protocol | Enhancement | Integrate v0.24.0-rc2 (#2718)
2022-11-24 18:11:19 +03:00

354 lines
13 KiB
Kotlin

package com.anytypeio.anytype.features.relations
import android.os.Bundle
import androidx.core.os.bundleOf
import androidx.fragment.app.testing.FragmentScenario
import androidx.fragment.app.testing.launchFragmentInContainer
import androidx.test.ext.junit.runners.AndroidJUnit4
import androidx.test.filters.LargeTest
import com.anytypeio.anytype.R
import com.anytypeio.anytype.analytics.base.Analytics
import com.anytypeio.anytype.core_models.Block
import com.anytypeio.anytype.core_models.Id
import com.anytypeio.anytype.core_models.Payload
import com.anytypeio.anytype.core_models.Relation
import com.anytypeio.anytype.core_models.ThemeColor
import com.anytypeio.anytype.core_ui.extensions.dark
import com.anytypeio.anytype.domain.`object`.UpdateDetail
import com.anytypeio.anytype.domain.block.repo.BlockRepository
import com.anytypeio.anytype.domain.config.Gateway
import com.anytypeio.anytype.domain.misc.UrlBuilder
import com.anytypeio.anytype.domain.objects.DefaultObjectStore
import com.anytypeio.anytype.domain.objects.DefaultStoreOfObjectTypes
import com.anytypeio.anytype.domain.objects.DefaultStoreOfRelations
import com.anytypeio.anytype.domain.objects.ObjectStore
import com.anytypeio.anytype.domain.objects.StoreOfObjectTypes
import com.anytypeio.anytype.domain.objects.StoreOfRelations
import com.anytypeio.anytype.domain.relations.AddFileToObject
import com.anytypeio.anytype.presentation.relations.ObjectSetConfig
import com.anytypeio.anytype.presentation.relations.providers.DataViewObjectRelationProvider
import com.anytypeio.anytype.presentation.relations.providers.DataViewObjectValueProvider
import com.anytypeio.anytype.presentation.relations.providers.ObjectDetailProvider
import com.anytypeio.anytype.presentation.sets.ObjectSet
import com.anytypeio.anytype.presentation.sets.ObjectSetDatabase
import com.anytypeio.anytype.presentation.sets.RelationValueDVViewModel
import com.anytypeio.anytype.presentation.util.CopyFileToCacheDirectory
import com.anytypeio.anytype.presentation.util.Dispatcher
import com.anytypeio.anytype.test_utils.MockDataFactory
import com.anytypeio.anytype.test_utils.utils.checkHasText
import com.anytypeio.anytype.test_utils.utils.checkHasTextColor
import com.anytypeio.anytype.test_utils.utils.checkIsDisplayed
import com.anytypeio.anytype.test_utils.utils.checkIsNotDisplayed
import com.anytypeio.anytype.test_utils.utils.checkIsRecyclerSize
import com.anytypeio.anytype.test_utils.utils.matchView
import com.anytypeio.anytype.test_utils.utils.onItemView
import com.anytypeio.anytype.test_utils.utils.performClick
import com.anytypeio.anytype.test_utils.utils.rVMatcher
import com.anytypeio.anytype.test_utils.utils.resources
import com.anytypeio.anytype.ui.relations.RelationValueBaseFragment
import com.anytypeio.anytype.utils.CoroutinesTestRule
import com.bartoszlipinski.disableanimationsrule.DisableAnimationsRule
import kotlinx.coroutines.flow.MutableStateFlow
import org.junit.Before
import org.junit.Rule
import org.junit.Test
import org.junit.runner.RunWith
import org.mockito.Mock
import org.mockito.MockitoAnnotations
import org.mockito.kotlin.times
import org.mockito.kotlin.verifyBlocking
@RunWith(AndroidJUnit4::class)
@LargeTest
class EditRelationTagValueTest {
@Mock
lateinit var repo: BlockRepository
@Mock
lateinit var gateway: Gateway
@Mock
lateinit var dispatcher: Dispatcher<Payload>
@Mock
lateinit var analytics: Analytics
@Mock
lateinit var copyFileToCacheDirectory: CopyFileToCacheDirectory
private lateinit var updateDetail: UpdateDetail
private lateinit var urlBuilder: UrlBuilder
private lateinit var addFileToObject: AddFileToObject
@get:Rule
val animationsRule = DisableAnimationsRule()
@get:Rule
val coroutineTestRule = CoroutinesTestRule()
private val ctx = MockDataFactory.randomUuid()
private val state = MutableStateFlow(ObjectSet.init())
private val store: ObjectStore = DefaultObjectStore()
private val storeOfRelations: StoreOfRelations = DefaultStoreOfRelations()
private val db = ObjectSetDatabase(store = store)
private val storeOfObjectTypes: StoreOfObjectTypes = DefaultStoreOfObjectTypes()
@Before
fun setup() {
MockitoAnnotations.openMocks(this)
updateDetail = UpdateDetail(repo)
urlBuilder = UrlBuilder(gateway)
addFileToObject = AddFileToObject(repo)
TestRelationValueDVFragment.testVmFactory = RelationValueDVViewModel.Factory(
relations = DataViewObjectRelationProvider(
objectSetState = state,
storeOfRelations = storeOfRelations
),
values = DataViewObjectValueProvider(db = db),
details = object : ObjectDetailProvider {
override fun provide(): Map<Id, Block.Fields> = state.value.details
},
urlBuilder = urlBuilder,
copyFileToCache = copyFileToCacheDirectory,
addFileToObject = addFileToObject,
dispatcher = dispatcher,
analytics = analytics,
setObjectDetails = updateDetail,
storeOfObjectTypes = storeOfObjectTypes
)
}
@Test
fun shouldRenderTwoTagsInReadThenInEditMode() {
// SETUP
val option1Color = ThemeColor.values().random()
val option2Color = ThemeColor.values().random()
val option1 = Relation.Option(
id = MockDataFactory.randomUuid(),
text = "Architect",
color = option1Color.code
)
val option2 = Relation.Option(
id = MockDataFactory.randomUuid(),
text = "Manager",
color = option2Color.code
)
val option3 = Relation.Option(
id = MockDataFactory.randomUuid(),
text = "Developer",
color = ""
)
val relationKey = MockDataFactory.randomUuid()
val target = MockDataFactory.randomUuid()
val record: Map<String, Any?> = mapOf(
ObjectSetConfig.ID_KEY to target,
relationKey to listOf(option2.id, option3.id)
)
val viewer = Block.Content.DataView.Viewer(
id = MockDataFactory.randomUuid(),
name = MockDataFactory.randomString(),
filters = emptyList(),
sorts = emptyList(),
viewerRelations = emptyList(),
type = Block.Content.DataView.Viewer.Type.GRID
)
state.value = ObjectSet(
blocks = listOf(
Block(
id = MockDataFactory.randomUuid(),
children = emptyList(),
fields = Block.Fields.empty(),
content = Block.Content.DataView(
relations = listOf(
Relation(
key = relationKey,
defaultValue = null,
isHidden = false,
isReadOnly = false,
isMulti = true,
name = "Roles",
source = Relation.Source.values().random(),
format = Relation.Format.TAG,
selections = listOf(option1, option2, option3)
)
),
viewers = listOf(viewer),
sources = listOf(
MockDataFactory.randomUuid()
)
)
)
),
// viewerDb = mapOf(
// viewer.id to ObjectSet.ViewerData(
// records = listOf(record),
// total = 1
// )
// )
)
// TESTING
launchFragment(
bundleOf(
RelationValueBaseFragment.CTX_KEY to ctx,
RelationValueBaseFragment.RELATION_KEY to relationKey,
RelationValueBaseFragment.TARGET_KEY to target,
RelationValueBaseFragment.IS_LOCKED_KEY to false,
)
)
val editOrDoneBtn = R.id.btnEditOrDone.matchView()
editOrDoneBtn.checkHasText(R.string.edit)
with(R.id.recycler.rVMatcher()) {
onItemView(0, R.id.tvTagName).checkHasText(option2.text)
onItemView(0, R.id.tvTagName).checkHasTextColor(resources.dark(option2Color))
onItemView(0, R.id.btnRemoveTag).checkIsNotDisplayed()
onItemView(0, R.id.btnDragAndDropTag).checkIsNotDisplayed()
onItemView(1, R.id.tvTagName).checkHasText(option3.text)
onItemView(1, R.id.btnRemoveTag).checkIsNotDisplayed()
onItemView(1, R.id.btnDragAndDropTag).checkIsNotDisplayed()
checkIsRecyclerSize(2)
}
editOrDoneBtn.performClick()
editOrDoneBtn.checkHasText(R.string.done)
with(R.id.recycler.rVMatcher()) {
onItemView(0, R.id.tvTagName).checkHasText(option2.text)
onItemView(0, R.id.tvTagName).checkHasTextColor(resources.dark(option2Color))
onItemView(0, R.id.btnRemoveTag).checkIsDisplayed()
onItemView(0, R.id.btnDragAndDropTag).checkIsDisplayed()
onItemView(1, R.id.tvTagName).checkHasText(option3.text)
onItemView(1, R.id.btnRemoveTag).checkIsDisplayed()
onItemView(1, R.id.btnDragAndDropTag).checkIsDisplayed()
checkIsRecyclerSize(2)
}
}
@Test
fun shouldRequestRemovingLastTagWhenRemoveButtonPressed() {
// SETUP
val option1Color = ThemeColor.values().random()
val option2Color = ThemeColor.values().random()
val option1 = Relation.Option(
id = MockDataFactory.randomUuid(),
text = "Architect",
color = option1Color.code
)
val option2 = Relation.Option(
id = MockDataFactory.randomUuid(),
text = "Manager",
color = option2Color.code
)
val option3 = Relation.Option(
id = MockDataFactory.randomUuid(),
text = "Developer",
color = ""
)
val relationKey = MockDataFactory.randomUuid()
val target = MockDataFactory.randomUuid()
val record: Map<String, Any?> = mapOf(
ObjectSetConfig.ID_KEY to target,
relationKey to listOf(option2.id, option3.id)
)
val viewer = Block.Content.DataView.Viewer(
id = MockDataFactory.randomUuid(),
name = MockDataFactory.randomString(),
filters = emptyList(),
sorts = emptyList(),
viewerRelations = emptyList(),
type = Block.Content.DataView.Viewer.Type.GRID
)
val dv = Block(
id = MockDataFactory.randomUuid(),
children = emptyList(),
fields = Block.Fields.empty(),
content = Block.Content.DataView(
relations = listOf(
Relation(
key = relationKey,
defaultValue = null,
isHidden = false,
isReadOnly = false,
isMulti = true,
name = "Roles",
source = Relation.Source.values().random(),
format = Relation.Format.TAG,
selections = listOf(option1, option2, option3)
)
),
viewers = listOf(viewer),
sources = listOf(
MockDataFactory.randomUuid()
)
)
)
state.value = ObjectSet(
blocks = listOf(dv),
// viewerDb = mapOf(
// viewer.id to ObjectSet.ViewerData(
// records = listOf(record),
// total = 1
// )
// )
)
// TESTING
launchFragment(
bundleOf(
RelationValueBaseFragment.CTX_KEY to ctx,
RelationValueBaseFragment.DATAVIEW_KEY to dv.id,
RelationValueBaseFragment.VIEWER_KEY to viewer.id,
RelationValueBaseFragment.RELATION_KEY to relationKey,
RelationValueBaseFragment.TARGET_KEY to target,
RelationValueBaseFragment.IS_LOCKED_KEY to false,
)
)
R.id.btnEditOrDone.performClick()
val rvMatcher = R.id.recycler.rVMatcher()
rvMatcher.onItemView(1, R.id.btnRemoveTag).performClick()
verifyBlocking(repo, times(1)) {
updateDetail(
ctx= target,
key = relationKey,
value = listOf(option2.id)
)
}
}
private fun launchFragment(args: Bundle): FragmentScenario<TestRelationValueDVFragment> {
return launchFragmentInContainer(
fragmentArgs = args,
themeResId = R.style.AppTheme
)
}
}