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

321 lines
11 KiB
Kotlin
Raw Normal View History

2021-10-28 16:04:14 +13:00
package io.heckel.ntfy.ui
2021-11-04 05:48:13 +13:00
import android.animation.Animator
import android.animation.AnimatorListenerAdapter
import android.app.AlertDialog
import android.content.Intent
2021-10-28 16:04:14 +13:00
import android.net.Uri
import android.os.Bundle
2021-11-02 02:57:05 +13:00
import android.util.Log
2021-11-04 05:48:13 +13:00
import android.view.*
2021-11-08 07:29:19 +13:00
import android.widget.Toast
import androidx.activity.viewModels
2021-10-26 07:24:44 +13:00
import androidx.appcompat.app.AppCompatActivity
2021-11-04 05:48:13 +13:00
import androidx.core.content.ContextCompat
2021-11-11 15:16:00 +13:00
import androidx.lifecycle.asFlow
2021-11-08 07:29:19 +13:00
import androidx.lifecycle.lifecycleScope
import androidx.recyclerview.widget.RecyclerView
2021-10-30 14:13:58 +13:00
import com.google.firebase.messaging.FirebaseMessaging
2021-10-31 13:16:12 +13:00
import io.heckel.ntfy.R
2021-10-30 14:13:58 +13:00
import io.heckel.ntfy.app.Application
2021-11-08 07:29:19 +13:00
import io.heckel.ntfy.data.Notification
2021-10-31 13:16:12 +13:00
import io.heckel.ntfy.data.Subscription
2021-11-01 08:19:25 +13:00
import io.heckel.ntfy.data.topicShortUrl
2021-11-08 07:29:19 +13:00
import io.heckel.ntfy.msg.ApiService
import kotlinx.coroutines.Dispatchers
2021-11-11 15:16:00 +13:00
import kotlinx.coroutines.flow.collect
2021-11-08 07:29:19 +13:00
import kotlinx.coroutines.launch
2021-11-01 08:19:25 +13:00
import java.util.*
2021-10-31 13:16:12 +13:00
import kotlin.random.Random
2021-11-04 05:48:13 +13:00
class MainActivity : AppCompatActivity(), ActionMode.Callback {
2021-11-01 08:19:25 +13:00
private val viewModel by viewModels<SubscriptionsViewModel> {
2021-10-30 14:13:58 +13:00
SubscriptionsViewModelFactory((application as Application).repository)
}
2021-11-08 07:29:19 +13:00
private val repository by lazy { (application as Application).repository }
2021-11-04 05:48:13 +13:00
private lateinit var mainList: RecyclerView
private lateinit var adapter: MainAdapter
private lateinit var fab: View
private var actionMode: ActionMode? = null
2021-11-08 07:29:19 +13:00
private lateinit var api: ApiService // Context-dependent
2021-10-30 14:13:58 +13:00
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
2021-10-29 01:28:22 +13:00
setContentView(R.layout.main_activity)
2021-11-08 07:29:19 +13:00
// Dependencies that depend on Context
api = ApiService(this)
2021-10-29 04:45:34 +13:00
// Action bar
title = getString(R.string.main_action_bar_title)
// Floating action button ("+")
2021-11-04 05:48:13 +13:00
fab = findViewById(R.id.fab)
fab.setOnClickListener {
2021-10-30 14:13:58 +13:00
onSubscribeButtonClick()
}
2021-11-01 08:19:25 +13:00
// Update main list based on viewModel (& its datasource/livedata)
2021-11-02 03:49:52 +13:00
val noEntries: View = findViewById(R.id.main_no_subscriptions)
2021-11-04 05:48:13 +13:00
val onSubscriptionClick = { s: Subscription -> onSubscriptionItemClick(s) }
val onSubscriptionLongClick = { s: Subscription -> onSubscriptionItemLongClick(s) }
mainList = findViewById(R.id.main_subscriptions_list)
adapter = MainAdapter(onSubscriptionClick, onSubscriptionLongClick)
2021-10-29 04:45:34 +13:00
mainList.adapter = adapter
2021-11-01 08:19:25 +13:00
viewModel.list().observe(this) {
it?.let {
2021-10-28 09:15:59 +13:00
adapter.submitList(it as MutableList<Subscription>)
2021-10-29 04:45:34 +13:00
if (it.isEmpty()) {
mainList.visibility = View.GONE
2021-11-02 03:49:52 +13:00
noEntries.visibility = View.VISIBLE
2021-10-29 04:45:34 +13:00
} else {
mainList.visibility = View.VISIBLE
2021-11-02 03:49:52 +13:00
noEntries.visibility = View.GONE
2021-10-29 04:45:34 +13:00
}
}
2021-10-26 06:45:56 +13:00
}
2021-10-26 14:14:09 +13:00
}
2021-10-26 13:25:54 +13:00
2021-10-28 16:04:14 +13:00
override fun onCreateOptionsMenu(menu: Menu): Boolean {
2021-10-29 04:45:34 +13:00
menuInflater.inflate(R.menu.main_action_bar_menu, menu)
2021-10-28 16:04:14 +13:00
return true
}
override fun onOptionsItemSelected(item: MenuItem): Boolean {
return when (item.itemId) {
2021-11-11 15:16:00 +13:00
R.id.main_menu_refresh -> {
refreshAllSubscriptions()
true
}
2021-11-02 02:57:05 +13:00
R.id.main_menu_source -> {
2021-10-29 01:28:22 +13:00
startActivity(Intent(Intent.ACTION_VIEW, Uri.parse(getString(R.string.main_menu_source_url))))
2021-10-28 16:04:14 +13:00
true
}
2021-11-02 02:57:05 +13:00
R.id.main_menu_website -> {
2021-10-30 14:13:58 +13:00
startActivity(Intent(Intent.ACTION_VIEW, Uri.parse(getString(R.string.app_base_url))))
2021-10-28 16:04:14 +13:00
true
}
else -> super.onOptionsItemSelected(item)
}
}
2021-10-30 14:13:58 +13:00
private fun onSubscribeButtonClick() {
val newFragment = AddFragment(viewModel) { topic, baseUrl -> onSubscribe(topic, baseUrl) }
2021-10-28 15:25:02 +13:00
newFragment.show(supportFragmentManager, "AddFragment")
}
private fun onSubscribe(topic: String, baseUrl: String) {
2021-11-08 07:29:19 +13:00
// FIXME ignores baseUrl
2021-11-02 02:57:05 +13:00
Log.d(TAG, "Adding subscription ${topicShortUrl(baseUrl, topic)}")
2021-11-08 07:29:19 +13:00
// Add subscription to database
2021-11-04 05:48:13 +13:00
val subscription = Subscription(
id = Random.nextLong(),
baseUrl = baseUrl,
topic = topic,
notifications = 0,
lastActive = Date().time/1000
)
2021-11-01 08:19:25 +13:00
viewModel.add(subscription)
2021-11-08 07:29:19 +13:00
// Subscribe to Firebase topic
2021-11-06 13:25:02 +13:00
FirebaseMessaging
.getInstance()
.subscribeToTopic(topic)
.addOnCompleteListener {
Log.d(TAG, "Subscribing to topic complete: result=${it.result}, exception=${it.exception}, successful=${it.isSuccessful}")
}
.addOnFailureListener {
Log.e(TAG, "Subscribing to topic failed: $it")
}
2021-11-08 07:29:19 +13:00
// Fetch cached messages
val successFn = { notifications: List<Notification> ->
lifecycleScope.launch(Dispatchers.IO) {
notifications.forEach { repository.addNotification(it) }
}
Unit
}
api.poll(subscription.id, subscription.baseUrl, subscription.topic, successFn, { _ -> })
2021-11-03 14:43:31 +13:00
2021-11-08 07:29:19 +13:00
// Switch to detail view after adding it
onSubscriptionItemClick(subscription)
}
2021-10-26 13:25:54 +13:00
2021-11-01 08:19:25 +13:00
private fun onSubscriptionItemClick(subscription: Subscription) {
2021-11-04 05:48:13 +13:00
if (actionMode != null) {
handleActionModeClick(subscription)
} else {
startDetailView(subscription)
}
}
private fun onSubscriptionItemLongClick(subscription: Subscription) {
if (actionMode == null) {
beginActionMode(subscription)
}
}
2021-11-11 15:16:00 +13:00
private fun refreshAllSubscriptions() {
lifecycleScope.launch(Dispatchers.IO) {
val successFn = { notifications: List<Notification> ->
lifecycleScope.launch(Dispatchers.IO) {
notifications.forEach {
repository.addNotification(it)
}
}
Unit
}
repository.getAllSubscriptions().asFlow().collect { subscriptions ->
subscriptions.forEach { subscription ->
api.poll(subscription.id, subscription.baseUrl, subscription.topic, successFn, { _ -> })
}
}
}
}
2021-11-04 05:48:13 +13:00
private fun startDetailView(subscription: Subscription) {
2021-11-02 02:57:05 +13:00
Log.d(TAG, "Entering detail view for subscription $subscription")
2021-11-01 08:19:25 +13:00
val intent = Intent(this, DetailActivity::class.java)
intent.putExtra(EXTRA_SUBSCRIPTION_ID, subscription.id)
intent.putExtra(EXTRA_SUBSCRIPTION_BASE_URL, subscription.baseUrl)
intent.putExtra(EXTRA_SUBSCRIPTION_TOPIC, subscription.topic)
startActivityForResult(intent, REQUEST_CODE_DELETE_SUBSCRIPTION)
}
override fun onActivityResult(requestCode: Int, resultCode: Int, data: Intent?) {
if (requestCode == REQUEST_CODE_DELETE_SUBSCRIPTION && resultCode == RESULT_OK) {
val subscriptionId = data?.getLongExtra(EXTRA_SUBSCRIPTION_ID, 0)
val subscriptionTopic = data?.getStringExtra(EXTRA_SUBSCRIPTION_TOPIC)
2021-11-02 02:57:05 +13:00
Log.d(TAG, "Deleting subscription with subscription ID $subscriptionId (topic: $subscriptionTopic)")
2021-11-01 08:19:25 +13:00
subscriptionId?.let { id -> viewModel.remove(id) }
subscriptionTopic?.let { topic -> FirebaseMessaging.getInstance().unsubscribeFromTopic(topic) } // FIXME This only works for ntfy.sh
} else {
super.onActivityResult(requestCode, resultCode, data)
}
2021-10-27 05:23:41 +13:00
}
2021-10-27 06:46:49 +13:00
2021-11-04 05:48:13 +13:00
private fun handleActionModeClick(subscription: Subscription) {
adapter.toggleSelection(subscription.id)
if (adapter.selected.size == 0) {
finishActionMode()
} else {
actionMode!!.title = adapter.selected.size.toString()
redrawList()
}
}
override fun onCreateActionMode(mode: ActionMode?, menu: Menu?): Boolean {
this.actionMode = mode
if (mode != null) {
mode.menuInflater.inflate(R.menu.main_action_mode_menu, menu)
mode.title = "1" // One item selected
}
return true
}
override fun onPrepareActionMode(mode: ActionMode?, menu: Menu?): Boolean {
return false
}
override fun onActionItemClicked(mode: ActionMode?, item: MenuItem?): Boolean {
return when (item?.itemId) {
2021-11-04 06:56:08 +13:00
R.id.main_action_mode_delete -> {
2021-11-04 05:48:13 +13:00
onMultiDeleteClick()
true
}
else -> false
}
}
private fun onMultiDeleteClick() {
Log.d(DetailActivity.TAG, "Showing multi-delete dialog for selected items")
val builder = AlertDialog.Builder(this)
builder
.setMessage(R.string.main_action_mode_delete_dialog_message)
.setPositiveButton(R.string.main_action_mode_delete_dialog_permanently_delete) { _, _ ->
adapter.selected.map { viewModel.remove(it) }
finishActionMode()
}
.setNegativeButton(R.string.main_action_mode_delete_dialog_cancel) { _, _ ->
finishActionMode()
}
.create()
.show()
}
override fun onDestroyActionMode(mode: ActionMode?) {
endActionModeAndRedraw()
}
private fun beginActionMode(subscription: Subscription) {
actionMode = startActionMode(this)
adapter.selected.add(subscription.id)
redrawList()
// Fade out FAB
fab.alpha = 1f
fab
.animate()
.alpha(0f)
.setDuration(ANIMATION_DURATION)
.setListener(object : AnimatorListenerAdapter() {
override fun onAnimationEnd(animation: Animator) {
fab.visibility = View.GONE
}
})
// Fade status bar color
val fromColor = ContextCompat.getColor(this, R.color.primaryColor)
val toColor = ContextCompat.getColor(this, R.color.primaryDarkColor)
2021-11-04 06:56:08 +13:00
fadeStatusBarColor(window, fromColor, toColor)
2021-11-04 05:48:13 +13:00
}
private fun finishActionMode() {
actionMode!!.finish()
endActionModeAndRedraw()
}
private fun endActionModeAndRedraw() {
actionMode = null
adapter.selected.clear()
redrawList()
// Fade in FAB
fab.alpha = 0f
fab.visibility = View.VISIBLE
fab
.animate()
.alpha(1f)
.setDuration(ANIMATION_DURATION)
.setListener(object : AnimatorListenerAdapter() {
override fun onAnimationEnd(animation: Animator) {
fab.visibility = View.VISIBLE // Required to replace the old listener
}
})
// Fade status bar color
val fromColor = ContextCompat.getColor(this, R.color.primaryDarkColor)
val toColor = ContextCompat.getColor(this, R.color.primaryColor)
2021-11-04 06:56:08 +13:00
fadeStatusBarColor(window, fromColor, toColor)
2021-11-04 05:48:13 +13:00
}
private fun redrawList() {
mainList.adapter = adapter // Oh, what a hack ...
}
2021-10-30 14:13:58 +13:00
companion object {
const val TAG = "NtfyMainActivity"
2021-11-01 08:19:25 +13:00
const val EXTRA_SUBSCRIPTION_ID = "subscriptionId"
const val EXTRA_SUBSCRIPTION_BASE_URL = "subscriptionBaseUrl"
const val EXTRA_SUBSCRIPTION_TOPIC = "subscriptionTopic"
2021-11-02 03:49:52 +13:00
const val REQUEST_CODE_DELETE_SUBSCRIPTION = 1
2021-11-04 05:48:13 +13:00
const val ANIMATION_DURATION = 80L
2021-10-27 06:46:49 +13:00
}
}