mirror of
https://github.com/nextcloud/talk-android
synced 2025-06-19 11:39:42 +01:00
With this change, all links that target files in the same nextcloud instance will be opened in the files app, no matter on which screen the link was. Signed-off-by: Marcel Hibbe <dev@mhibbe.de>
251 lines
9.0 KiB
Kotlin
251 lines
9.0 KiB
Kotlin
/*
|
|
* Nextcloud Talk - Android Client
|
|
*
|
|
* SPDX-FileCopyrightText: 2023 Ezhil Shanmugham <ezhil56x.contact@gmail.com>
|
|
* SPDX-FileCopyrightText: 2022 Álvaro Brey <alvaro@alvarobrey.com>
|
|
* SPDX-FileCopyrightText: 2022 Nextcloud GmbH
|
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
|
*/
|
|
package com.nextcloud.talk.messagesearch
|
|
|
|
import android.app.Activity
|
|
import android.content.Intent
|
|
import android.os.Bundle
|
|
import android.text.TextUtils
|
|
import android.view.Menu
|
|
import android.view.MenuItem
|
|
import android.view.View
|
|
import androidx.activity.OnBackPressedCallback
|
|
import androidx.appcompat.widget.SearchView
|
|
import androidx.lifecycle.ViewModelProvider
|
|
import autodagger.AutoInjector
|
|
import com.google.android.material.snackbar.Snackbar
|
|
import com.nextcloud.talk.R
|
|
import com.nextcloud.talk.activities.BaseActivity
|
|
import com.nextcloud.talk.adapters.items.LoadMoreResultsItem
|
|
import com.nextcloud.talk.adapters.items.MessageResultItem
|
|
import com.nextcloud.talk.application.NextcloudTalkApplication
|
|
import com.nextcloud.talk.conversationlist.ConversationsListActivity
|
|
import com.nextcloud.talk.data.user.model.User
|
|
import com.nextcloud.talk.databinding.ActivityMessageSearchBinding
|
|
import com.nextcloud.talk.utils.bundle.BundleKeys
|
|
import com.nextcloud.talk.utils.rx.SearchViewObservable.Companion.observeSearchView
|
|
import eu.davidea.flexibleadapter.FlexibleAdapter
|
|
import eu.davidea.flexibleadapter.items.AbstractFlexibleItem
|
|
import eu.davidea.viewholders.FlexibleViewHolder
|
|
import io.reactivex.Observable
|
|
import io.reactivex.android.schedulers.AndroidSchedulers
|
|
import io.reactivex.disposables.Disposable
|
|
import io.reactivex.schedulers.Schedulers
|
|
import java.util.concurrent.TimeUnit
|
|
import javax.inject.Inject
|
|
|
|
@AutoInjector(NextcloudTalkApplication::class)
|
|
class MessageSearchActivity : BaseActivity() {
|
|
|
|
@Inject
|
|
lateinit var viewModelFactory: ViewModelProvider.Factory
|
|
|
|
private lateinit var binding: ActivityMessageSearchBinding
|
|
private lateinit var searchView: SearchView
|
|
|
|
private lateinit var user: User
|
|
|
|
private lateinit var viewModel: MessageSearchViewModel
|
|
|
|
private var searchViewDisposable: Disposable? = null
|
|
private var adapter: FlexibleAdapter<AbstractFlexibleItem<*>>? = null
|
|
|
|
private val onBackPressedCallback = object : OnBackPressedCallback(true) {
|
|
override fun handleOnBackPressed() {
|
|
setResult(Activity.RESULT_CANCELED)
|
|
finish()
|
|
}
|
|
}
|
|
|
|
override fun onCreate(savedInstanceState: Bundle?) {
|
|
super.onCreate(savedInstanceState)
|
|
NextcloudTalkApplication.sharedApplication!!.componentApplication.inject(this)
|
|
|
|
binding = ActivityMessageSearchBinding.inflate(layoutInflater)
|
|
setupActionBar()
|
|
setContentView(binding.root)
|
|
setupSystemColors()
|
|
|
|
viewModel = ViewModelProvider(this, viewModelFactory)[MessageSearchViewModel::class.java]
|
|
user = currentUserProvider.currentUser.blockingGet()
|
|
val roomToken = intent.getStringExtra(BundleKeys.KEY_ROOM_TOKEN)!!
|
|
viewModel.initialize(roomToken)
|
|
setupStateObserver()
|
|
|
|
binding.swipeRefreshLayout.setOnRefreshListener {
|
|
viewModel.refresh(searchView.query?.toString())
|
|
}
|
|
|
|
onBackPressedDispatcher.addCallback(this, onBackPressedCallback)
|
|
}
|
|
|
|
private fun setupActionBar() {
|
|
setSupportActionBar(binding.messageSearchToolbar)
|
|
supportActionBar?.setDisplayHomeAsUpEnabled(true)
|
|
val conversationName = intent.getStringExtra(BundleKeys.KEY_CONVERSATION_NAME)
|
|
supportActionBar?.title = conversationName
|
|
viewThemeUtils.material.themeToolbar(binding.messageSearchToolbar)
|
|
}
|
|
|
|
private fun setupStateObserver() {
|
|
viewModel.state.observe(this) { state ->
|
|
when (state) {
|
|
MessageSearchViewModel.InitialState -> showInitial()
|
|
MessageSearchViewModel.EmptyState -> showEmpty()
|
|
is MessageSearchViewModel.LoadedState -> showLoaded(state)
|
|
MessageSearchViewModel.LoadingState -> showLoading()
|
|
MessageSearchViewModel.ErrorState -> showError()
|
|
is MessageSearchViewModel.FinishedState -> onFinish()
|
|
}
|
|
}
|
|
}
|
|
|
|
private fun showError() {
|
|
displayLoading(false)
|
|
Snackbar.make(binding.root, "Error while searching", Snackbar.LENGTH_SHORT).show()
|
|
}
|
|
|
|
private fun showLoading() {
|
|
displayLoading(true)
|
|
}
|
|
|
|
private fun displayLoading(loading: Boolean) {
|
|
binding.swipeRefreshLayout.isRefreshing = loading
|
|
}
|
|
|
|
private fun showLoaded(state: MessageSearchViewModel.LoadedState) {
|
|
displayLoading(false)
|
|
binding.emptyContainer.emptyListView.visibility = View.GONE
|
|
binding.messageSearchRecycler.visibility = View.VISIBLE
|
|
setAdapterItems(state)
|
|
}
|
|
|
|
private fun setAdapterItems(state: MessageSearchViewModel.LoadedState) {
|
|
val loadMoreItems = if (state.hasMore) {
|
|
listOf(LoadMoreResultsItem)
|
|
} else {
|
|
emptyList()
|
|
}
|
|
val newItems =
|
|
state.results.map { MessageResultItem(this, user, it, false, viewThemeUtils) } + loadMoreItems
|
|
|
|
if (adapter != null) {
|
|
adapter!!.updateDataSet(newItems)
|
|
} else {
|
|
createAdapter(newItems)
|
|
}
|
|
}
|
|
|
|
private fun createAdapter(items: List<AbstractFlexibleItem<out FlexibleViewHolder>>) {
|
|
adapter = FlexibleAdapter(items)
|
|
binding.messageSearchRecycler.adapter = adapter
|
|
adapter!!.addListener(object : FlexibleAdapter.OnItemClickListener {
|
|
override fun onItemClick(view: View?, position: Int): Boolean {
|
|
val item = adapter!!.getItem(position)
|
|
when (item?.itemViewType) {
|
|
LoadMoreResultsItem.VIEW_TYPE -> {
|
|
viewModel.loadMore()
|
|
}
|
|
MessageResultItem.VIEW_TYPE -> {
|
|
val messageItem = item as MessageResultItem
|
|
viewModel.selectMessage(messageItem.messageEntry)
|
|
}
|
|
}
|
|
return false
|
|
}
|
|
})
|
|
}
|
|
|
|
private fun onFinish() {
|
|
val state = viewModel.state.value
|
|
if (state is MessageSearchViewModel.FinishedState) {
|
|
val resultIntent = Intent().apply {
|
|
putExtra(RESULT_KEY_MESSAGE_ID, state.selectedMessageId)
|
|
}
|
|
setResult(Activity.RESULT_OK, resultIntent)
|
|
finish()
|
|
}
|
|
}
|
|
|
|
private fun showInitial() {
|
|
displayLoading(false)
|
|
binding.messageSearchRecycler.visibility = View.GONE
|
|
binding.emptyContainer.emptyListViewHeadline.text = getString(R.string.message_search_begin_typing)
|
|
binding.emptyContainer.emptyListView.visibility = View.VISIBLE
|
|
}
|
|
|
|
private fun showEmpty() {
|
|
displayLoading(false)
|
|
binding.messageSearchRecycler.visibility = View.GONE
|
|
binding.emptyContainer.emptyListViewHeadline.text = getString(R.string.message_search_begin_empty)
|
|
binding.emptyContainer.emptyListView.visibility = View.VISIBLE
|
|
}
|
|
|
|
override fun onCreateOptionsMenu(menu: Menu): Boolean {
|
|
menuInflater.inflate(R.menu.menu_search, menu)
|
|
return true
|
|
}
|
|
|
|
override fun onPrepareOptionsMenu(menu: Menu): Boolean {
|
|
val menuItem = menu.findItem(R.id.action_search)
|
|
searchView = menuItem.actionView as SearchView
|
|
setupSearchView()
|
|
menuItem.setOnActionExpandListener(object : MenuItem.OnActionExpandListener {
|
|
override fun onMenuItemActionExpand(item: MenuItem): Boolean {
|
|
searchView.requestFocus()
|
|
return true
|
|
}
|
|
|
|
override fun onMenuItemActionCollapse(item: MenuItem): Boolean {
|
|
onBackPressedDispatcher.onBackPressed()
|
|
return false
|
|
}
|
|
})
|
|
menuItem.expandActionView()
|
|
return true
|
|
}
|
|
|
|
private fun setupSearchView() {
|
|
searchView.queryHint = getString(R.string.message_search_hint)
|
|
searchViewDisposable = observeSearchView(searchView)
|
|
.debounce { query ->
|
|
when {
|
|
TextUtils.isEmpty(query) -> Observable.empty()
|
|
else -> Observable.timer(
|
|
ConversationsListActivity.SEARCH_DEBOUNCE_INTERVAL_MS.toLong(),
|
|
TimeUnit.MILLISECONDS
|
|
)
|
|
}
|
|
}
|
|
.distinctUntilChanged()
|
|
.subscribeOn(Schedulers.io())
|
|
.observeOn(AndroidSchedulers.mainThread())
|
|
.subscribe { newText -> viewModel.onQueryTextChange(newText) }
|
|
}
|
|
|
|
override fun onOptionsItemSelected(item: MenuItem): Boolean {
|
|
return when (item.itemId) {
|
|
android.R.id.home -> {
|
|
onBackPressedDispatcher.onBackPressed()
|
|
true
|
|
}
|
|
else -> super.onOptionsItemSelected(item)
|
|
}
|
|
}
|
|
|
|
override fun onDestroy() {
|
|
super.onDestroy()
|
|
searchViewDisposable?.dispose()
|
|
}
|
|
|
|
companion object {
|
|
const val RESULT_KEY_MESSAGE_ID = "MessageSearchActivity.result.message"
|
|
}
|
|
}
|