session-android/app/src/main/java/org/thoughtcrime/securesms/conversation/v2/messages/VisibleMessageView.kt

125 lines
6.5 KiB
Kotlin
Raw Normal View History

package org.thoughtcrime.securesms.conversation.v2.messages
import android.content.Context
import android.util.AttributeSet
2021-06-01 06:28:14 +02:00
import android.view.Gravity
2021-05-31 06:29:11 +02:00
import android.view.LayoutInflater
import android.view.View
2021-06-01 06:28:14 +02:00
import android.view.ViewGroup
import android.widget.LinearLayout
2021-06-07 08:06:37 +02:00
import androidx.core.view.isVisible
import kotlinx.android.synthetic.main.view_visible_message.view.*
2021-05-31 06:29:11 +02:00
import network.loki.messenger.R
import org.session.libsession.messaging.contacts.Contact.ContactContext
2021-06-07 07:37:21 +02:00
import org.session.libsession.utilities.ViewUtil
import org.thoughtcrime.securesms.database.DatabaseFactory
import org.thoughtcrime.securesms.database.model.MessageRecord
2021-06-07 07:37:21 +02:00
import org.thoughtcrime.securesms.util.DateUtils
2021-06-07 08:06:37 +02:00
import java.util.*
2021-06-07 07:37:21 +02:00
import kotlin.math.roundToInt
class VisibleMessageView : LinearLayout {
// region Lifecycle
constructor(context: Context) : super(context) {
setUpViewHierarchy()
}
constructor(context: Context, attrs: AttributeSet) : super(context, attrs) {
setUpViewHierarchy()
}
constructor(context: Context, attrs: AttributeSet, defStyleAttr: Int) : super(context, attrs, defStyleAttr) {
setUpViewHierarchy()
}
private fun setUpViewHierarchy() {
LayoutInflater.from(context).inflate(R.layout.view_visible_message, this)
2021-06-01 06:28:14 +02:00
layoutParams = ViewGroup.LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT, ViewGroup.LayoutParams.WRAP_CONTENT)
}
// endregion
// region Updating
2021-06-07 07:37:21 +02:00
fun bind(message: MessageRecord, previous: MessageRecord?, next: MessageRecord?) {
val sender = message.individualRecipient
val senderSessionID = sender.address.serialize()
val threadID = message.threadId
val threadDB = DatabaseFactory.getThreadDatabase(context)
val thread = threadDB.getRecipientForThreadId(threadID)
val contactDB = DatabaseFactory.getSessionContactDatabase(context)
val isGroupThread = (thread?.isGroupRecipient == true)
2021-06-01 06:28:14 +02:00
// Show profile picture and sender name if this is a group thread AND
// the message is incoming
if (isGroupThread && !message.isOutgoing) {
profilePictureContainer.visibility = View.VISIBLE
profilePictureView.publicKey = senderSessionID
// TODO: Set glide on the profile picture view and update it
// TODO: Show crown if this is an open group and the user is a moderator; otherwise hide it
senderNameTextView.visibility = View.VISIBLE
val context = if (thread?.isOpenGroupRecipient == true) ContactContext.OPEN_GROUP else ContactContext.REGULAR
senderNameTextView.text = contactDB.getContactWithSessionID(senderSessionID)?.displayName(context) ?: senderSessionID
} else {
profilePictureContainer.visibility = View.GONE
senderNameTextView.visibility = View.GONE
}
2021-06-01 06:38:52 +02:00
// Date break
2021-06-07 08:06:37 +02:00
val showDateBreak = (previous == null || !DateUtils.isSameDay(message.timestamp, previous.timestamp))
dateBreakTextView.isVisible = showDateBreak
dateBreakTextView.text = if (showDateBreak) DateUtils.getRelativeDate(context, Locale.getDefault(), message.timestamp) else ""
2021-06-08 06:06:16 +02:00
// Timestamp
messageTimestampTextView.text = DateUtils.getExtendedRelativeTimeSpanString(context, Locale.getDefault(), message.timestamp)
2021-06-01 06:28:14 +02:00
// Margins
2021-06-08 06:06:16 +02:00
val messageContentContainerLayoutParams = messageContentContainer.layoutParams as LinearLayout.LayoutParams
2021-06-01 07:43:37 +02:00
if (isGroupThread) {
2021-06-08 06:06:16 +02:00
messageContentContainerLayoutParams.leftMargin = if (message.isOutgoing) resources.getDimension(R.dimen.very_large_spacing).toInt() else 0
2021-06-01 07:43:37 +02:00
} else {
2021-06-08 06:06:16 +02:00
messageContentContainerLayoutParams.leftMargin = if (message.isOutgoing) resources.getDimension(R.dimen.very_large_spacing).toInt()
2021-06-01 07:43:37 +02:00
else resources.getDimension(R.dimen.medium_spacing).toInt()
}
2021-06-08 06:06:16 +02:00
messageContentContainerLayoutParams.rightMargin = if (message.isOutgoing) resources.getDimension(R.dimen.medium_spacing).toInt()
2021-06-01 07:43:37 +02:00
else resources.getDimension(R.dimen.very_large_spacing).toInt()
2021-06-08 06:06:16 +02:00
messageContentContainer.layoutParams = messageContentContainerLayoutParams
2021-06-07 07:37:21 +02:00
// Set inter-message spacing
2021-06-07 07:48:22 +02:00
val isStartOfMessageCluster = isStartOfMessageCluster(message, previous, isGroupThread)
val isEndOfMessageCluster = isEndOfMessageCluster(message, next, isGroupThread)
setMessageSpacing(isStartOfMessageCluster, isEndOfMessageCluster)
2021-06-01 06:28:14 +02:00
// Gravity
2021-06-01 06:38:52 +02:00
val gravity = if (message.isOutgoing) Gravity.RIGHT else Gravity.LEFT
mainContainer.gravity = gravity or Gravity.BOTTOM
2021-06-01 05:26:57 +02:00
// Populate content view
2021-06-07 07:48:22 +02:00
messageContentView.bind(message, isStartOfMessageCluster, isEndOfMessageCluster)
}
2021-06-07 07:48:22 +02:00
private fun setMessageSpacing(isStartOfMessageCluster: Boolean, isEndOfMessageCluster: Boolean) {
val topPadding = if (isStartOfMessageCluster) R.dimen.conversation_vertical_message_spacing_default else R.dimen.conversation_vertical_message_spacing_collapse
2021-06-07 07:37:21 +02:00
ViewUtil.setPaddingTop(this, resources.getDimension(topPadding).roundToInt())
2021-06-07 07:48:22 +02:00
val bottomPadding = if (isEndOfMessageCluster) R.dimen.conversation_vertical_message_spacing_default else R.dimen.conversation_vertical_message_spacing_collapse
2021-06-07 07:37:21 +02:00
ViewUtil.setPaddingBottom(this, resources.getDimension(bottomPadding).roundToInt())
}
private fun isStartOfMessageCluster(current: MessageRecord, previous: MessageRecord?, isGroupThread: Boolean): Boolean {
return if (isGroupThread) {
previous == null || previous.isUpdate || !DateUtils.isSameDay(current.timestamp, previous.timestamp)
|| current.recipient.address != previous.recipient.address
} else {
previous == null || previous.isUpdate || !DateUtils.isSameDay(current.timestamp, previous.timestamp)
|| current.isOutgoing != previous.isOutgoing
}
}
private fun isEndOfMessageCluster(current: MessageRecord, next: MessageRecord?, isGroupThread: Boolean): Boolean {
return if (isGroupThread) {
next == null || next.isUpdate || !DateUtils.isSameDay(current.timestamp, next.timestamp)
|| current.recipient.address != next.recipient.address
} else {
next == null || next.isUpdate || !DateUtils.isSameDay(current.timestamp, next.timestamp)
|| current.isOutgoing != next.isOutgoing
}
}
fun recycle() {
profilePictureView.recycle()
messageContentView.recycle()
}
// endregion
}