e-ntfy-android/app/src/main/java/io/heckel/ntfy/ui/DetailAdapter.kt

357 lines
18 KiB
Kotlin
Raw Normal View History

2021-10-31 20:19:25 +01:00
package io.heckel.ntfy.ui
import android.Manifest
import android.app.Activity
2022-01-10 19:46:31 +01:00
import android.content.*
import android.content.pm.PackageManager
2022-01-10 21:36:50 +01:00
import android.graphics.Bitmap
2022-01-09 04:17:41 +01:00
import android.graphics.BitmapFactory
import android.net.Uri
import android.os.Build
2021-10-31 20:19:25 +01:00
import android.view.LayoutInflater
import android.view.View
import android.view.ViewGroup
import android.widget.*
import androidx.core.app.ActivityCompat
2021-11-27 22:18:09 +01:00
import androidx.core.content.ContextCompat
2021-10-31 20:19:25 +01:00
import androidx.recyclerview.widget.DiffUtil
import androidx.recyclerview.widget.ListAdapter
import androidx.recyclerview.widget.RecyclerView
2022-01-10 21:36:50 +01:00
import com.stfalcon.imageviewer.StfalconImageViewer
2021-10-31 20:19:25 +01:00
import io.heckel.ntfy.R
2022-01-18 20:28:48 +01:00
import io.heckel.ntfy.db.*
import io.heckel.ntfy.util.Log
2022-01-12 00:21:30 +01:00
import io.heckel.ntfy.msg.DownloadManager
2021-11-29 01:28:58 +01:00
import io.heckel.ntfy.util.*
2022-01-12 02:47:28 +01:00
import kotlinx.coroutines.Dispatchers
import kotlinx.coroutines.GlobalScope
import kotlinx.coroutines.launch
2021-10-31 20:19:25 +01:00
import java.util.*
2022-01-12 02:47:28 +01:00
class DetailAdapter(private val activity: Activity, private val repository: Repository, private val onClick: (Notification) -> Unit, private val onLongClick: (Notification) -> Unit) :
2021-10-31 20:19:25 +01:00
ListAdapter<Notification, DetailAdapter.DetailViewHolder>(TopicDiffCallback) {
2021-11-03 18:56:08 +01:00
val selected = mutableSetOf<String>() // Notification IDs
2021-10-31 20:19:25 +01:00
/* Creates and inflates view and return TopicViewHolder. */
override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): DetailViewHolder {
val view = LayoutInflater.from(parent.context)
2021-11-22 21:45:43 +01:00
.inflate(R.layout.fragment_detail_item, parent, false)
2022-01-12 02:47:28 +01:00
return DetailViewHolder(activity, repository, view, selected, onClick, onLongClick)
2021-10-31 20:19:25 +01:00
}
/* Gets current topic and uses it to bind view. */
override fun onBindViewHolder(holder: DetailViewHolder, position: Int) {
holder.bind(getItem(position))
}
2022-02-06 03:02:05 +01:00
fun get(position: Int): Notification {
return getItem(position)
}
2021-11-03 18:56:08 +01:00
fun toggleSelection(notificationId: String) {
if (selected.contains(notificationId)) {
selected.remove(notificationId)
} else {
selected.add(notificationId)
}
}
2021-10-31 20:19:25 +01:00
/* ViewHolder for Topic, takes in the inflated view and the onClick behavior. */
2022-01-12 02:47:28 +01:00
class DetailViewHolder(private val activity: Activity, private val repository: Repository, itemView: View, private val selected: Set<String>, val onClick: (Notification) -> Unit, val onLongClick: (Notification) -> Unit) :
2021-10-31 20:19:25 +01:00
RecyclerView.ViewHolder(itemView) {
private var notification: Notification? = null
2021-11-27 22:18:09 +01:00
private val priorityImageView: ImageView = itemView.findViewById(R.id.detail_item_priority_image)
2021-10-31 20:19:25 +01:00
private val dateView: TextView = itemView.findViewById(R.id.detail_item_date_text)
2021-11-27 22:18:09 +01:00
private val titleView: TextView = itemView.findViewById(R.id.detail_item_title_text)
2021-10-31 20:19:25 +01:00
private val messageView: TextView = itemView.findViewById(R.id.detail_item_message_text)
2022-01-10 19:46:31 +01:00
private val newDotImageView: View = itemView.findViewById(R.id.detail_item_new_dot)
private val tagsView: TextView = itemView.findViewById(R.id.detail_item_tags_text)
private val menuButton: ImageButton = itemView.findViewById(R.id.detail_item_menu_button)
2022-01-10 19:46:31 +01:00
private val attachmentImageView: ImageView = itemView.findViewById(R.id.detail_item_attachment_image)
private val attachmentBoxView: View = itemView.findViewById(R.id.detail_item_attachment_box)
private val attachmentIconView: ImageView = itemView.findViewById(R.id.detail_item_attachment_icon)
private val attachmentInfoView: TextView = itemView.findViewById(R.id.detail_item_attachment_info)
2021-10-31 20:19:25 +01:00
fun bind(notification: Notification) {
this.notification = notification
2021-11-27 22:18:09 +01:00
val context = itemView.context
2021-11-29 01:28:58 +01:00
val unmatchedTags = unmatchedTags(splitTags(notification.tags))
2021-10-31 20:19:25 +01:00
dateView.text = Date(notification.timestamp * 1000).toString()
2021-11-27 22:18:09 +01:00
messageView.text = formatMessage(notification)
2022-01-10 19:46:31 +01:00
newDotImageView.visibility = if (notification.notificationId == 0) View.GONE else View.VISIBLE
2021-10-31 20:19:25 +01:00
itemView.setOnClickListener { onClick(notification) }
2021-11-03 18:56:08 +01:00
itemView.setOnLongClickListener { onLongClick(notification); true }
2021-11-27 22:18:09 +01:00
if (notification.title != "") {
titleView.visibility = View.VISIBLE
titleView.text = formatTitle(notification)
} else {
titleView.visibility = View.GONE
}
2021-11-29 01:28:58 +01:00
if (unmatchedTags.isNotEmpty()) {
tagsView.visibility = View.VISIBLE
tagsView.text = context.getString(R.string.detail_item_tags, unmatchedTags.joinToString(", "))
2021-11-29 01:28:58 +01:00
} else {
tagsView.visibility = View.GONE
}
2021-11-03 18:56:08 +01:00
if (selected.contains(notification.id)) {
itemView.setBackgroundResource(Colors.itemSelectedBackground(context))
2021-11-03 18:56:08 +01:00
}
2022-01-10 19:46:31 +01:00
renderPriority(context, notification)
maybeRenderAttachment(context, notification)
}
private fun renderPriority(context: Context, notification: Notification) {
2021-11-27 22:18:09 +01:00
when (notification.priority) {
1 -> {
priorityImageView.visibility = View.VISIBLE
priorityImageView.setImageDrawable(ContextCompat.getDrawable(context, R.drawable.ic_priority_1_24dp))
2021-11-27 22:18:09 +01:00
}
2 -> {
priorityImageView.visibility = View.VISIBLE
priorityImageView.setImageDrawable(ContextCompat.getDrawable(context, R.drawable.ic_priority_2_24dp))
2021-11-27 22:18:09 +01:00
}
3 -> {
priorityImageView.visibility = View.GONE
}
4 -> {
priorityImageView.visibility = View.VISIBLE
priorityImageView.setImageDrawable(ContextCompat.getDrawable(context, R.drawable.ic_priority_4_24dp))
2021-11-27 22:18:09 +01:00
}
5 -> {
priorityImageView.visibility = View.VISIBLE
priorityImageView.setImageDrawable(ContextCompat.getDrawable(context, R.drawable.ic_priority_5_24dp))
2021-11-27 22:18:09 +01:00
}
}
2022-01-10 19:46:31 +01:00
}
private fun maybeRenderAttachment(context: Context, notification: Notification) {
if (notification.attachment == null) {
menuButton.visibility = View.GONE
attachmentImageView.visibility = View.GONE
attachmentBoxView.visibility = View.GONE
return
2022-01-09 04:17:41 +01:00
}
2022-01-10 19:46:31 +01:00
val attachment = notification.attachment
val exists = if (attachment.contentUri != null) fileExists(context, attachment.contentUri) else false
2022-01-10 21:36:50 +01:00
val image = attachment.contentUri != null && exists && supportedImage(attachment.type)
maybeRenderMenu(context, notification, attachment, exists)
maybeRenderAttachmentImage(context, attachment, image)
maybeRenderAttachmentBox(context, notification, attachment, exists, image)
2022-01-10 19:46:31 +01:00
}
2022-01-10 21:36:50 +01:00
private fun maybeRenderMenu(context: Context, notification: Notification, attachment: Attachment, exists: Boolean) {
2022-01-10 19:46:31 +01:00
val menuButtonPopupMenu = createAttachmentPopup(context, menuButton, notification, attachment, exists) // Heavy lifting not during on-click
if (menuButtonPopupMenu != null) {
menuButton.setOnClickListener { menuButtonPopupMenu.show() }
menuButton.visibility = View.VISIBLE
2022-01-10 19:46:31 +01:00
} else {
menuButton.visibility = View.GONE
}
2022-01-10 21:36:50 +01:00
}
private fun maybeRenderAttachmentBox(context: Context, notification: Notification, attachment: Attachment, exists: Boolean, image: Boolean) {
if (image) {
attachmentBoxView.visibility = View.GONE
return
}
attachmentInfoView.text = formatAttachmentDetails(context, attachment, exists)
attachmentIconView.setImageResource(if (attachment.type?.startsWith("image/") == true) {
R.drawable.ic_file_image_red_24dp
} else if (attachment.type?.startsWith("video/") == true) {
R.drawable.ic_file_video_orange_24dp
} else if (attachment.type?.startsWith("audio/") == true) {
R.drawable.ic_file_audio_purple_24dp
} else if ("application/vnd.android.package-archive" == attachment.type) {
R.drawable.ic_file_app_gray_24dp
} else {
R.drawable.ic_file_document_blue_24dp
})
2022-01-10 19:46:31 +01:00
val attachmentBoxPopupMenu = createAttachmentPopup(context, attachmentBoxView, notification, attachment, exists) // Heavy lifting not during on-click
if (attachmentBoxPopupMenu != null) {
attachmentBoxView.setOnClickListener { attachmentBoxPopupMenu.show() }
} else {
attachmentBoxView.setOnClickListener {
Toast
.makeText(context, context.getString(R.string.detail_item_cannot_download), Toast.LENGTH_LONG)
.show()
}
}
attachmentBoxView.visibility = View.VISIBLE
}
private fun createAttachmentPopup(context: Context, anchor: View?, notification: Notification, attachment: Attachment, exists: Boolean): PopupMenu? {
val popup = PopupMenu(context, anchor)
popup.menuInflater.inflate(R.menu.menu_detail_attachment, popup.menu)
val downloadItem = popup.menu.findItem(R.id.detail_item_menu_download)
2022-01-12 00:21:30 +01:00
val cancelItem = popup.menu.findItem(R.id.detail_item_menu_cancel)
2022-01-10 19:46:31 +01:00
val openItem = popup.menu.findItem(R.id.detail_item_menu_open)
val browseItem = popup.menu.findItem(R.id.detail_item_menu_browse)
2022-01-12 02:47:28 +01:00
val deleteItem = popup.menu.findItem(R.id.detail_item_menu_delete)
2022-01-10 19:46:31 +01:00
val copyUrlItem = popup.menu.findItem(R.id.detail_item_menu_copy_url)
val expired = attachment.expires != null && attachment.expires < System.currentTimeMillis()/1000
2022-01-12 00:21:30 +01:00
val inProgress = attachment.progress in 0..99
2022-01-10 19:46:31 +01:00
if (attachment.contentUri != null) {
openItem.setOnMenuItemClickListener {
try {
val contentUri = Uri.parse(attachment.contentUri)
val intent = Intent(Intent.ACTION_VIEW, contentUri)
intent.setDataAndType(contentUri, attachment.type ?: "application/octet-stream") // Required for Android <= P
intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION)
context.startActivity(intent)
2022-01-10 19:46:31 +01:00
} catch (e: ActivityNotFoundException) {
Toast
.makeText(context, context.getString(R.string.detail_item_cannot_open_not_found), Toast.LENGTH_LONG)
.show()
} catch (e: Exception) {
Toast
.makeText(context, context.getString(R.string.detail_item_cannot_open, e.message), Toast.LENGTH_LONG)
2022-01-10 19:46:31 +01:00
.show()
}
2022-01-10 19:46:31 +01:00
true
}
}
browseItem.setOnMenuItemClickListener {
2022-01-12 00:21:30 +01:00
val intent = Intent(android.app.DownloadManager.ACTION_VIEW_DOWNLOADS)
intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION)
context.startActivity(intent)
2022-01-10 19:46:31 +01:00
true
}
2022-01-12 02:47:28 +01:00
if (attachment.contentUri != null) {
deleteItem.setOnMenuItemClickListener {
try {
val contentUri = Uri.parse(attachment.contentUri)
val resolver = context.applicationContext.contentResolver
val deleted = resolver.delete(contentUri, null, null) > 0
if (!deleted) throw Exception("no rows deleted")
val newAttachment = attachment.copy(progress = PROGRESS_DELETED)
val newNotification = notification.copy(attachment = newAttachment)
GlobalScope.launch(Dispatchers.IO) {
repository.updateNotification(newNotification)
}
} catch (e: Exception) {
Log.w(TAG, "Failed to update notification: ${e.message}", e)
Toast
.makeText(context, context.getString(R.string.detail_item_delete_failed, e.message), Toast.LENGTH_LONG)
.show()
}
true
}
}
2022-01-10 19:46:31 +01:00
copyUrlItem.setOnMenuItemClickListener {
val clipboard = context.getSystemService(Context.CLIPBOARD_SERVICE) as ClipboardManager
val clip = ClipData.newPlainText("attachment url", attachment.url)
clipboard.setPrimaryClip(clip)
Toast
.makeText(context, context.getString(R.string.detail_item_menu_copy_url_copied), Toast.LENGTH_LONG)
.show()
true
}
downloadItem.setOnMenuItemClickListener {
val requiresPermission = Build.VERSION.SDK_INT <= Build.VERSION_CODES.P && ContextCompat.checkSelfPermission(context, Manifest.permission.WRITE_EXTERNAL_STORAGE) == PackageManager.PERMISSION_DENIED
if (requiresPermission) {
ActivityCompat.requestPermissions(activity, arrayOf(Manifest.permission.WRITE_EXTERNAL_STORAGE), REQUEST_CODE_WRITE_STORAGE_PERMISSION_FOR_DOWNLOAD)
return@setOnMenuItemClickListener true
}
DownloadManager.enqueue(context, notification.id, userAction = true)
2022-01-12 00:21:30 +01:00
true
}
cancelItem.setOnMenuItemClickListener {
DownloadManager.cancel(context, notification.id)
2022-01-10 19:46:31 +01:00
true
}
openItem.isVisible = exists
browseItem.isVisible = exists
2022-01-12 00:21:30 +01:00
downloadItem.isVisible = !exists && !expired && !inProgress
2022-01-12 02:47:28 +01:00
deleteItem.isVisible = exists
2022-01-10 19:46:31 +01:00
copyUrlItem.isVisible = !expired
2022-01-12 00:21:30 +01:00
cancelItem.isVisible = inProgress
2022-01-12 02:47:28 +01:00
val noOptions = !openItem.isVisible && !browseItem.isVisible && !downloadItem.isVisible && !copyUrlItem.isVisible && !cancelItem.isVisible && !deleteItem.isVisible
2022-01-10 19:46:31 +01:00
if (noOptions) {
return null
}
return popup
}
2022-01-10 19:46:31 +01:00
private fun formatAttachmentDetails(context: Context, attachment: Attachment, exists: Boolean): String {
val name = queryFilename(context, attachment.contentUri, attachment.name)
2022-01-10 19:46:31 +01:00
val notYetDownloaded = !exists && attachment.progress == PROGRESS_NONE
val downloading = !exists && attachment.progress in 0..99
2022-01-12 02:47:28 +01:00
val deleted = !exists && (attachment.progress == PROGRESS_DONE || attachment.progress == PROGRESS_DELETED)
val failed = !exists && attachment.progress == PROGRESS_FAILED
2022-01-10 19:46:31 +01:00
val expired = attachment.expires != null && attachment.expires < System.currentTimeMillis()/1000
val expires = attachment.expires != null && attachment.expires > System.currentTimeMillis()/1000
val infos = mutableListOf<String>()
if (attachment.size != null) {
infos.add(formatBytes(attachment.size))
}
if (notYetDownloaded) {
if (expired) {
infos.add(context.getString(R.string.detail_item_download_info_not_downloaded_expired))
2022-01-10 19:46:31 +01:00
} else if (expires) {
infos.add(context.getString(R.string.detail_item_download_info_not_downloaded_expires_x, formatDateShort(attachment.expires!!)))
2022-01-10 19:46:31 +01:00
} else {
infos.add(context.getString(R.string.detail_item_download_info_not_downloaded))
2022-01-10 19:46:31 +01:00
}
} else if (downloading) {
infos.add(context.getString(R.string.detail_item_download_info_downloading_x_percent, attachment.progress))
2022-01-10 19:46:31 +01:00
} else if (deleted) {
if (expired) {
infos.add(context.getString(R.string.detail_item_download_info_deleted_expired))
2022-01-10 19:46:31 +01:00
} else if (expires) {
infos.add(context.getString(R.string.detail_item_download_info_deleted_expires_x, formatDateShort(attachment.expires!!)))
2022-01-10 19:46:31 +01:00
} else {
infos.add(context.getString(R.string.detail_item_download_info_deleted))
}
} else if (failed) {
infos.add(context.getString(R.string.detail_item_download_info_download_failed))
2022-01-10 19:46:31 +01:00
}
return if (infos.size > 0) {
"$name\n${infos.joinToString(", ")}"
} else {
2022-01-10 19:46:31 +01:00
name
}
}
2022-01-10 21:36:50 +01:00
private fun maybeRenderAttachmentImage(context: Context, attachment: Attachment, image: Boolean) {
if (!image) {
2022-01-10 19:46:31 +01:00
attachmentImageView.visibility = View.GONE
return
}
try {
val resolver = context.applicationContext.contentResolver
2022-01-10 21:36:50 +01:00
val bitmapStream = resolver.openInputStream(Uri.parse(attachment.contentUri))
2022-01-10 19:46:31 +01:00
val bitmap = BitmapFactory.decodeStream(bitmapStream)
attachmentImageView.setImageBitmap(bitmap)
2022-01-10 21:36:50 +01:00
attachmentImageView.setOnClickListener {
val loadImage = { view: ImageView, image: Bitmap -> view.setImageBitmap(image) }
StfalconImageViewer.Builder(context, listOf(bitmap), loadImage)
.allowZooming(true)
.withTransitionFrom(attachmentImageView)
.withHiddenStatusBar(false)
.show()
}
2022-01-10 19:46:31 +01:00
attachmentImageView.visibility = View.VISIBLE
} catch (_: Exception) {
attachmentImageView.visibility = View.GONE
}
}
2021-10-31 20:19:25 +01:00
}
object TopicDiffCallback : DiffUtil.ItemCallback<Notification>() {
override fun areItemsTheSame(oldItem: Notification, newItem: Notification): Boolean {
return oldItem.id == newItem.id
}
override fun areContentsTheSame(oldItem: Notification, newItem: Notification): Boolean {
return oldItem == newItem
}
}
companion object {
const val TAG = "NtfyDetailAdapter"
const val REQUEST_CODE_WRITE_STORAGE_PERMISSION_FOR_DOWNLOAD = 9876
}
2021-10-31 20:19:25 +01:00
}