Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Offline mode: Post conflict resolution overlay #20607

Merged
merged 36 commits into from
Apr 12, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
36 commits
Select commit Hold shift + click to select a range
e5de678
Add wrapper around DateUtils for testability
zwarm Apr 7, 2024
71ed026
Initial versions of fragment and vm for PostResolutionOverlay
zwarm Apr 7, 2024
434d94d
Set up di modules for post resolution overlay
zwarm Apr 7, 2024
f31e6c7
Add wrapper around timestampFromIso8601Millis
zwarm Apr 7, 2024
a2bf427
Add base strings for post resolution overlay
zwarm Apr 7, 2024
f8339fe
WIP - Show the post resolution overlay instead of the dialog.
zwarm Apr 7, 2024
fda28f0
Change save to confirm and use string resource for both buttons
zwarm Apr 8, 2024
88786ae
Add a listener interface for post resolution
zwarm Apr 8, 2024
27dc8a3
Hooked in the initial version of the PostResolutionOverlayListener
zwarm Apr 8, 2024
c205244
Hook in the initial version of the PostResolutionOverlayListener
zwarm Apr 8, 2024
98cb540
WIP - Added self comments and todos
zwarm Apr 8, 2024
308bec0
Rename: PostConflictResolutionType to PostResolutionType
zwarm Apr 8, 2024
19e5e87
Reduce the callbacks to a single function and handle downstream
zwarm Apr 8, 2024
5220ce3
Hook the dismiss actions (cancel, dismiss, and close) and the prelimi…
zwarm Apr 8, 2024
68b3928
Make fun public so enabled can be accessed
zwarm Apr 10, 2024
6619c9e
Pass the value of isSyncPublishingEnabled flag over to the dialogHelper
zwarm Apr 10, 2024
8abc56f
Add string resources for autosave overlay
zwarm Apr 10, 2024
d943074
Rename the listener function
zwarm Apr 10, 2024
64a8ea3
Updates to reflect name changes and signature types
zwarm Apr 10, 2024
d41b26d
Adjsut for name changes and handle onPostResolutionConfirmed
zwarm Apr 10, 2024
08db006
Adjust for renaming and implement autosave sync logic
zwarm Apr 10, 2024
253306d
Adjust to name changes. Handle the confirm action clicks for post res…
zwarm Apr 10, 2024
b2d8032
Address detekt, lint, and checkstyle issue. Remove log lines
zwarm Apr 10, 2024
a4e744d
Refactor: move composables out of fragment
zwarm Apr 10, 2024
c16d387
Refactor: move UiState to its own file
zwarm Apr 10, 2024
e0bfc66
Add unit tests for PostResolutionOverlayViewModel and minor adjustmen…
zwarm Apr 10, 2024
a69336c
refactor: rename SyncPublishingFeatureConfig
pantstamp Apr 11, 2024
f7dd712
Refactor: Allow PostResolutionListener to be null for detach purposes
zwarm Apr 11, 2024
37597dc
Remove todo
zwarm Apr 11, 2024
6cd3ad4
Check for fragment before creating
zwarm Apr 11, 2024
83e8880
Annotate with JvmStatic
zwarm Apr 11, 2024
bb4997b
Do not dismiss the dialog on an super dismiss to support orientation …
zwarm Apr 11, 2024
ee422b4
Add post resolution tracking events
zwarm Apr 11, 2024
e71f620
Add tracker for post resolution overlay
zwarm Apr 11, 2024
8089791
Implement tracking for post resolution conflict overlay
zwarm Apr 11, 2024
0f4f175
Fixes: PostListMainViewModel multiple starts
pantstamp Apr 12, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
Expand Up @@ -92,6 +92,7 @@
import org.wordpress.android.ui.posts.PostDatePickerDialogFragment;
import org.wordpress.android.ui.posts.PostListFragment;
import org.wordpress.android.ui.posts.PostNotificationScheduleTimeDialogFragment;
import org.wordpress.android.ui.posts.PostResolutionOverlayFragment;
import org.wordpress.android.ui.posts.PostSettingsListDialogFragment;
import org.wordpress.android.ui.posts.PostSettingsTagsFragment;
import org.wordpress.android.ui.posts.PostTimePickerDialogFragment;
Expand Down Expand Up @@ -555,4 +556,6 @@ public interface AppComponent {
void inject(WeekWidgetBlockListProviderFactory object);

void inject(WPMainNavigationView object);

void inject(PostResolutionOverlayFragment object);
}
Original file line number Diff line number Diff line change
Expand Up @@ -33,6 +33,7 @@
import org.wordpress.android.ui.posts.EditorBloggingPromptsViewModel;
import org.wordpress.android.ui.posts.EditorJetpackSocialViewModel;
import org.wordpress.android.ui.posts.PostListMainViewModel;
import org.wordpress.android.ui.posts.PostResolutionOverlayViewModel;
import org.wordpress.android.ui.posts.editor.StorePostViewModel;
import org.wordpress.android.ui.posts.prepublishing.PrepublishingViewModel;
import org.wordpress.android.ui.posts.prepublishing.categories.PrepublishingCategoriesViewModel;
Expand Down Expand Up @@ -539,4 +540,9 @@ abstract class ViewModelModule {
@IntoMap
@ViewModelKey(EditorJetpackSocialViewModel.class)
abstract ViewModel editorJetpackSocialViewModel(EditorJetpackSocialViewModel viewModel);

@Binds
@IntoMap
@ViewModelKey(PostResolutionOverlayViewModel.class)
abstract ViewModel postResolutionOverlayViewModel(PostResolutionOverlayViewModel viewModel);
}
Original file line number Diff line number Diff line change
Expand Up @@ -234,7 +234,7 @@ import org.wordpress.android.util.analytics.AnalyticsUtils
import org.wordpress.android.util.analytics.AnalyticsUtils.BlockEditorEnabledSource
import org.wordpress.android.util.config.ContactSupportFeatureConfig
import org.wordpress.android.util.config.GlobalStyleSupportFeatureConfig
import org.wordpress.android.util.config.SyncPublishingFeatureConfig
import org.wordpress.android.util.config.PostConflictResolutionFeatureConfig
import org.wordpress.android.util.extensions.setLiftOnScrollTargetViewIdAndRequestLayout
import org.wordpress.android.util.helpers.MediaFile
import org.wordpress.android.util.helpers.MediaGallery
Expand Down Expand Up @@ -401,7 +401,7 @@ class EditPostActivity : LocaleAwareActivity(), EditorFragmentActivity, EditorIm

