36fe52e5ad
* 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)
354 lines
13 KiB
Kotlin
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
|
|
)
|
|
}
|
|
} |