feat: 新增聊天界面左上角未读消息数量显示

This commit is contained in:
糕小菜 2024-12-17 23:10:07 +08:00
parent 8cd3c39b41
commit 0785dc60d5
11 changed files with 84 additions and 53 deletions

View File

@ -2,19 +2,7 @@
<project version="4"> <project version="4">
<component name="GitCommitMessageStorage"> <component name="GitCommitMessageStorage">
<option name="messageStorage"> <option name="messageStorage">
<MessageStorage> <MessageStorage />
<option name="commitTemplate">
<CommitTemplate>
<option name="body" value="" />
<option name="changes" value="" />
<option name="closes" value="" />
<option name="scope" value="" />
<option name="skipCi" value="" />
<option name="subject" value="" />
<option name="type" value="feat" />
</CommitTemplate>
</option>
</MessageStorage>
</option> </option>
</component> </component>
</project> </project>

View File

@ -30,6 +30,7 @@ object ConversationManager {
private val conversationMap: MutableMap<String, Conversation> = mutableMapOf() private val conversationMap: MutableMap<String, Conversation> = mutableMapOf()
private var unreadCount = 0 private var unreadCount = 0
val unReadMsgCount get() = unreadCount
init { init {
loadConversations() loadConversations()
@ -49,7 +50,8 @@ object ConversationManager {
} }
fun handleMessages(messages: Messages) { fun handleMessages(messages: Messages) {
if (messages.senderId != getUsername() || messages.talkerId != getCurrentContactId()) { val talkerId = getCurrentContactId()
if (messages.senderId != getUsername() && messages.talkerId != talkerId) {
unreadCount++ unreadCount++
} }
val updatedConversation = updateConversation(messages) val updatedConversation = updateConversation(messages)

View File

@ -1,5 +1,6 @@
package com.kaixed.kchat.manager package com.kaixed.kchat.manager
import com.kaixed.kchat.data.DataBase
import com.kaixed.kchat.data.LocalDatabase import com.kaixed.kchat.data.LocalDatabase
import com.kaixed.kchat.data.LocalDatabase.getAllHistoryMessages import com.kaixed.kchat.data.LocalDatabase.getAllHistoryMessages
import com.kaixed.kchat.data.LocalDatabase.getMessagesWithContact import com.kaixed.kchat.data.LocalDatabase.getMessagesWithContact
@ -25,6 +26,14 @@ object MessagesManager {
private var loading = false private var loading = false
private var tempIndex: Long = 0 private var tempIndex: Long = 0
fun deleteMessage(msgLocalId: Long) {
DataBase.messagesBox.remove(msgLocalId)
val msg = _messages.value.toMutableList().apply {
removeIf { it.msgLocalId == msgLocalId }
}
_messages.value = msg
}
fun queryHistory(msgLocalId: Long): Int { fun queryHistory(msgLocalId: Long): Int {
_messages.value = emptyList() _messages.value = emptyList()
val msg = getAllHistoryMessages(contactId, msgLocalId) val msg = getAllHistoryMessages(contactId, msgLocalId)
@ -49,6 +58,7 @@ object MessagesManager {
} }
fun receiveMessage(messages: Messages) { fun receiveMessage(messages: Messages) {
if (messages.talkerId != contactId) return
if (_messages.value.first() == messages) return if (_messages.value.first() == messages) return
_messages.value = _messages.value.toMutableList().apply { _messages.value = _messages.value.toMutableList().apply {
add(0, messages) add(0, messages)

View File

@ -6,8 +6,7 @@ package com.kaixed.kchat.network
*/ */
object ApiCall { object ApiCall {
suspend fun <T> apiCall( suspend fun <T> apiCall(
apiCall: suspend () -> ApiResponse<T>, apiCall: suspend () -> ApiResponse<T>, errorMessage: String = "操作失败"
errorMessage: String = "操作失败"
): Result<T?> { ): Result<T?> {
return try { return try {
val response = apiCall() val response = apiCall()

View File

@ -28,10 +28,12 @@ import androidx.recyclerview.widget.GridLayoutManager
import androidx.recyclerview.widget.LinearLayoutManager import androidx.recyclerview.widget.LinearLayoutManager
import androidx.recyclerview.widget.RecyclerView import androidx.recyclerview.widget.RecyclerView
import com.kaixed.kchat.R import com.kaixed.kchat.R
import com.kaixed.kchat.data.event.UnreadEvent
import com.kaixed.kchat.data.local.box.ObjectBox.getBoxStore import com.kaixed.kchat.data.local.box.ObjectBox.getBoxStore
import com.kaixed.kchat.data.local.entity.Messages import com.kaixed.kchat.data.local.entity.Messages
import com.kaixed.kchat.data.model.FunctionItem import com.kaixed.kchat.data.model.FunctionItem
import com.kaixed.kchat.databinding.ActivityChatBinding import com.kaixed.kchat.databinding.ActivityChatBinding
import com.kaixed.kchat.manager.ConversationManager
import com.kaixed.kchat.manager.MessagesManager import com.kaixed.kchat.manager.MessagesManager
import com.kaixed.kchat.processor.MessageProcessor import com.kaixed.kchat.processor.MessageProcessor
import com.kaixed.kchat.service.WebSocketService import com.kaixed.kchat.service.WebSocketService
@ -59,6 +61,9 @@ import com.tencent.mmkv.MMKV
import io.objectbox.Box import io.objectbox.Box
import io.objectbox.kotlin.boxFor import io.objectbox.kotlin.boxFor
import kotlinx.coroutines.launch import kotlinx.coroutines.launch
import org.greenrobot.eventbus.EventBus
import org.greenrobot.eventbus.Subscribe
import org.greenrobot.eventbus.ThreadMode
import org.json.JSONObject import org.json.JSONObject
import java.io.File import java.io.File
@ -108,6 +113,7 @@ class ChatActivity : BaseActivity<ActivityChatBinding>(), OnItemClickListener,
override fun onCreate(savedInstanceState: Bundle?) { override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState) super.onCreate(savedInstanceState)
enableEdgeToEdge() enableEdgeToEdge()
EventBus.getDefault().register(this)
firstLoadData() firstLoadData()
initView() initView()
setListener() setListener()
@ -367,6 +373,7 @@ class ChatActivity : BaseActivity<ActivityChatBinding>(), OnItemClickListener,
} }
private fun initView() { private fun initView() {
binding.ctb.setUnReadCount(ConversationManager.unReadMsgCount)
setupFunctionPanel() setupFunctionPanel()
setRecycleView() setRecycleView()
contactNickname?.let { contactNickname?.let {
@ -461,8 +468,15 @@ class ChatActivity : BaseActivity<ActivityChatBinding>(), OnItemClickListener,
) )
} }
@Subscribe(threadMode = ThreadMode.MAIN)
fun onMessageEvent(unreadEvent: UnreadEvent) {
val unreadCount = unreadEvent.unreadCount
binding.ctb.setUnReadCount(unreadCount)
}
override fun onDestroy() { override fun onDestroy() {
super.onDestroy() super.onDestroy()
EventBus.getDefault().unregister(this)
MessagesManager.resetMessages() MessagesManager.resetMessages()
mmkv.putString(CURRENT_CONTACT_ID, "") mmkv.putString(CURRENT_CONTACT_ID, "")
if (bound) { if (bound) {

View File

@ -54,9 +54,7 @@ class UpdatePasswordActivity : BaseActivity<ActivityUpdatePasswordBinding>() {
} else { } else {
loadingDialog.showLoading(supportFragmentManager) loadingDialog.showLoading(supportFragmentManager)
val updatePassword = UpdatePasswordRequest( val updatePassword = UpdatePasswordRequest(
ConstantsUtil.getUsername(), ConstantsUtil.getUsername(), oldPassword, newPassword
oldPassword,
newPassword
) )
userViewModel.updatePassword(updatePassword) userViewModel.updatePassword(updatePassword)
} }
@ -93,9 +91,8 @@ class UpdatePasswordActivity : BaseActivity<ActivityUpdatePasswordBinding>() {
} }
override fun afterTextChanged(s: Editable?) { override fun afterTextChanged(s: Editable?) {
val enable = binding.cetNewPassword.text?.length != 0 && val enable =
binding.cetOldPassword.text?.length != 0 && binding.cetNewPassword.text?.length != 0 && binding.cetOldPassword.text?.length != 0 && binding.cetConfirmPassword.text?.length != 0
binding.cetConfirmPassword.text?.length != 0
binding.titleBar.setBtnEnable(enable) binding.titleBar.setBtnEnable(enable)
} }
} }

View File

@ -1,7 +1,6 @@
package com.kaixed.kchat.ui.adapter package com.kaixed.kchat.ui.adapter
import android.content.Context import android.content.Context
import android.util.Log
import android.view.LayoutInflater import android.view.LayoutInflater
import android.view.ViewGroup import android.view.ViewGroup
import android.widget.ImageView import android.widget.ImageView
@ -19,10 +18,10 @@ import com.kaixed.kchat.data.local.entity.Messages
import com.kaixed.kchat.databinding.ChatRecycleItemCustomNormalBinding import com.kaixed.kchat.databinding.ChatRecycleItemCustomNormalBinding
import com.kaixed.kchat.databinding.ChatRecycleItemImageNormalBinding import com.kaixed.kchat.databinding.ChatRecycleItemImageNormalBinding
import com.kaixed.kchat.databinding.ChatRecycleItemTipBinding import com.kaixed.kchat.databinding.ChatRecycleItemTipBinding
import com.kaixed.kchat.manager.MessagesManager
import com.kaixed.kchat.utils.ConstantsUtil import com.kaixed.kchat.utils.ConstantsUtil
import com.kaixed.kchat.utils.ConstantsUtil.getUsername import com.kaixed.kchat.utils.ConstantsUtil.getUsername
import com.kaixed.kchat.utils.PopWindowUtil.showPopupWindow import com.kaixed.kchat.utils.PopWindowUtil.showPopupWindow
import com.kaixed.kchat.utils.TextUtil
import com.kaixed.kchat.utils.TextUtil.extractDimensionsAndPrefix import com.kaixed.kchat.utils.TextUtil.extractDimensionsAndPrefix
import com.kaixed.kchat.utils.ViewUtil.changeTimerVisibility import com.kaixed.kchat.utils.ViewUtil.changeTimerVisibility
import com.kaixed.kchat.utils.ViewUtil.setViewVisibility import com.kaixed.kchat.utils.ViewUtil.setViewVisibility
@ -57,22 +56,25 @@ class ChatAdapter(
return when (viewType) { return when (viewType) {
CUSTOM -> { CUSTOM -> {
CustomViewHolder( CustomViewHolder(
ChatRecycleItemCustomNormalBinding ChatRecycleItemCustomNormalBinding.inflate(
.inflate(LayoutInflater.from(parent.context), parent, false) LayoutInflater.from(parent.context), parent, false
)
) )
} }
IMAGE -> { IMAGE -> {
ImageViewHolder( ImageViewHolder(
ChatRecycleItemImageNormalBinding ChatRecycleItemImageNormalBinding.inflate(
.inflate(LayoutInflater.from(parent.context), parent, false) LayoutInflater.from(parent.context), parent, false
)
) )
} }
else -> { else -> {
TipViewHolder( TipViewHolder(
ChatRecycleItemTipBinding ChatRecycleItemTipBinding.inflate(
.inflate(LayoutInflater.from(parent.context), parent, false) LayoutInflater.from(parent.context), parent, false
)
) )
} }
} }
@ -108,30 +110,18 @@ class ChatAdapter(
} }
contentView?.setOnLongClickListener { contentView?.setOnLongClickListener {
Log.d("haha", "长按了:${message.content} position: $position")
val popupWindow = showPopupWindow(context, it, message.senderId == getUsername()) val popupWindow = showPopupWindow(context, it, message.senderId == getUsername())
val deleteButton = popupWindow.contentView.findViewById<TextView>(R.id.tv_delete) val deleteButton = popupWindow.contentView.findViewById<TextView>(R.id.tv_delete)
deleteButton.setOnClickListener { deleteButton.setOnClickListener {
deleteMessage(position, message) deleteMessage(message)
Log.d("haha", "长按并删除了了:${message.content} position: $position")
popupWindow.dismiss() popupWindow.dismiss()
} }
true true
} }
} }
private fun deleteMessage(position: Int, message: Messages) { private fun deleteMessage(message: Messages) {
// 从数据库删除 MessagesManager.deleteMessage(message.msgLocalId)
// val messagesBox: Box<Messages> = getBox(Messages::class.java)
// messagesBox.remove(message.msgLocalId)
//
// // 更新数据源并通知 RecyclerView
// messages.removeAt(position)
// notifyItemRemoved(position)
//
// if (position != messages.size) { // 如果移除的是最后一个,忽略
// notifyItemRangeChanged(position, messages.size - position);
// }
} }
interface HasTimer { interface HasTimer {
@ -162,8 +152,7 @@ class ChatAdapter(
contentId = binding.tvMsgContent.id, contentId = binding.tvMsgContent.id,
contentMineId = binding.tvMsgContentMine.id contentMineId = binding.tvMsgContentMine.id
) )
val avatarUrl = val avatarUrl = if (sender) ConstantsUtil.getAvatarUrl() else message.avatarUrl
if (sender) ConstantsUtil.getAvatarUrl() else message.avatarUrl
Glide.with(binding.root.context).load(avatarUrl) Glide.with(binding.root.context).load(avatarUrl)
.into(if (sender) binding.ifvAvatarMine else binding.ifvAvatar) .into(if (sender) binding.ifvAvatarMine else binding.ifvAvatar)
@ -182,8 +171,7 @@ class ChatAdapter(
RecyclerView.ViewHolder(binding.root), HasTimer { RecyclerView.ViewHolder(binding.root), HasTimer {
fun bindData(message: Messages) { fun bindData(message: Messages) {
val sender = message.senderId == getUsername() val sender = message.senderId == getUsername()
val avatarUrl = val avatarUrl = if (sender) ConstantsUtil.getAvatarUrl() else message.avatarUrl
if (sender) ConstantsUtil.getAvatarUrl() else message.avatarUrl
Glide.with(binding.root.context).load(avatarUrl) Glide.with(binding.root.context).load(avatarUrl)
.into(if (sender) binding.ifvAvatarMine else binding.ifvAvatar) .into(if (sender) binding.ifvAvatarMine else binding.ifvAvatar)
@ -209,8 +197,8 @@ class ChatAdapter(
this.width = width this.width = width
} }
} }
Glide.with(binding.root.context).load(url) Glide.with(binding.root.context).load(url).placeholder(R.drawable.image_loading)
.placeholder(R.drawable.image_loading).into(imageView) .into(imageView)
} ?: run { } ?: run {
Glide.with(binding.root.context).load(message.content) Glide.with(binding.root.context).load(message.content)
.placeholder(R.drawable.image_loading).into(imageView) .placeholder(R.drawable.image_loading).into(imageView)

View File

@ -67,8 +67,9 @@ class CustomTitleBar @JvmOverloads constructor(
} }
} }
fun setOnBackClickListener(listener: OnClickListener?) { fun setUnReadCount(count: Int) {
binding.ivBack.setOnClickListener(listener) binding.rlUnreadCount.visibility = if (count > 0) View.VISIBLE else View.INVISIBLE
binding.tvUnreadCount.text = count.toString()
} }
fun setOnSettingClickListener(listener: OnClickListener?) { fun setOnSettingClickListener(listener: OnClickListener?) {

View File

@ -36,6 +36,7 @@ class UserViewModel : ViewModel() {
viewModelScope.launch { viewModelScope.launch {
val result = userProfileRepo.updatePassword(updatePasswordRequest) val result = userProfileRepo.updatePassword(updatePasswordRequest)
_updatePasswordResult.postValue(result) _updatePasswordResult.postValue(result)
_updatePasswordResult.value = result
} }
} }

View File

@ -0,0 +1,8 @@
<?xml version="1.0" encoding="utf-8"?>
<shape xmlns:android="http://schemas.android.com/apk/res/android">
<solid android:color="#D5D5D5" />
<corners android:radius="50dp" />
<size
android:width="10dp"
android:height="10dp" />
</shape>

View File

@ -14,6 +14,29 @@
app:layout_constraintStart_toStartOf="parent" app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toTopOf="parent" /> app:layout_constraintTop_toTopOf="parent" />
<RelativeLayout
android:id="@+id/rl_unread_count"
android:layout_width="20dp"
android:layout_height="20dp"
android:layout_marginStart="3dp"
android:background="@drawable/icon_gray_dot"
android:visibility="invisible"
app:layout_constraintBottom_toBottomOf="@id/iv_back"
app:layout_constraintStart_toEndOf="@id/iv_back"
app:layout_constraintTop_toTopOf="@id/iv_back">
<TextView
android:id="@+id/tv_unread_count"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_centerInParent="true"
android:gravity="center"
android:text="9"
android:textColor="@color/black"
android:textSize="12sp" />
</RelativeLayout>
<TextView <TextView
android:id="@+id/tv_title_name" android:id="@+id/tv_title_name"
android:layout_width="wrap_content" android:layout_width="wrap_content"