@Inject lateinit var contactSupportFeatureConfig: ContactSupportFeatureConfig

@Inject lateinit var syncPublishingFeatureConfig: SyncPublishingFeatureConfig
@Inject lateinit var postConflictResolutionFeatureConfig: PostConflictResolutionFeatureConfig

@Inject lateinit var storePostViewModel: StorePostViewModel
@Inject lateinit var storageUtilsViewModel: StorageUtilsViewModel
Expand Down Expand Up @@ -591,7 +591,7 @@ class EditPostActivity : LocaleAwareActivity(), EditorFragmentActivity, EditorIm
updatingPostArea = findViewById(R.id.updating)

// check if post content needs updating
if (syncPublishingFeatureConfig.isEnabled()) {
if (postConflictResolutionFeatureConfig.isEnabled()) {
storePostViewModel.checkIfUpdatedPostVersionExists((editPostRepository), siteModel)
}
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -74,7 +74,7 @@ class PostActionHandler(
private val showToast: (ToastMessageHolder) -> Unit,
private val triggerPreviewStateUpdate: (PostListRemotePreviewState, PostInfoType) -> Unit,
private val copyPost: (SiteModel, PostModel, Boolean) -> Unit,
private val syncPublishingFeatureUtils: SyncPublishingFeatureUtils
private val postConflictResolutionFeatureUtils: PostConflictResolutionFeatureUtils
) {
private val criticalPostActionTracker = CriticalPostActionTracker(onStateChanged = {
invalidateList.invoke()
Expand Down Expand Up @@ -209,7 +209,7 @@ class PostActionHandler(
}
post.setStatus(DRAFT.toString())
dispatcher.dispatch(PostActionBuilder.newPushPostAction(
syncPublishingFeatureUtils.getRemotePostPayloadForPush(RemotePostPayload(post, site))
postConflictResolutionFeatureUtils.getRemotePostPayloadForPush(RemotePostPayload(post, site))
))

val localPostId = LocalId(post.id)
Expand Down
Original file line number Diff line number Diff line change
@@ -1,14 +1,14 @@
package org.wordpress.android.ui.posts

import org.wordpress.android.fluxc.store.PostStore.RemotePostPayload
import org.wordpress.android.util.config.SyncPublishingFeatureConfig
import org.wordpress.android.util.config.PostConflictResolutionFeatureConfig
import javax.inject.Inject

class SyncPublishingFeatureUtils @Inject constructor(
private val syncPublishingFeatureConfig: SyncPublishingFeatureConfig
class PostConflictResolutionFeatureUtils @Inject constructor(
private val postConflictResolutionFeatureConfig: PostConflictResolutionFeatureConfig
) {
private fun isSyncPublishingEnabled(): Boolean {
return syncPublishingFeatureConfig.isEnabled()
fun isPostConflictResolutionEnabled(): Boolean {
return postConflictResolutionFeatureConfig.isEnabled()
}

/**
Expand All @@ -21,7 +21,7 @@ class SyncPublishingFeatureUtils @Inject constructor(
* the remote version.
*/
fun getRemotePostPayloadForPush(payload: RemotePostPayload): RemotePostPayload {
if (isSyncPublishingEnabled().not()) {
if (isPostConflictResolutionEnabled().not()) {
payload.shouldSkipConflictResolutionCheck = true
}
return payload
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -28,7 +28,9 @@ private const val POST_TYPE = "post_type"
class PostListDialogHelper(
private val showDialog: (DialogHolder) -> Unit,
private val checkNetworkConnection: () -> Boolean,
private val analyticsTracker: AnalyticsTrackerWrapper
private val analyticsTracker: AnalyticsTrackerWrapper,
private val showConflictResolutionOverlay: ((PostResolutionOverlayActionEvent.ShowDialogAction) -> Unit)? = null,
private val isPostConflictResolutionEnabled: Boolean
) {
// Since we are using DialogFragments we need to hold onto which post will be published or trashed / resolved
private var localPostIdForDeleteDialog: Int? = null
Expand Down Expand Up @@ -115,28 +117,45 @@ class PostListDialogHelper(
}

fun showConflictedPostResolutionDialog(post: PostModel) {
val dialogHolder = DialogHolder(
tag = CONFIRM_ON_CONFLICT_LOAD_REMOTE_POST_DIALOG_TAG,
title = UiStringRes(R.string.dialog_confirm_load_remote_post_title),
message = UiStringText(PostUtils.getConflictedPostCustomStringForDialog(post)),
positiveButton = UiStringRes(R.string.dialog_confirm_load_remote_post_discard_local),
negativeButton = UiStringRes(R.string.dialog_confirm_load_remote_post_discard_web)
)
localPostIdForConflictResolutionDialog = post.id
showDialog.invoke(dialogHolder)
if (isPostConflictResolutionEnabled) {
showConflictResolutionOverlay?.invoke(
PostResolutionOverlayActionEvent.ShowDialogAction(
post,
PostResolutionType.SYNC_CONFLICT
)
)
} else {
val dialogHolder = DialogHolder(
tag = CONFIRM_ON_CONFLICT_LOAD_REMOTE_POST_DIALOG_TAG,
title = UiStringRes(R.string.dialog_confirm_load_remote_post_title),
message = UiStringText(PostUtils.getConflictedPostCustomStringForDialog(post)),
positiveButton = UiStringRes(R.string.dialog_confirm_load_remote_post_discard_local),
negativeButton = UiStringRes(R.string.dialog_confirm_load_remote_post_discard_web)
)
showDialog.invoke(dialogHolder)
}
}

fun showAutoSaveRevisionDialog(post: PostModel) {
analyticsTracker.track(UNPUBLISHED_REVISION_DIALOG_SHOWN, mapOf(POST_TYPE to "post"))
val dialogHolder = DialogHolder(
tag = CONFIRM_ON_AUTOSAVE_REVISION_DIALOG_TAG,
title = UiStringRes(R.string.dialog_confirm_autosave_title),
message = PostUtils.getCustomStringForAutosaveRevisionDialog(post),
positiveButton = UiStringRes(R.string.dialog_confirm_autosave_restore_button),
negativeButton = UiStringRes(R.string.dialog_confirm_autosave_dont_restore_button)
)
localPostIdForAutosaveRevisionResolutionDialog = post.id
showDialog.invoke(dialogHolder)
if (isPostConflictResolutionEnabled) {
showConflictResolutionOverlay?.invoke(
PostResolutionOverlayActionEvent.ShowDialogAction(
post, PostResolutionType.AUTOSAVE_REVISION_CONFLICT
)
)
} else {
analyticsTracker.track(UNPUBLISHED_REVISION_DIALOG_SHOWN, mapOf(POST_TYPE to "post"))
val dialogHolder = DialogHolder(
tag = CONFIRM_ON_AUTOSAVE_REVISION_DIALOG_TAG,
title = UiStringRes(R.string.dialog_confirm_autosave_title),
message = PostUtils.getCustomStringForAutosaveRevisionDialog(post),
positiveButton = UiStringRes(R.string.dialog_confirm_autosave_restore_button),
negativeButton = UiStringRes(R.string.dialog_confirm_autosave_dont_restore_button)
)
showDialog.invoke(dialogHolder)
}
}

fun showCopyConflictDialog(post: PostModel) {
Expand Down Expand Up @@ -256,4 +275,71 @@ class PostListDialogHelper(
)
}
}

fun onPostResolutionConfirmed(
event: PostResolutionOverlayActionEvent.PostResolutionConfirmationEvent,
updateConflictedPostWithRemoteVersion: (Int) -> Unit,
editRestoredAutoSavePost: (Int) -> Unit,
editLocalPost: (Int) -> Unit,
updateConflictedPostWithLocalVersion: (Int) -> Unit
) {
when (event.postResolutionType) {
PostResolutionType.AUTOSAVE_REVISION_CONFLICT -> {
handleAutosaveRevisionConflict(event, editRestoredAutoSavePost, editLocalPost)
}

PostResolutionType.SYNC_CONFLICT -> {
handleSyncRevisionConflict(
event,
updateConflictedPostWithLocalVersion,
updateConflictedPostWithRemoteVersion
)
}
}
}

private fun handleAutosaveRevisionConflict(
event: PostResolutionOverlayActionEvent.PostResolutionConfirmationEvent,
editRestoredAutoSavePost: (Int) -> Unit,
editLocalPost: (Int) -> Unit
) {
when (event.postResolutionConfirmationType) {
PostResolutionConfirmationType.CONFIRM_LOCAL -> {
localPostIdForAutosaveRevisionResolutionDialog?.let {
// open the editor with the local post (don't use the auto save version)
editLocalPost(it)
}
}

PostResolutionConfirmationType.CONFIRM_OTHER -> {
localPostIdForAutosaveRevisionResolutionDialog?.let {
// open the editor with the restored auto save
localPostIdForAutosaveRevisionResolutionDialog = null
editRestoredAutoSavePost(it)
}
}
}
}

private fun handleSyncRevisionConflict(
event: PostResolutionOverlayActionEvent.PostResolutionConfirmationEvent,
updateConflictedPostWithLocalVersion: (Int) -> Unit,
updateConflictedPostWithRemoteVersion: (Int) -> Unit
) {
when (event.postResolutionConfirmationType) {
PostResolutionConfirmationType.CONFIRM_LOCAL -> {
localPostIdForConflictResolutionDialog?.let {
updateConflictedPostWithLocalVersion(it)
}
}

PostResolutionConfirmationType.CONFIRM_OTHER -> {
localPostIdForConflictResolutionDialog?.let {
localPostIdForConflictResolutionDialog = null
// here load version from remote
updateConflictedPostWithRemoteVersion(it)
}
}
}
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -83,8 +83,10 @@ class PostListMainViewModel @Inject constructor(
@Named(UI_THREAD) private val mainDispatcher: CoroutineDispatcher,
@Named(BG_THREAD) private val bgDispatcher: CoroutineDispatcher,
private val uploadStarter: UploadStarter,
private val syncPublishingFeatureUtils: SyncPublishingFeatureUtils
private val postConflictResolutionFeatureUtils: PostConflictResolutionFeatureUtils
) : ViewModel(), CoroutineScope {
private var isStarted = false

private val lifecycleOwner = object : LifecycleOwner {
val lifecycleRegistry = LifecycleRegistry(this)
override val lifecycle: Lifecycle = lifecycleRegistry
Expand Down Expand Up @@ -128,6 +130,10 @@ class PostListMainViewModel @Inject constructor(
private val _dialogAction = SingleLiveEvent<DialogHolder>()
val dialogAction: LiveData<DialogHolder> = _dialogAction

private val _conflictResolutionAction = SingleLiveEvent<PostResolutionOverlayActionEvent.ShowDialogAction>()
val conflictResolutionAction: LiveData<PostResolutionOverlayActionEvent.ShowDialogAction> =
_conflictResolutionAction

private val _postUploadAction = SingleLiveEvent<PostUploadAction>()
val postUploadAction: LiveData<PostUploadAction> = _postUploadAction

Expand All @@ -150,8 +156,10 @@ class PostListMainViewModel @Inject constructor(
private val postListDialogHelper: PostListDialogHelper by lazy {
PostListDialogHelper(
showDialog = { _dialogAction.postValue(it) },
showConflictResolutionOverlay = { _conflictResolutionAction.postValue(it) },
checkNetworkConnection = this::checkNetworkConnection,
analyticsTracker = analyticsTracker
analyticsTracker = analyticsTracker,
isPostConflictResolutionEnabled = postConflictResolutionFeatureUtils.isPostConflictResolutionEnabled()
)
}

Expand Down Expand Up @@ -186,7 +194,7 @@ class PostListMainViewModel @Inject constructor(
showToast = { _toastMessage.postValue(it) },
triggerPreviewStateUpdate = this::updatePreviewAndDialogState,
copyPost = this::copyPost,
syncPublishingFeatureUtils = syncPublishingFeatureUtils
postConflictResolutionFeatureUtils = postConflictResolutionFeatureUtils
)
}

Expand Down Expand Up @@ -235,6 +243,7 @@ class PostListMainViewModel @Inject constructor(
currentBottomSheetPostId: LocalId,
editPostRepository: EditPostRepository
) {
if (isStarted) return
this.site = site
this.editPostRepository = editPostRepository

Expand Down Expand Up @@ -294,6 +303,8 @@ class PostListMainViewModel @Inject constructor(
savePostToDbUseCase.savePostToDb(editPostRepository, site)
})
}

isStarted = true
}

override fun onCleared() {
Expand Down Expand Up @@ -478,6 +489,17 @@ class PostListMainViewModel @Inject constructor(
)
}

// Post Resolution Overlay Actions
fun onPostResolutionConfirmed(event: PostResolutionOverlayActionEvent.PostResolutionConfirmationEvent) {
postListDialogHelper.onPostResolutionConfirmed(
event = event,
updateConflictedPostWithRemoteVersion = postConflictResolver::updateConflictedPostWithRemoteVersion,
editRestoredAutoSavePost = this::editRestoredAutoSavePost,
editLocalPost = this::editLocalPost,
updateConflictedPostWithLocalVersion = postConflictResolver::updateConflictedPostWithLocalVersion
)
}

private fun showPrepublishingBottomSheet(post: PostModel) {
currentBottomSheetPostId = LocalId(post.id)
editPostRepository.loadPostByLocalPostId(post.id)
Expand Down
Loading
Loading