|
@@ -59,6 +59,7 @@ import androidx.emoji2.text.EmojiCompat
|
|
|
import androidx.fragment.app.DialogFragment
|
|
|
import androidx.fragment.app.commit
|
|
|
import androidx.lifecycle.ViewModelProvider
|
|
|
+import androidx.lifecycle.lifecycleScope
|
|
|
import androidx.recyclerview.widget.ItemTouchHelper
|
|
|
import androidx.recyclerview.widget.LinearLayoutManager
|
|
|
import androidx.recyclerview.widget.RecyclerView
|
|
@@ -104,6 +105,7 @@ import com.nextcloud.talk.adapters.messages.UnreadNoticeMessageViewHolder
|
|
|
import com.nextcloud.talk.adapters.messages.VoiceMessageInterface
|
|
|
import com.nextcloud.talk.api.NcApi
|
|
|
import com.nextcloud.talk.application.NextcloudTalkApplication
|
|
|
+import com.nextcloud.talk.chat.data.model.ChatMessage
|
|
|
import com.nextcloud.talk.chat.viewmodels.ChatViewModel
|
|
|
import com.nextcloud.talk.chat.viewmodels.MessageInputViewModel
|
|
|
import com.nextcloud.talk.conversationinfo.ConversationInfoActivity
|
|
@@ -119,14 +121,9 @@ import com.nextcloud.talk.jobs.UploadAndShareFilesWorker
|
|
|
import com.nextcloud.talk.location.LocationPickerActivity
|
|
|
import com.nextcloud.talk.messagesearch.MessageSearchActivity
|
|
|
import com.nextcloud.talk.models.domain.ConversationModel
|
|
|
-import com.nextcloud.talk.models.domain.ConversationReadOnlyState
|
|
|
-import com.nextcloud.talk.models.domain.ConversationType
|
|
|
-import com.nextcloud.talk.models.domain.LobbyState
|
|
|
-import com.nextcloud.talk.models.domain.ObjectType
|
|
|
import com.nextcloud.talk.models.json.capabilities.SpreedCapability
|
|
|
-import com.nextcloud.talk.models.json.chat.ChatMessage
|
|
|
-import com.nextcloud.talk.models.json.chat.ChatOverall
|
|
|
import com.nextcloud.talk.models.json.chat.ReadStatus
|
|
|
+import com.nextcloud.talk.models.json.conversations.ConversationEnums
|
|
|
import com.nextcloud.talk.polls.ui.PollCreateDialogFragment
|
|
|
import com.nextcloud.talk.remotefilebrowser.activities.RemoteFileBrowserActivity
|
|
|
import com.nextcloud.talk.shareditems.activities.SharedItemsActivity
|
|
@@ -183,6 +180,8 @@ import com.stfalcon.chatkit.messages.MessagesListAdapter
|
|
|
import com.stfalcon.chatkit.utils.DateFormatter
|
|
|
import kotlinx.coroutines.CoroutineScope
|
|
|
import kotlinx.coroutines.Dispatchers
|
|
|
+import kotlinx.coroutines.flow.collect
|
|
|
+import kotlinx.coroutines.flow.onEach
|
|
|
import kotlinx.coroutines.launch
|
|
|
import kotlinx.coroutines.withContext
|
|
|
import org.greenrobot.eventbus.Subscribe
|
|
@@ -408,6 +407,7 @@ class ChatActivity :
|
|
|
handleIntent(intent)
|
|
|
|
|
|
chatViewModel = ViewModelProvider(this, viewModelFactory)[ChatViewModel::class.java]
|
|
|
+
|
|
|
messageInputViewModel = ViewModelProvider(this, viewModelFactory)[MessageInputViewModel::class.java]
|
|
|
|
|
|
binding.progressBar.visibility = View.VISIBLE
|
|
@@ -521,12 +521,37 @@ class ChatActivity :
|
|
|
@Suppress("LongMethod")
|
|
|
private fun initObservers() {
|
|
|
Log.d(TAG, "initObservers Called")
|
|
|
+
|
|
|
+ this.lifecycleScope.launch {
|
|
|
+ chatViewModel.getConversationFlow
|
|
|
+ .onEach { conversationModel ->
|
|
|
+ currentConversation = conversationModel
|
|
|
+
|
|
|
+ val urlForChatting = ApiUtils.getUrlForChat(chatApiVersion, conversationUser?.baseUrl, roomToken)
|
|
|
+ val credentials = ApiUtils.getCredentials(conversationUser!!.username, conversationUser!!.token)
|
|
|
+
|
|
|
+ chatViewModel.setData(
|
|
|
+ currentConversation!!,
|
|
|
+ credentials!!,
|
|
|
+ urlForChatting
|
|
|
+ )
|
|
|
+
|
|
|
+ logConversationInfos("GetRoomSuccessState")
|
|
|
+
|
|
|
+ if (adapter == null) {
|
|
|
+ initAdapter()
|
|
|
+ binding.messagesListView.setAdapter(adapter)
|
|
|
+ layoutManager = binding.messagesListView.layoutManager as LinearLayoutManager?
|
|
|
+ }
|
|
|
+
|
|
|
+ chatViewModel.getCapabilities(conversationUser!!, roomToken, currentConversation!!)
|
|
|
+ }.collect()
|
|
|
+ }
|
|
|
+
|
|
|
chatViewModel.getRoomViewState.observe(this) { state ->
|
|
|
when (state) {
|
|
|
is ChatViewModel.GetRoomSuccessState -> {
|
|
|
- currentConversation = state.conversationModel
|
|
|
- logConversationInfos("GetRoomSuccessState")
|
|
|
- chatViewModel.getCapabilities(conversationUser!!, roomToken, currentConversation!!)
|
|
|
+ // unused atm
|
|
|
}
|
|
|
|
|
|
is ChatViewModel.GetRoomErrorState -> {
|
|
@@ -569,24 +594,29 @@ class ChatActivity :
|
|
|
binding.chatToolbar.setOnClickListener { _ -> showConversationInfoScreen() }
|
|
|
}
|
|
|
|
|
|
- if (adapter == null) {
|
|
|
- initAdapter()
|
|
|
- binding.messagesListView.setAdapter(adapter)
|
|
|
- layoutManager = binding.messagesListView.layoutManager as LinearLayoutManager?
|
|
|
- }
|
|
|
+ // if (adapter == null) {
|
|
|
+ // initAdapter()
|
|
|
+ // binding.messagesListView.setAdapter(adapter)
|
|
|
+ // layoutManager = binding.messagesListView.layoutManager as LinearLayoutManager?
|
|
|
+ // }
|
|
|
|
|
|
loadAvatarForStatusBar()
|
|
|
setupSwipeToReply()
|
|
|
setActionBarTitle()
|
|
|
updateRoomTimerHandler()
|
|
|
|
|
|
- chatViewModel.refreshChatParams(
|
|
|
- setupFieldsForPullChatMessages(
|
|
|
- false,
|
|
|
- 0,
|
|
|
- false
|
|
|
- )
|
|
|
+ val urlForChatting = ApiUtils.getUrlForChat(chatApiVersion, conversationUser?.baseUrl, roomToken)
|
|
|
+
|
|
|
+ chatViewModel.loadMessages(
|
|
|
+ withCredentials = credentials!!,
|
|
|
+ withUrl = urlForChatting,
|
|
|
)
|
|
|
+
|
|
|
+ // chatViewModel.initMessagePolling(
|
|
|
+ // withCredentials = credentials!!,
|
|
|
+ // withUrl = urlForChatting,
|
|
|
+ // roomToken = currentConversation!!.token!!
|
|
|
+ // )
|
|
|
}
|
|
|
|
|
|
is ChatViewModel.GetCapabilitiesErrorState -> {
|
|
@@ -705,6 +735,11 @@ class ChatActivity :
|
|
|
Snackbar.LENGTH_LONG
|
|
|
).show()
|
|
|
}
|
|
|
+
|
|
|
+ val id = state.msg.ocs!!.data!!.parentMessage!!.id.toString()
|
|
|
+ val index = adapter?.getMessagePositionById(id) ?: 0
|
|
|
+ val message = adapter?.items?.get(index)?.item as ChatMessage
|
|
|
+ setMessageAsDeleted(message)
|
|
|
}
|
|
|
|
|
|
is ChatViewModel.DeleteChatMessageErrorState -> {
|
|
@@ -738,128 +773,70 @@ class ChatActivity :
|
|
|
}
|
|
|
}
|
|
|
|
|
|
- chatViewModel.getFieldMapForChat.observe(this) { fieldMap ->
|
|
|
- if (fieldMap.isNotEmpty()) {
|
|
|
- chatViewModel.pullChatMessages(
|
|
|
- credentials!!,
|
|
|
- ApiUtils.getUrlForChat(chatApiVersion, conversationUser?.baseUrl, roomToken)
|
|
|
- )
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- chatViewModel.pullChatMessageViewState.observe(this) { state ->
|
|
|
+ chatViewModel.chatMessageViewState.observe(this) { state ->
|
|
|
when (state) {
|
|
|
- is ChatViewModel.PullChatMessageSuccessState -> {
|
|
|
- Log.d(TAG, "PullChatMessageSuccess: Code: ${state.response.code()}")
|
|
|
- when (state.response.code()) {
|
|
|
- HTTP_CODE_OK -> {
|
|
|
- Log.d(TAG, "lookIntoFuture: ${state.lookIntoFuture}")
|
|
|
- val chatOverall = state.response.body() as ChatOverall?
|
|
|
- var chatMessageList = chatOverall?.ocs!!.data!!
|
|
|
-
|
|
|
- val newXChatLastCommonRead = state.response.headers()["X-Chat-Last-Common-Read"]?.let {
|
|
|
- Integer.parseInt(it)
|
|
|
- }
|
|
|
-
|
|
|
- processHeaderChatLastGiven(state.response, state.lookIntoFuture)
|
|
|
-
|
|
|
- chatMessageList = handleSystemMessages(chatMessageList)
|
|
|
-
|
|
|
- if (chatMessageList.isEmpty()) {
|
|
|
- chatViewModel.refreshChatParams(
|
|
|
- setupFieldsForPullChatMessages(
|
|
|
- true,
|
|
|
- newXChatLastCommonRead,
|
|
|
- true
|
|
|
- )
|
|
|
- )
|
|
|
- return@observe
|
|
|
- }
|
|
|
-
|
|
|
- determinePreviousMessageIds(chatMessageList)
|
|
|
-
|
|
|
- handleExpandableSystemMessages(chatMessageList)
|
|
|
-
|
|
|
- if (chatMessageList.isNotEmpty() &&
|
|
|
- ChatMessage.SystemMessageType.CLEARED_CHAT == chatMessageList[0].systemMessageType
|
|
|
- ) {
|
|
|
- adapter?.clear()
|
|
|
- adapter?.notifyDataSetChanged()
|
|
|
- }
|
|
|
+ is ChatViewModel.ChatMessageStartState -> {
|
|
|
+ // Handle UI on first load
|
|
|
+ cancelNotificationsForCurrentConversation()
|
|
|
+ binding.progressBar.visibility = View.GONE
|
|
|
+ binding.messagesListView.visibility = View.VISIBLE
|
|
|
+ collapseSystemMessages()
|
|
|
+ }
|
|
|
|
|
|
- var lastAdapterId = getLastAdapterId()
|
|
|
- val oneNewMessage = (lastAdapterId != 0 || chatMessageList.size == 1)
|
|
|
-
|
|
|
- if (
|
|
|
- state.lookIntoFuture &&
|
|
|
- oneNewMessage &&
|
|
|
- chatMessageList[0].jsonMessageId > lastAdapterId
|
|
|
- ) {
|
|
|
- processMessagesFromTheFuture(chatMessageList)
|
|
|
- } else if (!state.lookIntoFuture) {
|
|
|
- processMessagesNotFromTheFuture(chatMessageList)
|
|
|
- collapseSystemMessages()
|
|
|
- }
|
|
|
+ is ChatViewModel.ChatMessageUpdateState -> {
|
|
|
+ // unused atm
|
|
|
+ }
|
|
|
|
|
|
- updateReadStatusOfAllMessages(newXChatLastCommonRead)
|
|
|
+ is ChatViewModel.ChatMessageErrorState -> {
|
|
|
+ // unused atm
|
|
|
+ }
|
|
|
|
|
|
- processCallStartedMessages(chatMessageList)
|
|
|
+ else -> {}
|
|
|
+ }
|
|
|
+ }
|
|
|
|
|
|
- adapter?.notifyDataSetChanged()
|
|
|
+ this.lifecycleScope.launch {
|
|
|
+ chatViewModel.getMessageFlow
|
|
|
+ .onEach { pair ->
|
|
|
+ val lookIntoFuture = pair.first
|
|
|
+ var chatMessageList = pair.second
|
|
|
|
|
|
- chatViewModel.refreshChatParams(
|
|
|
- setupFieldsForPullChatMessages(
|
|
|
- true,
|
|
|
- newXChatLastCommonRead,
|
|
|
- true
|
|
|
- )
|
|
|
- )
|
|
|
- }
|
|
|
+ chatMessageList = handleSystemMessages(chatMessageList)
|
|
|
|
|
|
- HTTP_CODE_NOT_MODIFIED -> {
|
|
|
- chatViewModel.refreshChatParams(
|
|
|
- setupFieldsForPullChatMessages(
|
|
|
- true,
|
|
|
- globalLastKnownPastMessageId,
|
|
|
- true
|
|
|
- )
|
|
|
- )
|
|
|
- }
|
|
|
+ determinePreviousMessageIds(chatMessageList)
|
|
|
|
|
|
- HTTP_CODE_PRECONDITION_FAILED -> {
|
|
|
- chatViewModel.refreshChatParams(
|
|
|
- setupFieldsForPullChatMessages(
|
|
|
- true,
|
|
|
- globalLastKnownPastMessageId,
|
|
|
- true
|
|
|
- )
|
|
|
- )
|
|
|
- }
|
|
|
+ handleExpandableSystemMessages(chatMessageList)
|
|
|
|
|
|
- else -> {}
|
|
|
+ if (chatMessageList.isNotEmpty() &&
|
|
|
+ ChatMessage.SystemMessageType.CLEARED_CHAT == chatMessageList[0].systemMessageType
|
|
|
+ ) {
|
|
|
+ adapter?.clear()
|
|
|
+ adapter?.notifyDataSetChanged()
|
|
|
+ // TODO: remove messages from DB, Should be handled beforehand (in viewModel?)
|
|
|
}
|
|
|
|
|
|
- processExpiredMessages()
|
|
|
- if (isFirstMessagesProcessing) {
|
|
|
- cancelNotificationsForCurrentConversation()
|
|
|
- isFirstMessagesProcessing = false
|
|
|
- binding.progressBar.visibility = View.GONE
|
|
|
- binding.messagesListView.visibility = View.VISIBLE
|
|
|
-
|
|
|
+ if (lookIntoFuture) {
|
|
|
+ processMessagesFromTheFuture(chatMessageList)
|
|
|
+ } else {
|
|
|
+ processMessagesNotFromTheFuture(chatMessageList)
|
|
|
collapseSystemMessages()
|
|
|
}
|
|
|
- }
|
|
|
|
|
|
- is ChatViewModel.PullChatMessageCompleteState -> {
|
|
|
- Log.d(TAG, "PullChatMessageCompleted")
|
|
|
- }
|
|
|
+ processCallStartedMessages(chatMessageList)
|
|
|
|
|
|
- is ChatViewModel.PullChatMessageErrorState -> {
|
|
|
- Log.d(TAG, "PullChatMessageError")
|
|
|
+ adapter?.notifyDataSetChanged()
|
|
|
}
|
|
|
+ .collect()
|
|
|
|
|
|
- else -> {}
|
|
|
- }
|
|
|
+
|
|
|
+ chatViewModel.getUpdateMessageFlow
|
|
|
+ .onEach { pair ->
|
|
|
+ val lookIntoFuture = pair.first
|
|
|
+ var chatMessageList = pair.second
|
|
|
+
|
|
|
+ adapter!!.update(chatMessageList[0])
|
|
|
+ }
|
|
|
+ .collect()
|
|
|
}
|
|
|
|
|
|
chatViewModel.reactionDeletedViewState.observe(this) { state ->
|
|
@@ -916,6 +893,11 @@ class ChatActivity :
|
|
|
).show()
|
|
|
}
|
|
|
}
|
|
|
+ val newString = state.messageEdited.ocs?.data?.parentMessage?.message ?: "(null)"
|
|
|
+ val id = state.messageEdited.ocs?.data?.parentMessage?.id.toString()
|
|
|
+ val index = adapter?.getMessagePositionById(id) ?: 0
|
|
|
+ val message = adapter?.items?.get(index)?.item as ChatMessage
|
|
|
+ setMessageAsEdited(message, newString)
|
|
|
}
|
|
|
|
|
|
is MessageInputViewModel.EditMessageErrorState -> {
|
|
@@ -1412,15 +1394,15 @@ class ChatActivity :
|
|
|
|
|
|
fun isOneToOneConversation() =
|
|
|
currentConversation != null && currentConversation?.type != null &&
|
|
|
- currentConversation?.type == ConversationType.ROOM_TYPE_ONE_TO_ONE_CALL
|
|
|
+ currentConversation?.type == ConversationEnums.ConversationType.ROOM_TYPE_ONE_TO_ONE_CALL
|
|
|
|
|
|
private fun isGroupConversation() =
|
|
|
currentConversation != null && currentConversation?.type != null &&
|
|
|
- currentConversation?.type == ConversationType.ROOM_GROUP_CALL
|
|
|
+ currentConversation?.type == ConversationEnums.ConversationType.ROOM_GROUP_CALL
|
|
|
|
|
|
private fun isPublicConversation() =
|
|
|
currentConversation != null && currentConversation?.type != null &&
|
|
|
- currentConversation?.type == ConversationType.ROOM_PUBLIC_CALL
|
|
|
+ currentConversation?.type == ConversationEnums.ConversationType.ROOM_PUBLIC_CALL
|
|
|
|
|
|
private fun updateRoomTimerHandler() {
|
|
|
val delayForRecursiveCall = if (shouldShowLobby()) {
|
|
@@ -1443,7 +1425,7 @@ class ChatActivity :
|
|
|
private fun switchToRoom(token: String, startCallAfterRoomSwitch: Boolean, isVoiceOnlyCall: Boolean) {
|
|
|
if (conversationUser != null) {
|
|
|
runOnUiThread {
|
|
|
- if (currentConversation?.objectType == ObjectType.ROOM) {
|
|
|
+ if (currentConversation?.objectType == ConversationEnums.ObjectType.ROOM) {
|
|
|
Snackbar.make(
|
|
|
binding.root,
|
|
|
context.resources.getString(R.string.switch_to_main_room),
|
|
@@ -1826,7 +1808,7 @@ class ChatActivity :
|
|
|
private fun shouldShowLobby(): Boolean {
|
|
|
if (currentConversation != null) {
|
|
|
return CapabilitiesUtil.hasSpreedFeatureCapability(spreedCapabilities, SpreedFeatures.WEBINARY_LOBBY) &&
|
|
|
- currentConversation?.lobbyState == LobbyState.LOBBY_STATE_MODERATORS_ONLY &&
|
|
|
+ currentConversation?.lobbyState == ConversationEnums.LobbyState.LOBBY_STATE_MODERATORS_ONLY &&
|
|
|
!ConversationUtils.canModerate(currentConversation!!, spreedCapabilities) &&
|
|
|
!participantPermissions.canIgnoreLobby()
|
|
|
}
|
|
@@ -1862,7 +1844,7 @@ class ChatActivity :
|
|
|
private fun isReadOnlyConversation(): Boolean {
|
|
|
return currentConversation?.conversationReadOnlyState != null &&
|
|
|
currentConversation?.conversationReadOnlyState ==
|
|
|
- ConversationReadOnlyState.CONVERSATION_READ_ONLY
|
|
|
+ ConversationEnums.ConversationReadOnlyState.CONVERSATION_READ_ONLY
|
|
|
}
|
|
|
|
|
|
private fun checkLobbyState() {
|
|
@@ -2327,7 +2309,7 @@ class ChatActivity :
|
|
|
""
|
|
|
}
|
|
|
|
|
|
- if (currentConversation?.type == ConversationType.ROOM_TYPE_ONE_TO_ONE_CALL) {
|
|
|
+ if (currentConversation?.type == ConversationEnums.ConversationType.ROOM_TYPE_ONE_TO_ONE_CALL) {
|
|
|
var statusMessage = ""
|
|
|
if (currentConversation?.statusIcon != null) {
|
|
|
statusMessage += currentConversation?.statusIcon
|
|
@@ -2337,8 +2319,8 @@ class ChatActivity :
|
|
|
}
|
|
|
statusMessageViewContents(statusMessage)
|
|
|
} else {
|
|
|
- if (currentConversation?.type == ConversationType.ROOM_GROUP_CALL ||
|
|
|
- currentConversation?.type == ConversationType.ROOM_PUBLIC_CALL
|
|
|
+ if (currentConversation?.type == ConversationEnums.ConversationType.ROOM_GROUP_CALL ||
|
|
|
+ currentConversation?.type == ConversationEnums.ConversationType.ROOM_PUBLIC_CALL
|
|
|
) {
|
|
|
var descriptionMessage = ""
|
|
|
descriptionMessage += currentConversation?.description
|
|
@@ -2610,9 +2592,9 @@ class ChatActivity :
|
|
|
GROUPED_MESSAGES_SAME_AUTHOR_THRESHOLD > 0
|
|
|
)
|
|
|
chatMessage.isOneToOneConversation =
|
|
|
- (currentConversation?.type == ConversationType.ROOM_TYPE_ONE_TO_ONE_CALL)
|
|
|
+ (currentConversation?.type == ConversationEnums.ConversationType.ROOM_TYPE_ONE_TO_ONE_CALL)
|
|
|
chatMessage.isFormerOneToOneConversation =
|
|
|
- (currentConversation?.type == ConversationType.FORMER_ONE_TO_ONE)
|
|
|
+ (currentConversation?.type == ConversationEnums.ConversationType.FORMER_ONE_TO_ONE)
|
|
|
it.addToStart(chatMessage, scrollToEndOnUpdate)
|
|
|
}
|
|
|
}
|
|
@@ -2640,9 +2622,9 @@ class ChatActivity :
|
|
|
|
|
|
val chatMessage = chatMessageList[i]
|
|
|
chatMessage.isOneToOneConversation =
|
|
|
- currentConversation?.type == ConversationType.ROOM_TYPE_ONE_TO_ONE_CALL
|
|
|
+ currentConversation?.type == ConversationEnums.ConversationType.ROOM_TYPE_ONE_TO_ONE_CALL
|
|
|
chatMessage.isFormerOneToOneConversation =
|
|
|
- (currentConversation?.type == ConversationType.FORMER_ONE_TO_ONE)
|
|
|
+ (currentConversation?.type == ConversationEnums.ConversationType.FORMER_ONE_TO_ONE)
|
|
|
chatMessage.activeUser = conversationUser
|
|
|
chatMessage.token = roomToken
|
|
|
}
|
|
@@ -2721,6 +2703,7 @@ class ChatActivity :
|
|
|
if (!voiceMessageToRestoreId.equals("")) {
|
|
|
Log.d(RESUME_AUDIO_TAG, "begin method to resume audio playback")
|
|
|
|
|
|
+ // TODO: replace this logic by calling getItemFromAdapter(messageId)
|
|
|
if (adapter != null) {
|
|
|
Log.d(RESUME_AUDIO_TAG, "adapter is not null, proceeding")
|
|
|
val voiceMessagePosition = adapter!!.items!!.indexOfFirst {
|
|
@@ -2748,7 +2731,7 @@ class ChatActivity :
|
|
|
)
|
|
|
}
|
|
|
} else {
|
|
|
- Log.d(RESUME_AUDIO_TAG, "TalkMessagesListAdapater is null")
|
|
|
+ Log.d(RESUME_AUDIO_TAG, "TalkMessagesListAdapter is null")
|
|
|
}
|
|
|
} else {
|
|
|
Log.d(RESUME_AUDIO_TAG, "No voice message to restore")
|
|
@@ -2758,6 +2741,29 @@ class ChatActivity :
|
|
|
voiceMessageToRestoreWasPlaying = false
|
|
|
}
|
|
|
|
|
|
+ private fun getItemFromAdapter(messageId: String): ChatMessage? {
|
|
|
+ if (adapter != null) {
|
|
|
+ val messagePosition = adapter!!.items!!.indexOfFirst {
|
|
|
+ it.item is ChatMessage && (it.item as ChatMessage).id == messageId
|
|
|
+ }
|
|
|
+ if (messagePosition >= 0) {
|
|
|
+ val currentItem = adapter?.items?.get(messagePosition)?.item
|
|
|
+ if (currentItem is ChatMessage && currentItem.id == messageId) {
|
|
|
+ return currentItem
|
|
|
+ } else {
|
|
|
+ Log.d(TAG, "currentItem retrieved was not chatmessage or its id was not correct")
|
|
|
+ }
|
|
|
+ } else {
|
|
|
+ Log.d(
|
|
|
+ TAG, "messagePosition is -1, adapter # of items: " + adapter!!.itemCount
|
|
|
+ )
|
|
|
+ }
|
|
|
+ } else {
|
|
|
+ Log.d(TAG, "TalkMessagesListAdapter is null")
|
|
|
+ }
|
|
|
+ return null
|
|
|
+ }
|
|
|
+
|
|
|
private fun scrollToRequestedMessageIfNeeded() {
|
|
|
intent.getStringExtra(BundleKeys.KEY_MESSAGE_ID)?.let {
|
|
|
scrollToMessageWithId(it)
|
|
@@ -2771,16 +2777,21 @@ class ChatActivity :
|
|
|
}
|
|
|
|
|
|
override fun onLoadMore(page: Int, totalItemsCount: Int) {
|
|
|
- val calculatedPage = totalItemsCount / PAGE_SIZE
|
|
|
- if (calculatedPage > 0) {
|
|
|
- chatViewModel.refreshChatParams(
|
|
|
- setupFieldsForPullChatMessages(
|
|
|
- false,
|
|
|
- null,
|
|
|
- true
|
|
|
- )
|
|
|
- )
|
|
|
- }
|
|
|
+ val id = (
|
|
|
+ adapter?.items?.last {
|
|
|
+ it.item is ChatMessage
|
|
|
+ }?.item as ChatMessage
|
|
|
+ ).jsonMessageId
|
|
|
+
|
|
|
+ val urlForChatting = ApiUtils.getUrlForChat(chatApiVersion, conversationUser?.baseUrl, roomToken)
|
|
|
+
|
|
|
+ chatViewModel.loadMoreMessages(
|
|
|
+ beforeMessageId = id.toLong(),
|
|
|
+ withUrl = urlForChatting,
|
|
|
+ withCredentials = credentials!!,
|
|
|
+ withMessageLimit = MESSAGE_PULL_LIMIT,
|
|
|
+ roomToken = currentConversation!!.token!!
|
|
|
+ )
|
|
|
}
|
|
|
|
|
|
override fun format(date: Date): String {
|
|
@@ -2923,18 +2934,25 @@ class ChatActivity :
|
|
|
|
|
|
// setDeletionFlagsAndRemoveInfomessages
|
|
|
if (isInfoMessageAboutDeletion(currentMessage)) {
|
|
|
- if (!chatMessageMap.containsKey(currentMessage.value.parentMessage!!.id)) {
|
|
|
+ if (!chatMessageMap.containsKey(currentMessage.value.parentMessageId.toString())) {
|
|
|
// if chatMessageMap doesn't contain message to delete (this happens when lookingIntoFuture),
|
|
|
// the message to delete has to be modified directly inside the adapter
|
|
|
- setMessageAsDeleted(currentMessage.value.parentMessage)
|
|
|
+
|
|
|
+ val id = currentMessage.value.parentMessageId.toString()
|
|
|
+ val index = adapter?.getMessagePositionById(id) ?: 0
|
|
|
+
|
|
|
+ if (index > 0) {
|
|
|
+ val message = adapter?.items?.get(index)?.item as ChatMessage
|
|
|
+ setMessageAsDeleted(message)
|
|
|
+ }
|
|
|
} else {
|
|
|
- chatMessageMap[currentMessage.value.parentMessage!!.id]!!.isDeleted = true
|
|
|
+ chatMessageMap[currentMessage.value.parentMessageId.toString()]!!.isDeleted = true
|
|
|
}
|
|
|
chatMessageIterator.remove()
|
|
|
} else if (isReactionsMessage(currentMessage)) {
|
|
|
// delete reactions system messages
|
|
|
- if (!chatMessageMap.containsKey(currentMessage.value.parentMessage!!.id)) {
|
|
|
- updateAdapterForReaction(currentMessage.value.parentMessage)
|
|
|
+ if (!chatMessageMap.containsKey(currentMessage.value.parentMessageId.toString())) {
|
|
|
+ // updateAdapterForReaction(currentMessage.value.parentMessage) TODO
|
|
|
}
|
|
|
|
|
|
chatMessageIterator.remove()
|
|
@@ -2942,8 +2960,8 @@ class ChatActivity :
|
|
|
// delete poll system messages
|
|
|
chatMessageIterator.remove()
|
|
|
} else if (isEditMessage(currentMessage)) {
|
|
|
- if (!chatMessageMap.containsKey(currentMessage.value.parentMessage!!.id)) {
|
|
|
- setMessageAsEdited(currentMessage.value.parentMessage)
|
|
|
+ if (!chatMessageMap.containsKey(currentMessage.value.parentMessageId.toString())) {
|
|
|
+ // setMessageAsEdited(currentMessage.value.parentMessage) TODO
|
|
|
}
|
|
|
|
|
|
chatMessageIterator.remove()
|
|
@@ -2977,7 +2995,7 @@ class ChatActivity :
|
|
|
}
|
|
|
|
|
|
private fun isInfoMessageAboutDeletion(currentMessage: MutableMap.MutableEntry<String, ChatMessage>): Boolean {
|
|
|
- return currentMessage.value.parentMessage != null && currentMessage.value.systemMessageType == ChatMessage
|
|
|
+ return currentMessage.value.parentMessageId != null && currentMessage.value.systemMessageType == ChatMessage
|
|
|
.SystemMessageType.MESSAGE_DELETED
|
|
|
}
|
|
|
|
|
@@ -2988,7 +3006,7 @@ class ChatActivity :
|
|
|
}
|
|
|
|
|
|
private fun isEditMessage(currentMessage: MutableMap.MutableEntry<String, ChatMessage>): Boolean {
|
|
|
- return currentMessage.value.parentMessage != null && currentMessage.value.systemMessageType == ChatMessage
|
|
|
+ return currentMessage.value.parentMessageId != null && currentMessage.value.systemMessageType == ChatMessage
|
|
|
.SystemMessageType.MESSAGE_EDITED
|
|
|
}
|
|
|
|
|
@@ -3039,7 +3057,7 @@ class ChatActivity :
|
|
|
bundle.putBoolean(BundleKeys.KEY_CALL_WITHOUT_NOTIFICATION, true)
|
|
|
}
|
|
|
|
|
|
- if (it.objectType == ObjectType.ROOM) {
|
|
|
+ if (it.objectType == ConversationEnums.ObjectType.ROOM) {
|
|
|
bundle.putBoolean(KEY_IS_BREAKOUT_ROOM, true)
|
|
|
}
|
|
|
|
|
@@ -3285,7 +3303,7 @@ class ChatActivity :
|
|
|
val lon = data["longitude"]!!
|
|
|
metaData =
|
|
|
"{\"type\":\"geo-location\",\"id\":\"geo:$lat,$lon\",\"latitude\":\"$lat\"," +
|
|
|
- "\"longitude\":\"$lon\",\"name\":\"$name\"}"
|
|
|
+ "\"longitude\":\"$lon\",\"name\":\"$name\"}"
|
|
|
}
|
|
|
|
|
|
when (type) {
|
|
@@ -3350,7 +3368,7 @@ class ChatActivity :
|
|
|
conversationUser?.userId?.isNotEmpty() == true && conversationUser!!.userId != "?" &&
|
|
|
message.user.id.startsWith("users/") &&
|
|
|
message.user.id.substring(ACTOR_LENGTH) != currentConversation?.actorId &&
|
|
|
- currentConversation?.type != ConversationType.ROOM_TYPE_ONE_TO_ONE_CALL ||
|
|
|
+ currentConversation?.type != ConversationEnums.ConversationType.ROOM_TYPE_ONE_TO_ONE_CALL ||
|
|
|
isShowMessageDeletionButton(message) || // delete
|
|
|
ChatMessage.MessageType.REGULAR_TEXT_MESSAGE == message.getCalculateMessageType() || // forward
|
|
|
message.previousMessageId > NO_PREVIOUS_MESSAGE_ID && // mark as unread
|
|
@@ -3361,39 +3379,43 @@ class ChatActivity :
|
|
|
private fun setMessageAsDeleted(message: IMessage?) {
|
|
|
val messageTemp = message as ChatMessage
|
|
|
messageTemp.isDeleted = true
|
|
|
+ messageTemp.message = getString(R.string.message_deleted_by_you)
|
|
|
|
|
|
messageTemp.isOneToOneConversation =
|
|
|
- currentConversation?.type == ConversationType.ROOM_TYPE_ONE_TO_ONE_CALL
|
|
|
+ currentConversation?.type == ConversationEnums.ConversationType.ROOM_TYPE_ONE_TO_ONE_CALL
|
|
|
messageTemp.activeUser = conversationUser
|
|
|
|
|
|
adapter?.update(messageTemp)
|
|
|
}
|
|
|
|
|
|
- private fun setMessageAsEdited(message: IMessage?) {
|
|
|
+ private fun setMessageAsEdited(message: IMessage?, newString: String) {
|
|
|
val messageTemp = message as ChatMessage
|
|
|
messageTemp.lastEditTimestamp = message.lastEditTimestamp
|
|
|
+ messageTemp.message = newString
|
|
|
|
|
|
val index = adapter?.getMessagePositionById(messageTemp.id)!!
|
|
|
if (index > 0) {
|
|
|
val adapterMsg = adapter?.items?.get(index)?.item as ChatMessage
|
|
|
- messageTemp.parentMessage = adapterMsg.parentMessage
|
|
|
+ messageTemp.parentMessageId = adapterMsg.parentMessageId
|
|
|
}
|
|
|
|
|
|
messageTemp.isOneToOneConversation =
|
|
|
- currentConversation?.type == ConversationType.ROOM_TYPE_ONE_TO_ONE_CALL
|
|
|
+ currentConversation?.type == ConversationEnums.ConversationType.ROOM_TYPE_ONE_TO_ONE_CALL
|
|
|
messageTemp.activeUser = conversationUser
|
|
|
|
|
|
adapter?.update(messageTemp)
|
|
|
}
|
|
|
|
|
|
private fun updateAdapterForReaction(message: IMessage?) {
|
|
|
- val messageTemp = message as ChatMessage
|
|
|
+ message?.let {
|
|
|
+ val messageTemp = message as ChatMessage
|
|
|
|
|
|
- messageTemp.isOneToOneConversation =
|
|
|
- currentConversation?.type == ConversationType.ROOM_TYPE_ONE_TO_ONE_CALL
|
|
|
- messageTemp.activeUser = conversationUser
|
|
|
+ messageTemp.isOneToOneConversation =
|
|
|
+ currentConversation?.type == ConversationEnums.ConversationType.ROOM_TYPE_ONE_TO_ONE_CALL
|
|
|
+ messageTemp.activeUser = conversationUser
|
|
|
|
|
|
- adapter?.update(messageTemp)
|
|
|
+ adapter?.update(messageTemp)
|
|
|
+ }
|
|
|
}
|
|
|
|
|
|
fun updateUiToAddReaction(message: ChatMessage, emoji: String) {
|
|
@@ -3428,6 +3450,9 @@ class ChatActivity :
|
|
|
amount = 0
|
|
|
}
|
|
|
message.reactions!![emoji] = amount - 1
|
|
|
+ if (message.reactions!![emoji]!! <= 0) {
|
|
|
+ message.reactions!!.remove(emoji)
|
|
|
+ }
|
|
|
message.reactionsSelf!!.remove(emoji)
|
|
|
adapter?.update(message)
|
|
|
}
|
|
@@ -3529,7 +3554,7 @@ class ChatActivity :
|
|
|
|
|
|
@Subscribe(threadMode = ThreadMode.BACKGROUND)
|
|
|
fun onMessageEvent(userMentionClickEvent: UserMentionClickEvent) {
|
|
|
- if (currentConversation?.type != ConversationType.ROOM_TYPE_ONE_TO_ONE_CALL ||
|
|
|
+ if (currentConversation?.type != ConversationEnums.ConversationType.ROOM_TYPE_ONE_TO_ONE_CALL ||
|
|
|
currentConversation?.name != userMentionClickEvent.userId
|
|
|
) {
|
|
|
var apiVersion = 1
|
|
@@ -3602,13 +3627,21 @@ class ChatActivity :
|
|
|
}
|
|
|
|
|
|
fun jumpToQuotedMessage(parentMessage: ChatMessage) {
|
|
|
+ var foundMessage = false
|
|
|
for (position in 0 until (adapter!!.items.size)) {
|
|
|
val currentItem = adapter?.items?.get(position)?.item
|
|
|
if (currentItem is ChatMessage && currentItem.id == parentMessage.id) {
|
|
|
layoutManager!!.scrollToPosition(position)
|
|
|
+ foundMessage = true
|
|
|
break
|
|
|
}
|
|
|
}
|
|
|
+ if (!foundMessage) {
|
|
|
+ Log.d(TAG, "quoted message with id " + parentMessage.id + " was not found in adapter")
|
|
|
+ // TODO: show better info
|
|
|
+ // TODO: improve handling how this can be avoided. E.g. loading chat until message is reached...
|
|
|
+ Snackbar.make(binding.root, "Message was not found", Snackbar.LENGTH_LONG).show()
|
|
|
+ }
|
|
|
}
|
|
|
|
|
|
override fun joinAudioCall() {
|
|
@@ -3688,6 +3721,7 @@ class ChatActivity :
|
|
|
private const val QUOTED_MESSAGE_IMAGE_MAX_HEIGHT = 96f
|
|
|
private const val MENTION_AUTO_COMPLETE_ELEVATION = 6f
|
|
|
private const val MESSAGE_PULL_LIMIT = 100
|
|
|
+ private const val PAGE_SIZE = 100
|
|
|
private const val INVITE_LENGTH = 6
|
|
|
private const val ACTOR_LENGTH = 6
|
|
|
private const val ANIMATION_DURATION: Long = 750
|
|
@@ -3715,6 +3749,5 @@ class ChatActivity :
|
|
|
private const val CURRENT_AUDIO_POSITION_KEY = "CURRENT_AUDIO_POSITION"
|
|
|
private const val CURRENT_AUDIO_WAS_PLAYING_KEY = "CURRENT_AUDIO_PLAYING"
|
|
|
private const val RESUME_AUDIO_TAG = "RESUME_AUDIO_TAG"
|
|
|
- private const val PAGE_SIZE = 50
|
|
|
}
|
|
|
}
|