session-android/app/src/main/java/org/thoughtcrime/securesms/home/HomeActivity.kt

437 lines
19 KiB
Kotlin
Raw Normal View History

2021-07-09 03:14:21 +02:00
package org.thoughtcrime.securesms.home
2019-12-17 14:27:59 +01:00
2020-02-19 05:34:02 +01:00
import android.app.AlertDialog
2020-07-16 04:49:37 +02:00
import android.content.BroadcastReceiver
import android.content.Context
2019-12-17 15:15:13 +01:00
import android.content.Intent
2020-07-16 04:49:37 +02:00
import android.content.IntentFilter
2019-12-19 11:49:23 +01:00
import android.database.Cursor
2019-12-17 14:27:59 +01:00
import android.os.Bundle
import android.text.Spannable
import android.text.SpannableString
import android.text.style.ForegroundColorSpan
import android.view.View
2020-02-19 05:34:02 +01:00
import android.widget.Toast
2021-07-29 09:02:58 +02:00
import androidx.core.view.isVisible
import androidx.lifecycle.Observer
2020-11-23 06:59:44 +01:00
import androidx.lifecycle.lifecycleScope
import androidx.loader.app.LoaderManager
import androidx.loader.content.Loader
import androidx.localbroadcastmanager.content.LocalBroadcastManager
import androidx.recyclerview.widget.LinearLayoutManager
import dagger.hilt.android.AndroidEntryPoint
2019-12-17 14:27:59 +01:00
import kotlinx.android.synthetic.main.activity_home.*
import kotlinx.android.synthetic.main.seed_reminder_stub.*
2021-07-29 09:02:58 +02:00
import kotlinx.android.synthetic.main.seed_reminder_stub.view.*
2020-11-23 06:59:44 +01:00
import kotlinx.coroutines.Dispatchers
import kotlinx.coroutines.flow.*
2020-11-23 06:59:44 +01:00
import kotlinx.coroutines.launch
2021-07-29 09:02:58 +02:00
import kotlinx.coroutines.withContext
2019-12-17 14:27:59 +01:00
import network.loki.messenger.R
import org.greenrobot.eventbus.EventBus
import org.greenrobot.eventbus.Subscribe
import org.greenrobot.eventbus.ThreadMode
2021-03-17 01:30:03 +01:00
import org.session.libsession.messaging.jobs.JobQueue
import org.session.libsession.messaging.sending_receiving.MessageSender
import org.session.libsession.utilities.*
2021-07-09 05:18:48 +02:00
import org.session.libsession.utilities.Util
import org.session.libsignal.utilities.ThreadUtils
import org.session.libsignal.utilities.toHexString
2019-12-17 14:27:59 +01:00
import org.thoughtcrime.securesms.ApplicationContext
import org.thoughtcrime.securesms.MuteDialog
2019-12-17 14:27:59 +01:00
import org.thoughtcrime.securesms.PassphraseRequiredActionBarActivity
2021-05-31 06:29:11 +02:00
import org.thoughtcrime.securesms.conversation.v2.ConversationActivityV2
import org.thoughtcrime.securesms.conversation.v2.utilities.NotificationUtils
2021-07-08 05:38:14 +02:00
import org.thoughtcrime.securesms.crypto.IdentityKeyUtil
import org.thoughtcrime.securesms.database.GroupDatabase
import org.thoughtcrime.securesms.database.RecipientDatabase
import org.thoughtcrime.securesms.database.ThreadDatabase
2019-12-17 15:15:13 +01:00
import org.thoughtcrime.securesms.database.model.ThreadRecord
import org.thoughtcrime.securesms.dependencies.DatabaseComponent
2021-07-09 05:56:38 +02:00
import org.thoughtcrime.securesms.dms.CreatePrivateChatActivity
2021-07-09 03:14:21 +02:00
import org.thoughtcrime.securesms.groups.CreateClosedGroupActivity
2021-07-09 05:56:38 +02:00
import org.thoughtcrime.securesms.groups.JoinPublicChatActivity
2021-07-09 05:25:57 +02:00
import org.thoughtcrime.securesms.groups.OpenGroupManager
2019-12-19 11:15:58 +01:00
import org.thoughtcrime.securesms.mms.GlideApp
import org.thoughtcrime.securesms.mms.GlideRequests
2021-07-09 03:14:21 +02:00
import org.thoughtcrime.securesms.onboarding.SeedActivity
import org.thoughtcrime.securesms.onboarding.SeedReminderViewDelegate
2021-07-09 03:14:21 +02:00
import org.thoughtcrime.securesms.preferences.SettingsActivity
import org.thoughtcrime.securesms.util.*
2020-09-11 01:03:57 +02:00
import java.io.IOException
2021-07-08 05:38:14 +02:00
import java.util.*
import javax.inject.Inject
2019-12-17 14:27:59 +01:00
@AndroidEntryPoint
2021-05-27 07:00:16 +02:00
class HomeActivity : PassphraseRequiredActionBarActivity(), ConversationClickListener, SeedReminderViewDelegate, NewConversationButtonSetViewDelegate {
2019-12-19 11:15:58 +01:00
private lateinit var glide: GlideRequests
2020-07-16 04:49:37 +02:00
private var broadcastReceiver: BroadcastReceiver? = null
2019-12-17 14:27:59 +01:00
@Inject lateinit var threadDb: ThreadDatabase
@Inject lateinit var recipientDatabase: RecipientDatabase
@Inject lateinit var groupDatabase: GroupDatabase
2020-07-15 06:26:20 +02:00
private val publicKey: String
2021-05-27 07:00:16 +02:00
get() = TextSecurePreferences.getLocalNumber(this)!!
2020-01-06 04:26:52 +01:00
// region Lifecycle
2019-12-17 14:27:59 +01:00
override fun onCreate(savedInstanceState: Bundle?, isReady: Boolean) {
super.onCreate(savedInstanceState, isReady)
// Set content view
setContentView(R.layout.activity_home)
2020-01-07 02:00:30 +01:00
// Set custom toolbar
2020-01-06 02:07:55 +01:00
setSupportActionBar(toolbar)
2019-12-19 11:15:58 +01:00
// Set up Glide
glide = GlideApp.with(this)
2020-01-06 02:07:55 +01:00
// Set up toolbar buttons
profileButton.glide = glide
profileButton.setOnClickListener { openSettings() }
2020-09-08 08:21:50 +02:00
pathStatusViewContainer.disableClipping()
2020-05-29 03:16:52 +02:00
pathStatusViewContainer.setOnClickListener { showPath() }
// Set up seed reminder view
val hasViewedSeed = TextSecurePreferences.getHasViewedSeed(this)
2021-02-18 05:12:30 +01:00
if (!hasViewedSeed) {
2021-08-02 08:59:55 +02:00
seedReminderStub.inflate().apply {
2021-07-29 09:02:58 +02:00
val seedReminderView = this.seedReminderView
val seedReminderViewTitle = SpannableString("You're almost finished! 80%") // Intentionally not yet translated
seedReminderViewTitle.setSpan(ForegroundColorSpan(resources.getColorWithID(R.color.accent, theme)), 24, 27, Spannable.SPAN_EXCLUSIVE_EXCLUSIVE)
seedReminderView.title = seedReminderViewTitle
seedReminderView.subtitle = resources.getString(R.string.view_seed_reminder_subtitle_1)
seedReminderView.setProgress(80, false)
seedReminderView.delegate = this@HomeActivity
}
} else {
2021-07-29 09:02:58 +02:00
seedReminderStub.isVisible = false
}
2019-12-17 14:27:59 +01:00
// Set up recycler view
val cursor = threadDb.conversationList
2019-12-19 11:15:58 +01:00
val homeAdapter = HomeAdapter(this, cursor)
homeAdapter.setHasStableIds(true)
2019-12-19 11:15:58 +01:00
homeAdapter.glide = glide
homeAdapter.conversationClickListener = this
recyclerView.adapter = homeAdapter
2019-12-17 14:27:59 +01:00
recyclerView.layoutManager = LinearLayoutManager(this)
2020-04-20 03:54:56 +02:00
// Set up empty state view
createNewPrivateChatButton.setOnClickListener { createNewPrivateChat() }
IP2Country.configureIfNeeded(this@HomeActivity)
2019-12-19 11:49:23 +01:00
// This is a workaround for the fact that CursorRecyclerViewAdapter doesn't actually auto-update (even though it says it will)
LoaderManager.getInstance(this).restartLoader(0, null, object : LoaderManager.LoaderCallbacks<Cursor> {
override fun onCreateLoader(id: Int, bundle: Bundle?): Loader<Cursor> {
return HomeLoader(this@HomeActivity)
}
override fun onLoadFinished(loader: Loader<Cursor>, cursor: Cursor?) {
homeAdapter.changeCursor(cursor)
2020-04-20 03:54:56 +02:00
updateEmptyState()
2019-12-19 11:49:23 +01:00
}
override fun onLoaderReset(cursor: Loader<Cursor>) {
homeAdapter.changeCursor(null)
}
})
2020-03-16 05:35:14 +01:00
// Set up new conversation button set
newConversationButtonSet.delegate = this
2020-07-16 04:49:37 +02:00
// Observe blocked contacts changed events
val broadcastReceiver = object : BroadcastReceiver() {
override fun onReceive(context: Context, intent: Intent) {
recyclerView.adapter!!.notifyDataSetChanged()
}
}
this.broadcastReceiver = broadcastReceiver
LocalBroadcastManager.getInstance(this).registerReceiver(broadcastReceiver, IntentFilter("blockedContactsChanged"))
2021-07-30 03:00:53 +02:00
lifecycleScope.launchWhenStarted {
2021-07-29 09:02:58 +02:00
launch(Dispatchers.IO) {
// Double check that the long poller is up
(applicationContext as ApplicationContext).startPollingIfNeeded()
// update things based on TextSecurePrefs (profile info etc)
// Set up typing observer
withContext(Dispatchers.Main) {
ApplicationContext.getInstance(this@HomeActivity).typingStatusRepository.typingThreads.observe(this@HomeActivity, Observer<Set<Long>> { threadIDs ->
val adapter = recyclerView.adapter as HomeAdapter
adapter.typingThreadIDs = threadIDs ?: setOf()
})
updateProfileButton()
TextSecurePreferences.events.filter { it == TextSecurePreferences.PROFILE_NAME_PREF }.collect {
updateProfileButton()
}
}
// Set up remaining components if needed
val application = ApplicationContext.getInstance(this@HomeActivity)
application.registerForFCMIfNeeded(false)
val userPublicKey = TextSecurePreferences.getLocalNumber(this@HomeActivity)
if (userPublicKey != null) {
OpenGroupManager.startPolling()
JobQueue.shared.resumePendingJobs()
}
}
}
EventBus.getDefault().register(this@HomeActivity)
2019-12-17 14:27:59 +01:00
}
override fun onResume() {
super.onResume()
ApplicationContext.getInstance(this).messageNotifier.setHomeScreenVisible(true)
if (TextSecurePreferences.getLocalNumber(this) == null) { return; } // This can be the case after a secondary device is auto-cleared
IdentityKeyUtil.checkUpdate(this)
profileButton.recycle() // clear cached image before update tje profilePictureView
profileButton.update()
val hasViewedSeed = TextSecurePreferences.getHasViewedSeed(this)
2021-02-18 05:12:30 +01:00
if (hasViewedSeed) {
seedReminderView?.isVisible = false
}
if (TextSecurePreferences.getConfigurationMessageSynced(this)) {
lifecycleScope.launch(Dispatchers.IO) {
ConfigurationMessageUtilities.syncConfigurationIfNeeded(this@HomeActivity)
}
}
2021-01-13 06:13:49 +01:00
}
override fun onPause() {
super.onPause()
ApplicationContext.getInstance(this).messageNotifier.setHomeScreenVisible(false)
}
override fun onActivityResult(requestCode: Int, resultCode: Int, data: Intent?) {
super.onActivityResult(requestCode, resultCode, data)
2020-08-18 00:55:17 +02:00
if (resultCode == CreateClosedGroupActivity.closedGroupCreatedResultCode) {
createNewPrivateChat()
}
}
2020-07-16 04:49:37 +02:00
override fun onDestroy() {
val broadcastReceiver = this.broadcastReceiver
if (broadcastReceiver != null) {
LocalBroadcastManager.getInstance(this).unregisterReceiver(broadcastReceiver)
}
super.onDestroy()
EventBus.getDefault().unregister(this)
2020-07-16 04:49:37 +02:00
}
// endregion
2020-04-20 03:54:56 +02:00
// region Updating
private fun updateEmptyState() {
val threadCount = (recyclerView.adapter as HomeAdapter).itemCount
emptyStateContainer.visibility = if (threadCount == 0) View.VISIBLE else View.GONE
}
@Subscribe(threadMode = ThreadMode.MAIN)
fun onUpdateProfileEvent(event: ProfilePictureModifiedEvent) {
if (event.recipient.isLocalNumber) {
updateProfileButton()
}
}
private fun updateProfileButton() {
profileButton.publicKey = publicKey
profileButton.displayName = TextSecurePreferences.getProfileName(this)
profileButton.recycle()
profileButton.update()
}
2020-04-20 03:54:56 +02:00
// endregion
// region Interaction
override fun handleSeedReminderViewContinueButtonTapped() {
val intent = Intent(this, SeedActivity::class.java)
show(intent)
}
2019-12-17 15:15:13 +01:00
override fun onConversationClick(view: ConversationView) {
val thread = view.thread ?: return
openConversation(thread)
}
override fun onLongConversationClick(view: ConversationView) {
val thread = view.thread ?: return
val bottomSheet = ConversationOptionsBottomSheet()
bottomSheet.recipient = thread.recipient
2020-09-07 02:57:25 +02:00
bottomSheet.onViewDetailsTapped = {
bottomSheet.dismiss()
val userDetailsBottomSheet = UserDetailsBottomSheet()
2020-09-07 07:20:32 +02:00
val bundle = Bundle()
2021-01-15 05:36:30 +01:00
bundle.putString("publicKey", thread.recipient.address.toString())
2020-09-07 07:20:32 +02:00
userDetailsBottomSheet.arguments = bundle
2020-09-07 02:57:25 +02:00
userDetailsBottomSheet.show(supportFragmentManager, userDetailsBottomSheet.tag)
}
bottomSheet.onBlockTapped = {
bottomSheet.dismiss()
if (!thread.recipient.isBlocked) {
blockConversation(thread)
}
}
bottomSheet.onUnblockTapped = {
bottomSheet.dismiss()
if (thread.recipient.isBlocked) {
unblockConversation(thread)
}
}
bottomSheet.onDeleteTapped = {
bottomSheet.dismiss()
deleteConversation(thread)
}
bottomSheet.onSetMuteTapped = { muted ->
bottomSheet.dismiss()
setConversationMuted(thread, muted)
}
bottomSheet.onNotificationTapped = {
bottomSheet.dismiss()
NotificationUtils.showNotifyDialog(this, thread.recipient) { notifyType ->
setNotifyType(thread, notifyType)
}
}
bottomSheet.show(supportFragmentManager, bottomSheet.tag)
}
private fun blockConversation(thread: ThreadRecord) {
AlertDialog.Builder(this)
.setTitle(R.string.RecipientPreferenceActivity_block_this_contact_question)
.setMessage(R.string.RecipientPreferenceActivity_you_will_no_longer_receive_messages_and_calls_from_this_contact)
.setNegativeButton(android.R.string.cancel, null)
.setPositiveButton(R.string.RecipientPreferenceActivity_block) { dialog, _ ->
ThreadUtils.queue {
recipientDatabase.setBlocked(thread.recipient, true)
Util.runOnMain {
recyclerView.adapter!!.notifyDataSetChanged()
dialog.dismiss()
}
}
}.show()
}
private fun unblockConversation(thread: ThreadRecord) {
AlertDialog.Builder(this)
.setTitle(R.string.RecipientPreferenceActivity_unblock_this_contact_question)
.setMessage(R.string.RecipientPreferenceActivity_you_will_once_again_be_able_to_receive_messages_and_calls_from_this_contact)
.setNegativeButton(android.R.string.cancel, null)
.setPositiveButton(R.string.RecipientPreferenceActivity_unblock) { dialog, _ ->
ThreadUtils.queue {
recipientDatabase.setBlocked(thread.recipient, false)
Util.runOnMain {
recyclerView.adapter!!.notifyDataSetChanged()
dialog.dismiss()
}
}
}.show()
}
private fun setConversationMuted(thread: ThreadRecord, isMuted: Boolean) {
if (!isMuted) {
ThreadUtils.queue {
recipientDatabase.setMuted(thread.recipient, 0)
Util.runOnMain {
recyclerView.adapter!!.notifyDataSetChanged()
}
}
} else {
MuteDialog.show(this) { until: Long ->
ThreadUtils.queue {
recipientDatabase.setMuted(thread.recipient, until)
Util.runOnMain {
recyclerView.adapter!!.notifyDataSetChanged()
}
}
}
}
}
private fun setNotifyType(thread: ThreadRecord, newNotifyType: Int) {
ThreadUtils.queue {
recipientDatabase.setNotifyType(thread.recipient, newNotifyType)
Util.runOnMain {
recyclerView.adapter!!.notifyDataSetChanged()
}
}
}
private fun deleteConversation(thread: ThreadRecord) {
val threadID = thread.threadId
val recipient = thread.recipient
2021-05-14 05:13:02 +02:00
val message: String
2021-01-13 06:13:49 +01:00
if (recipient.isGroupRecipient) {
val group = groupDatabase.getGroup(recipient.address.toString()).orNull()
2021-01-15 05:36:30 +01:00
if (group != null && group.admins.map { it.toString() }.contains(TextSecurePreferences.getLocalNumber(this))) {
2021-05-14 05:13:02 +02:00
message = "Because you are the creator of this group it will be deleted for everyone. This cannot be undone."
2021-01-13 06:13:49 +01:00
} else {
2021-05-14 05:13:02 +02:00
message = resources.getString(R.string.activity_home_leave_group_dialog_message)
2021-01-13 06:13:49 +01:00
}
} else {
2021-05-14 05:13:02 +02:00
message = resources.getString(R.string.activity_home_delete_conversation_dialog_message)
2021-01-13 06:13:49 +01:00
}
val dialog = AlertDialog.Builder(this)
2021-05-14 05:13:02 +02:00
dialog.setMessage(message)
dialog.setPositiveButton(R.string.yes) { _, _ ->
lifecycleScope.launch(Dispatchers.Main) {
val context = this@HomeActivity as Context
2021-05-14 05:13:02 +02:00
// Cancel any outstanding jobs
DatabaseComponent.get(context).sessionJobDatabase().cancelPendingMessageSendJobs(threadID)
// Send a leave group message if this is an active closed group
if (recipient.address.isClosedGroup && DatabaseComponent.get(context).groupDatabase().isActive(recipient.address.toGroupString())) {
var isClosedGroup: Boolean
var groupPublicKey: String?
try {
groupPublicKey = GroupUtil.doubleDecodeGroupID(recipient.address.toString()).toHexString()
isClosedGroup = DatabaseComponent.get(context).lokiAPIDatabase().isClosedGroup(groupPublicKey)
} catch (e: IOException) {
groupPublicKey = null
isClosedGroup = false
}
if (isClosedGroup) {
2021-03-26 05:46:37 +01:00
MessageSender.explicitLeave(groupPublicKey!!, false)
}
}
2021-05-14 05:13:02 +02:00
// Delete the conversation
val v2OpenGroup = DatabaseComponent.get(this@HomeActivity).lokiThreadDatabase().getOpenGroupChat(threadID)
2021-05-21 07:02:34 +02:00
if (v2OpenGroup != null) {
2021-05-19 03:12:29 +02:00
OpenGroupManager.delete(v2OpenGroup.server, v2OpenGroup.room, this@HomeActivity)
2021-05-14 05:13:02 +02:00
} else {
2021-05-20 05:35:43 +02:00
ThreadUtils.queue {
threadDb.deleteConversation(threadID)
2021-05-20 05:35:43 +02:00
}
}
2021-05-14 05:13:02 +02:00
// Update the badge count
ApplicationContext.getInstance(context).messageNotifier.updateNotification(context)
// Notify the user
val toastMessage = if (recipient.isGroupRecipient) R.string.MessageRecord_left_group else R.string.activity_home_conversation_deleted_message
Toast.makeText(context, toastMessage, Toast.LENGTH_LONG).show()
}
}
dialog.setNegativeButton(R.string.no) { _, _ ->
// Do nothing
}
dialog.create().show()
2019-12-17 15:15:13 +01:00
}
private fun openConversation(thread: ThreadRecord) {
2021-05-31 06:29:11 +02:00
val intent = Intent(this, ConversationActivityV2::class.java)
intent.putExtra(ConversationActivityV2.THREAD_ID, thread.threadId)
2019-12-17 15:15:13 +01:00
push(intent)
}
2020-01-06 02:07:55 +01:00
private fun openSettings() {
2020-01-06 04:26:52 +01:00
val intent = Intent(this, SettingsActivity::class.java)
show(intent, isForResult = true)
2020-01-06 02:07:55 +01:00
}
2020-05-28 08:43:37 +02:00
private fun showPath() {
val intent = Intent(this, PathActivity::class.java)
show(intent)
}
override fun createNewPrivateChat() {
val intent = Intent(this, CreatePrivateChatActivity::class.java)
show(intent)
2019-12-17 15:15:13 +01:00
}
override fun createNewClosedGroup() {
2020-01-31 03:57:24 +01:00
val intent = Intent(this, CreateClosedGroupActivity::class.java)
show(intent, true)
2020-01-31 03:57:24 +01:00
}
override fun joinOpenGroup() {
val intent = Intent(this, JoinPublicChatActivity::class.java)
show(intent)
}
// endregion
2021-05-20 07:41:16 +02:00
}