|
|
|
@ -13,9 +13,12 @@ import androidx.compose.foundation.layout.Box
|
|
|
|
|
import androidx.compose.foundation.layout.Column
|
|
|
|
|
import androidx.compose.foundation.layout.ExperimentalLayoutApi
|
|
|
|
|
import androidx.compose.foundation.layout.FlowRow
|
|
|
|
|
import androidx.compose.foundation.layout.IntrinsicSize
|
|
|
|
|
import androidx.compose.foundation.layout.Row
|
|
|
|
|
import androidx.compose.foundation.layout.aspectRatio
|
|
|
|
|
import androidx.compose.foundation.layout.padding
|
|
|
|
|
import androidx.compose.foundation.layout.width
|
|
|
|
|
import androidx.compose.foundation.layout.widthIn
|
|
|
|
|
import androidx.compose.foundation.pager.HorizontalPager
|
|
|
|
|
import androidx.compose.foundation.pager.PagerState
|
|
|
|
|
import androidx.compose.foundation.pager.rememberPagerState
|
|
|
|
@ -31,9 +34,11 @@ import androidx.compose.runtime.getValue
|
|
|
|
|
import androidx.compose.runtime.livedata.observeAsState
|
|
|
|
|
import androidx.compose.ui.Alignment
|
|
|
|
|
import androidx.compose.ui.Modifier
|
|
|
|
|
import androidx.compose.ui.graphics.Color
|
|
|
|
|
import androidx.compose.ui.layout.ContentScale
|
|
|
|
|
import androidx.compose.ui.platform.ComposeView
|
|
|
|
|
import androidx.compose.ui.res.painterResource
|
|
|
|
|
import androidx.compose.ui.res.stringResource
|
|
|
|
|
import androidx.compose.ui.text.TextStyle
|
|
|
|
|
import androidx.compose.ui.text.font.FontFamily
|
|
|
|
|
import androidx.compose.ui.text.font.FontWeight
|
|
|
|
@ -55,8 +60,8 @@ import org.session.libsession.messaging.sending_receiving.attachments.DatabaseAt
|
|
|
|
|
import org.thoughtcrime.securesms.MediaPreviewActivity
|
|
|
|
|
import org.thoughtcrime.securesms.PassphraseRequiredActionBarActivity
|
|
|
|
|
import org.thoughtcrime.securesms.database.Storage
|
|
|
|
|
import org.thoughtcrime.securesms.database.ThreadDatabase
|
|
|
|
|
import org.thoughtcrime.securesms.dependencies.DatabaseComponent
|
|
|
|
|
import org.thoughtcrime.securesms.mms.ImageSlide
|
|
|
|
|
import org.thoughtcrime.securesms.mms.Slide
|
|
|
|
|
import org.thoughtcrime.securesms.ui.AppTheme
|
|
|
|
|
import org.thoughtcrime.securesms.ui.Avatar
|
|
|
|
@ -68,6 +73,7 @@ import org.thoughtcrime.securesms.ui.CellWithPaddingAndMargin
|
|
|
|
|
import org.thoughtcrime.securesms.ui.Divider
|
|
|
|
|
import org.thoughtcrime.securesms.ui.HorizontalPagerIndicator
|
|
|
|
|
import org.thoughtcrime.securesms.ui.ItemButton
|
|
|
|
|
import org.thoughtcrime.securesms.ui.Theme
|
|
|
|
|
import org.thoughtcrime.securesms.ui.blackAlpha40
|
|
|
|
|
import org.thoughtcrime.securesms.ui.colorDestructive
|
|
|
|
|
import org.thoughtcrime.securesms.ui.destructiveButtonColors
|
|
|
|
@ -118,27 +124,32 @@ class MessageDetailActivity : PassphraseRequiredActionBarActivity() {
|
|
|
|
|
@Composable
|
|
|
|
|
private fun MessageDetailsScreen() {
|
|
|
|
|
val details by viewModel.details.observeAsState(MessageDetails())
|
|
|
|
|
MessageDetails(
|
|
|
|
|
details,
|
|
|
|
|
onReply = { setResultAndFinish(ON_REPLY) },
|
|
|
|
|
onResend = { setResultAndFinish(ON_RESEND) },
|
|
|
|
|
onDelete = { setResultAndFinish(ON_DELETE) },
|
|
|
|
|
onClickImage = { slide ->
|
|
|
|
|
// only open to downloaded images
|
|
|
|
|
if (slide.transferState == AttachmentTransferProgress.TRANSFER_PROGRESS_FAILED) {
|
|
|
|
|
// Restart download here (on IO thread)
|
|
|
|
|
(slide.asAttachment() as? DatabaseAttachment)?.let { attachment ->
|
|
|
|
|
onAttachmentNeedsDownload(attachment.attachmentId.rowId, details.mmsRecord!!.getId())
|
|
|
|
|
val threadDb = DatabaseComponent.get(this@MessageDetailActivity).threadDatabase()
|
|
|
|
|
AppTheme {
|
|
|
|
|
MessageDetails(
|
|
|
|
|
threadDb = threadDb,
|
|
|
|
|
messageDetails = details,
|
|
|
|
|
onReply = { setResultAndFinish(ON_REPLY) },
|
|
|
|
|
onResend = { setResultAndFinish(ON_RESEND) },
|
|
|
|
|
onDelete = { setResultAndFinish(ON_DELETE) },
|
|
|
|
|
onClickImage = { slide ->
|
|
|
|
|
// only open to downloaded images
|
|
|
|
|
if (slide.transferState == AttachmentTransferProgress.TRANSFER_PROGRESS_FAILED) {
|
|
|
|
|
// Restart download here (on IO thread)
|
|
|
|
|
(slide.asAttachment() as? DatabaseAttachment)?.let { attachment ->
|
|
|
|
|
onAttachmentNeedsDownload(attachment.attachmentId.rowId, details.mmsRecord!!.getId())
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (!slide.isInProgress) MediaPreviewActivity.getPreviewIntent(
|
|
|
|
|
this,
|
|
|
|
|
slide,
|
|
|
|
|
details.mmsRecord,
|
|
|
|
|
DatabaseComponent.get(this).threadDatabase().getRecipientForThreadId(details.mmsRecord!!.threadId),
|
|
|
|
|
).let(::startActivity)
|
|
|
|
|
}
|
|
|
|
|
)
|
|
|
|
|
if (!slide.isInProgress) MediaPreviewActivity.getPreviewIntent(
|
|
|
|
|
this,
|
|
|
|
|
slide,
|
|
|
|
|
details.mmsRecord,
|
|
|
|
|
threadDb.getRecipientForThreadId(details.mmsRecord!!.threadId),
|
|
|
|
|
).let(::startActivity)
|
|
|
|
|
},
|
|
|
|
|
onAttachmentNeedsDownload = ::onAttachmentNeedsDownload,
|
|
|
|
|
)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private fun setResultAndFinish(code: Int) {
|
|
|
|
@ -149,254 +160,276 @@ class MessageDetailActivity : PassphraseRequiredActionBarActivity() {
|
|
|
|
|
finish()
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@Preview
|
|
|
|
|
@Composable
|
|
|
|
|
fun PreviewMessageDetails() {
|
|
|
|
|
private fun onAttachmentNeedsDownload(attachmentId: Long, mmsId: Long) {
|
|
|
|
|
lifecycleScope.launch(Dispatchers.IO) {
|
|
|
|
|
JobQueue.shared.add(AttachmentDownloadJob(attachmentId, mmsId))
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@Composable
|
|
|
|
|
fun PreviewMessageDetails() {
|
|
|
|
|
AppTheme {
|
|
|
|
|
MessageDetails(
|
|
|
|
|
MessageDetails(
|
|
|
|
|
messageDetails = MessageDetails(
|
|
|
|
|
attachments = listOf(),
|
|
|
|
|
sent = TitledText("Sent:", "6:12 AM Tue, 09/08/2022"),
|
|
|
|
|
received = TitledText("Received:", "6:12 AM Tue, 09/08/2022"),
|
|
|
|
|
error = TitledText("Error:", "Message failed to send"),
|
|
|
|
|
senderInfo = TitledText("Connor", "d4f1g54sdf5g1d5f4g65ds4564df65f4g65d54gdfsg"),
|
|
|
|
|
senderInfo = TitledText("Connor", "d4f1g54sdf5g1d5f4g65ds4564df65f4g65d54"),
|
|
|
|
|
)
|
|
|
|
|
)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private fun onAttachmentNeedsDownload(attachmentId: Long, mmsId: Long) {
|
|
|
|
|
lifecycleScope.launch(Dispatchers.IO) {
|
|
|
|
|
JobQueue.shared.add(AttachmentDownloadJob(attachmentId, mmsId))
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@SuppressLint("ClickableViewAccessibility")
|
|
|
|
|
@Composable
|
|
|
|
|
fun MessageDetails(
|
|
|
|
|
messageDetails: MessageDetails,
|
|
|
|
|
onReply: () -> Unit = {},
|
|
|
|
|
onResend: () -> Unit = {},
|
|
|
|
|
onDelete: () -> Unit = {},
|
|
|
|
|
onClickImage: (Slide) -> Unit = {},
|
|
|
|
|
@SuppressLint("ClickableViewAccessibility")
|
|
|
|
|
@Composable
|
|
|
|
|
fun MessageDetails(
|
|
|
|
|
threadDb: ThreadDatabase? = null,
|
|
|
|
|
messageDetails: MessageDetails,
|
|
|
|
|
onReply: () -> Unit = {},
|
|
|
|
|
onResend: (() -> Unit)? = null,
|
|
|
|
|
onDelete: () -> Unit = {},
|
|
|
|
|
onClickImage: (Slide) -> Unit = {},
|
|
|
|
|
onAttachmentNeedsDownload: (Long, Long) -> Unit = { _, _ -> }
|
|
|
|
|
) {
|
|
|
|
|
Column(
|
|
|
|
|
modifier = Modifier
|
|
|
|
|
.verticalScroll(rememberScrollState())
|
|
|
|
|
.padding(vertical = 16.dp),
|
|
|
|
|
verticalArrangement = Arrangement.spacedBy(16.dp)
|
|
|
|
|
) {
|
|
|
|
|
AppTheme {
|
|
|
|
|
Column(
|
|
|
|
|
modifier = Modifier.verticalScroll(rememberScrollState()),
|
|
|
|
|
verticalArrangement = Arrangement.spacedBy(16.dp)
|
|
|
|
|
) {
|
|
|
|
|
messageDetails.record?.let { message ->
|
|
|
|
|
AndroidView(
|
|
|
|
|
modifier = Modifier.padding(horizontal = 32.dp),
|
|
|
|
|
factory = {
|
|
|
|
|
ViewVisibleMessageContentBinding.inflate(LayoutInflater.from(it)).mainContainerConstraint.apply {
|
|
|
|
|
bind(
|
|
|
|
|
message,
|
|
|
|
|
thread = DatabaseComponent.get(this@MessageDetailActivity).threadDatabase().getRecipientForThreadId(message.threadId)!!,
|
|
|
|
|
onAttachmentNeedsDownload = ::onAttachmentNeedsDownload,
|
|
|
|
|
suppressThumbnails = true
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
setOnTouchListener { _, event ->
|
|
|
|
|
if (event.actionMasked == ACTION_UP) onContentClick(event)
|
|
|
|
|
true
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
messageDetails.record?.let { message ->
|
|
|
|
|
AndroidView(
|
|
|
|
|
modifier = Modifier.padding(horizontal = 32.dp),
|
|
|
|
|
factory = {
|
|
|
|
|
ViewVisibleMessageContentBinding.inflate(LayoutInflater.from(it)).mainContainerConstraint.apply {
|
|
|
|
|
bind(
|
|
|
|
|
message,
|
|
|
|
|
thread = threadDb?.getRecipientForThreadId(message.threadId)!!,
|
|
|
|
|
onAttachmentNeedsDownload = onAttachmentNeedsDownload,
|
|
|
|
|
suppressThumbnails = true
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
setOnTouchListener { _, event ->
|
|
|
|
|
if (event.actionMasked == ACTION_UP) onContentClick(event)
|
|
|
|
|
true
|
|
|
|
|
}
|
|
|
|
|
)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
Attachments(messageDetails.attachments) { onClickImage(it) }
|
|
|
|
|
MetadataCell(messageDetails)
|
|
|
|
|
Buttons(
|
|
|
|
|
messageDetails.error != null,
|
|
|
|
|
onReply,
|
|
|
|
|
onResend,
|
|
|
|
|
onDelete,
|
|
|
|
|
)
|
|
|
|
|
}
|
|
|
|
|
)
|
|
|
|
|
}
|
|
|
|
|
Carousel(messageDetails.attachments) { onClickImage(it) }
|
|
|
|
|
MetadataCell(messageDetails)
|
|
|
|
|
Buttons(
|
|
|
|
|
onReply,
|
|
|
|
|
onResend,
|
|
|
|
|
onDelete,
|
|
|
|
|
)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@Composable
|
|
|
|
|
fun MetadataCell(
|
|
|
|
|
messageDetails: MessageDetails,
|
|
|
|
|
) {
|
|
|
|
|
messageDetails.apply {
|
|
|
|
|
if (sent != null || received != null || senderInfo != null) CellWithPaddingAndMargin {
|
|
|
|
|
Column(verticalArrangement = Arrangement.spacedBy(16.dp)) {
|
|
|
|
|
sent?.let { TitledText(it) }
|
|
|
|
|
received?.let { TitledText(it) }
|
|
|
|
|
error?.let { TitledErrorText(it) }
|
|
|
|
|
senderInfo?.let {
|
|
|
|
|
TitledView("From:") {
|
|
|
|
|
Row {
|
|
|
|
|
sender?.let { Avatar(it) }
|
|
|
|
|
TitledMonospaceText(it)
|
|
|
|
|
}
|
|
|
|
|
@Composable
|
|
|
|
|
fun MetadataCell(
|
|
|
|
|
messageDetails: MessageDetails,
|
|
|
|
|
) {
|
|
|
|
|
messageDetails.apply {
|
|
|
|
|
if (sent != null || received != null || senderInfo != null) CellWithPaddingAndMargin {
|
|
|
|
|
Column(verticalArrangement = Arrangement.spacedBy(16.dp)) {
|
|
|
|
|
sent?.let { TitledText(it) }
|
|
|
|
|
received?.let { TitledText(it) }
|
|
|
|
|
error?.let { TitledErrorText(it) }
|
|
|
|
|
senderInfo?.let {
|
|
|
|
|
TitledView(stringResource(id = R.string.message_details_header__from)) {
|
|
|
|
|
Row {
|
|
|
|
|
sender?.let { Avatar(it) }
|
|
|
|
|
TitledMonospaceText(it)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@Composable
|
|
|
|
|
fun Buttons(
|
|
|
|
|
hasError: Boolean,
|
|
|
|
|
onReply: () -> Unit = {},
|
|
|
|
|
onResend: () -> Unit = {},
|
|
|
|
|
onDelete: () -> Unit = {},
|
|
|
|
|
) {
|
|
|
|
|
Cell {
|
|
|
|
|
Column {
|
|
|
|
|
@Composable
|
|
|
|
|
fun Buttons(
|
|
|
|
|
onReply: () -> Unit = {},
|
|
|
|
|
onResend: (() -> Unit)? = null,
|
|
|
|
|
onDelete: () -> Unit = {},
|
|
|
|
|
) {
|
|
|
|
|
Cell {
|
|
|
|
|
Column {
|
|
|
|
|
ItemButton(
|
|
|
|
|
stringResource(id = R.string.reply),
|
|
|
|
|
R.drawable.ic_message_details__reply,
|
|
|
|
|
onClick = onReply
|
|
|
|
|
)
|
|
|
|
|
Divider()
|
|
|
|
|
onResend?.let {
|
|
|
|
|
ItemButton(
|
|
|
|
|
"Reply",
|
|
|
|
|
R.drawable.ic_message_details__reply,
|
|
|
|
|
onClick = onReply
|
|
|
|
|
stringResource(id = R.string.resend),
|
|
|
|
|
R.drawable.ic_message_details__refresh,
|
|
|
|
|
onClick = it
|
|
|
|
|
)
|
|
|
|
|
Divider()
|
|
|
|
|
if (hasError) {
|
|
|
|
|
ItemButton(
|
|
|
|
|
"Resend",
|
|
|
|
|
R.drawable.ic_message_details__refresh,
|
|
|
|
|
onClick = onResend
|
|
|
|
|
)
|
|
|
|
|
Divider()
|
|
|
|
|
}
|
|
|
|
|
ItemButton(
|
|
|
|
|
"Delete",
|
|
|
|
|
R.drawable.ic_message_details__trash,
|
|
|
|
|
colors = destructiveButtonColors(),
|
|
|
|
|
onClick = onDelete
|
|
|
|
|
)
|
|
|
|
|
}
|
|
|
|
|
ItemButton(
|
|
|
|
|
stringResource(id = R.string.delete),
|
|
|
|
|
R.drawable.ic_message_details__trash,
|
|
|
|
|
colors = destructiveButtonColors(),
|
|
|
|
|
onClick = onDelete
|
|
|
|
|
)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@Composable
|
|
|
|
|
fun Attachments(attachments: List<Attachment>, onClick: (Slide) -> Unit) {
|
|
|
|
|
when(attachments.firstOrNull()?.slide) {
|
|
|
|
|
is ImageSlide -> Carousel(attachments, onClick)
|
|
|
|
|
@OptIn(ExperimentalFoundationApi::class)
|
|
|
|
|
@Composable
|
|
|
|
|
fun Carousel(attachments: List<Attachment>, onClick: (Slide) -> Unit) {
|
|
|
|
|
val imageAttachments = attachments.filter { it.hasImage() }.takeIf { it.isNotEmpty() } ?: return
|
|
|
|
|
val pagerState = rememberPagerState { imageAttachments.size }
|
|
|
|
|
|
|
|
|
|
Column(verticalArrangement = Arrangement.spacedBy(16.dp)) {
|
|
|
|
|
Row {
|
|
|
|
|
CarouselPrevButton(pagerState)
|
|
|
|
|
Box(modifier = Modifier.weight(1f)) {
|
|
|
|
|
CellCarousel(pagerState, imageAttachments, onClick)
|
|
|
|
|
HorizontalPagerIndicator(pagerState)
|
|
|
|
|
ExpandButton(
|
|
|
|
|
modifier = Modifier
|
|
|
|
|
.align(Alignment.BottomEnd)
|
|
|
|
|
.padding(8.dp)
|
|
|
|
|
) { onClick(imageAttachments[pagerState.currentPage].slide) }
|
|
|
|
|
}
|
|
|
|
|
CarouselNextButton(pagerState)
|
|
|
|
|
}
|
|
|
|
|
FileDetails(attachments, pagerState)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@OptIn(ExperimentalFoundationApi::class)
|
|
|
|
|
@Composable
|
|
|
|
|
fun Carousel(attachments: List<Attachment>, onClick: (Slide) -> Unit) {
|
|
|
|
|
val imageAttachments = attachments.filter { it.slide.hasImage() }
|
|
|
|
|
val pagerState = rememberPagerState { imageAttachments.size }
|
|
|
|
|
|
|
|
|
|
Column(verticalArrangement = Arrangement.spacedBy(16.dp)) {
|
|
|
|
|
Row {
|
|
|
|
|
CarouselPrevButton(pagerState)
|
|
|
|
|
Box(modifier = Modifier.weight(1f)) {
|
|
|
|
|
CellPager(pagerState, imageAttachments, onClick)
|
|
|
|
|
HorizontalPagerIndicator(pagerState)
|
|
|
|
|
ExpandButton(
|
|
|
|
|
modifier = Modifier
|
|
|
|
|
.align(Alignment.BottomEnd)
|
|
|
|
|
.padding(8.dp)
|
|
|
|
|
) { onClick(imageAttachments[pagerState.currentPage].slide) }
|
|
|
|
|
}
|
|
|
|
|
CarouselNextButton(pagerState)
|
|
|
|
|
}
|
|
|
|
|
FileDetails(attachments, pagerState)
|
|
|
|
|
@OptIn(
|
|
|
|
|
ExperimentalFoundationApi::class,
|
|
|
|
|
ExperimentalGlideComposeApi::class
|
|
|
|
|
)
|
|
|
|
|
@Composable
|
|
|
|
|
private fun CellCarousel(
|
|
|
|
|
pagerState: PagerState,
|
|
|
|
|
imageAttachments: List<Attachment>,
|
|
|
|
|
onClick: (Slide) -> Unit
|
|
|
|
|
) {
|
|
|
|
|
CellNoMargin {
|
|
|
|
|
HorizontalPager(state = pagerState) { i ->
|
|
|
|
|
val slide = imageAttachments[i].slide
|
|
|
|
|
GlideImage(
|
|
|
|
|
contentScale = ContentScale.Crop,
|
|
|
|
|
modifier = Modifier
|
|
|
|
|
.aspectRatio(1f)
|
|
|
|
|
.clickable { onClick(slide) },
|
|
|
|
|
model = slide.uri,
|
|
|
|
|
contentDescription = slide.fileName.orNull() ?: stringResource(id = R.string.image)
|
|
|
|
|
)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@OptIn(
|
|
|
|
|
ExperimentalFoundationApi::class,
|
|
|
|
|
ExperimentalGlideComposeApi::class
|
|
|
|
|
)
|
|
|
|
|
@Composable
|
|
|
|
|
private fun CellPager(
|
|
|
|
|
pagerState: PagerState,
|
|
|
|
|
imageAttachments: List<Attachment>,
|
|
|
|
|
onClick: (Slide) -> Unit
|
|
|
|
|
@Composable
|
|
|
|
|
fun ExpandButton(modifier: Modifier = Modifier, onClick: () -> Unit) {
|
|
|
|
|
Surface(
|
|
|
|
|
shape = CircleShape,
|
|
|
|
|
color = blackAlpha40,
|
|
|
|
|
modifier = modifier,
|
|
|
|
|
contentColor = Color.White,
|
|
|
|
|
) {
|
|
|
|
|
CellNoMargin {
|
|
|
|
|
HorizontalPager(state = pagerState) { i ->
|
|
|
|
|
val slide = imageAttachments[i].slide
|
|
|
|
|
GlideImage(
|
|
|
|
|
contentScale = ContentScale.Crop,
|
|
|
|
|
modifier = Modifier
|
|
|
|
|
.aspectRatio(1f)
|
|
|
|
|
.clickable { onClick(slide) },
|
|
|
|
|
model = slide.uri,
|
|
|
|
|
contentDescription = slide.fileName.orNull() ?: "image"
|
|
|
|
|
)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
Icon(
|
|
|
|
|
painter = painterResource(id = R.drawable.ic_expand),
|
|
|
|
|
contentDescription = "",
|
|
|
|
|
modifier = Modifier.clickable { onClick() },
|
|
|
|
|
)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@Composable
|
|
|
|
|
fun ExpandButton(modifier: Modifier, onClick: () -> Unit) {
|
|
|
|
|
Surface(
|
|
|
|
|
shape = CircleShape,
|
|
|
|
|
color = blackAlpha40,
|
|
|
|
|
modifier = modifier
|
|
|
|
|
) {
|
|
|
|
|
Icon(
|
|
|
|
|
painter = painterResource(id = R.drawable.ic_expand),
|
|
|
|
|
contentDescription = "",
|
|
|
|
|
modifier = Modifier.clickable { onClick() }
|
|
|
|
|
@Preview
|
|
|
|
|
@Composable
|
|
|
|
|
fun PreviewFileDetails() {
|
|
|
|
|
Theme(R.style.Ocean_Dark) {
|
|
|
|
|
FileDetails(
|
|
|
|
|
fileDetails = listOf(
|
|
|
|
|
TitledText("File Id:", "Screen Shot 2023-07-06 at 11.35.50 am.png"),
|
|
|
|
|
TitledText("File Type:", "image/png"),
|
|
|
|
|
TitledText("File Size:", "195.6kB"),
|
|
|
|
|
TitledText("Resolution:", "342x312"),
|
|
|
|
|
)
|
|
|
|
|
}
|
|
|
|
|
)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@OptIn(ExperimentalFoundationApi::class, ExperimentalLayoutApi::class)
|
|
|
|
|
@Composable
|
|
|
|
|
fun FileDetails(attachments: List<Attachment>, pagerState: PagerState) {
|
|
|
|
|
attachments[pagerState.currentPage].fileDetails.takeIf { it.isNotEmpty() }?.let {
|
|
|
|
|
CellWithPaddingAndMargin {
|
|
|
|
|
FlowRow(
|
|
|
|
|
verticalArrangement = Arrangement.spacedBy(16.dp),
|
|
|
|
|
maxItemsInEachRow = 2
|
|
|
|
|
) {
|
|
|
|
|
it.forEach { TitledText(it, Modifier.weight(1f)) }
|
|
|
|
|
}
|
|
|
|
|
@OptIn(ExperimentalFoundationApi::class)
|
|
|
|
|
@Composable
|
|
|
|
|
fun FileDetails(attachments: List<Attachment>, pagerState: PagerState) {
|
|
|
|
|
FileDetails(attachments[pagerState.currentPage].fileDetails)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@OptIn(ExperimentalLayoutApi::class)
|
|
|
|
|
@Composable
|
|
|
|
|
fun FileDetails(fileDetails: List<TitledText>) {
|
|
|
|
|
if (fileDetails.isEmpty()) return
|
|
|
|
|
|
|
|
|
|
CellWithPaddingAndMargin {
|
|
|
|
|
FlowRow(verticalArrangement = Arrangement.spacedBy(16.dp)) {
|
|
|
|
|
fileDetails.forEach {
|
|
|
|
|
TitledText(
|
|
|
|
|
it,
|
|
|
|
|
modifier = Modifier
|
|
|
|
|
.widthIn(min = 100.dp) // set minimum width
|
|
|
|
|
.width(IntrinsicSize.Max) // make the text as wide as necessary
|
|
|
|
|
.weight(1f) // space evenly
|
|
|
|
|
)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@Composable
|
|
|
|
|
fun TitledErrorText(titledText: TitledText, modifier: Modifier = Modifier) {
|
|
|
|
|
TitledText(
|
|
|
|
|
titledText,
|
|
|
|
|
modifier = modifier,
|
|
|
|
|
valueStyle = LocalTextStyle.current.copy(color = colorDestructive)
|
|
|
|
|
)
|
|
|
|
|
}
|
|
|
|
|
@Composable
|
|
|
|
|
fun TitledErrorText(titledText: TitledText, modifier: Modifier = Modifier) {
|
|
|
|
|
TitledText(
|
|
|
|
|
titledText,
|
|
|
|
|
modifier = modifier,
|
|
|
|
|
valueStyle = LocalTextStyle.current.copy(color = colorDestructive)
|
|
|
|
|
)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@Composable
|
|
|
|
|
fun TitledMonospaceText(titledText: TitledText, modifier: Modifier = Modifier) {
|
|
|
|
|
TitledText(
|
|
|
|
|
titledText,
|
|
|
|
|
modifier = modifier,
|
|
|
|
|
valueStyle = LocalTextStyle.current.copy(fontFamily = FontFamily.Monospace)
|
|
|
|
|
)
|
|
|
|
|
}
|
|
|
|
|
@Composable
|
|
|
|
|
fun TitledMonospaceText(titledText: TitledText, modifier: Modifier = Modifier) {
|
|
|
|
|
TitledText(
|
|
|
|
|
titledText,
|
|
|
|
|
modifier = modifier,
|
|
|
|
|
valueStyle = LocalTextStyle.current.copy(fontFamily = FontFamily.Monospace)
|
|
|
|
|
)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@Composable
|
|
|
|
|
fun TitledText(
|
|
|
|
|
titledText: TitledText,
|
|
|
|
|
modifier: Modifier = Modifier,
|
|
|
|
|
valueStyle: TextStyle = LocalTextStyle.current
|
|
|
|
|
) {
|
|
|
|
|
Column(modifier = modifier, verticalArrangement = Arrangement.spacedBy(4.dp)) {
|
|
|
|
|
Title(titledText.title)
|
|
|
|
|
Text(titledText.value, style = valueStyle)
|
|
|
|
|
}
|
|
|
|
|
@Composable
|
|
|
|
|
fun TitledText(
|
|
|
|
|
titledText: TitledText,
|
|
|
|
|
modifier: Modifier = Modifier,
|
|
|
|
|
valueStyle: TextStyle = LocalTextStyle.current
|
|
|
|
|
) {
|
|
|
|
|
Column(modifier = modifier, verticalArrangement = Arrangement.spacedBy(4.dp)) {
|
|
|
|
|
Title(titledText.title)
|
|
|
|
|
Text(titledText.value, style = valueStyle)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@Composable
|
|
|
|
|
fun TitledView(title: String, modifier: Modifier = Modifier, content: @Composable () -> Unit) {
|
|
|
|
|
Column(modifier = modifier, verticalArrangement = Arrangement.spacedBy(4.dp)) {
|
|
|
|
|
Title(title)
|
|
|
|
|
content()
|
|
|
|
|
}
|
|
|
|
|
@Composable
|
|
|
|
|
fun TitledView(title: String, modifier: Modifier = Modifier, content: @Composable () -> Unit) {
|
|
|
|
|
Column(modifier = modifier, verticalArrangement = Arrangement.spacedBy(4.dp)) {
|
|
|
|
|
Title(title)
|
|
|
|
|
content()
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@Composable
|
|
|
|
|
fun Title(text: String) {
|
|
|
|
|
Text(text, fontWeight = FontWeight.Bold)
|
|
|
|
|
}
|
|
|
|
|
@Composable
|
|
|
|
|
fun Title(text: String, modifier: Modifier = Modifier) {
|
|
|
|
|
Text(text, modifier = modifier, fontWeight = FontWeight.Bold)
|
|
|
|
|
}
|
|
|
|
|