Twidere-App-Android-Twitter.../twidere/src/main/kotlin/org/mariotaku/twidere/fragment/message/MessagesEntriesFragment.kt

266 lines
11 KiB
Kotlin
Raw Normal View History

2017-02-15 09:11:11 +01:00
/*
* Twidere - Twitter client for Android
*
* Copyright (C) 2012-2017 Mariotaku Lee <mariotaku.lee@gmail.com>
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
package org.mariotaku.twidere.fragment.message
2016-07-08 09:52:32 +02:00
2017-10-03 11:06:04 +02:00
import android.app.Activity
2017-02-09 14:43:51 +01:00
import android.content.Context
2017-02-15 07:55:18 +01:00
import android.content.Intent
2017-02-09 14:43:51 +01:00
import android.os.Bundle
2020-01-26 08:35:15 +01:00
import androidx.loader.app.LoaderManager.LoaderCallbacks
import androidx.loader.content.Loader
2017-03-19 06:07:45 +01:00
import android.view.ContextMenu
import android.view.MenuInflater
import android.view.MenuItem
import android.view.View
import androidx.loader.app.LoaderManager
import com.bumptech.glide.RequestManager
2017-02-10 18:03:40 +01:00
import com.squareup.otto.Subscribe
2017-03-19 06:07:45 +01:00
import kotlinx.android.synthetic.main.activity_premium_dashboard.*
import org.mariotaku.abstask.library.TaskStarter
2017-02-09 14:43:51 +01:00
import org.mariotaku.kpreferences.get
2017-02-10 19:01:31 +01:00
import org.mariotaku.ktextension.toStringArray
import org.mariotaku.sqliteqb.library.*
import org.mariotaku.sqliteqb.library.Columns.Column
2017-02-09 14:43:51 +01:00
import org.mariotaku.twidere.R
2017-02-15 07:55:18 +01:00
import org.mariotaku.twidere.TwidereConstants.EXTRA_ACCOUNT_KEYS
import org.mariotaku.twidere.TwidereConstants.REQUEST_SELECT_ACCOUNT
import org.mariotaku.twidere.activity.AccountSelectorActivity
2017-02-10 07:17:48 +01:00
import org.mariotaku.twidere.adapter.MessagesEntriesAdapter
2017-02-15 07:48:10 +01:00
import org.mariotaku.twidere.adapter.MessagesEntriesAdapter.MessageConversationClickListener
2017-02-10 12:55:00 +01:00
import org.mariotaku.twidere.adapter.iface.ILoadMoreSupportAdapter
2017-10-03 11:06:04 +02:00
import org.mariotaku.twidere.annotation.AccountType
import org.mariotaku.twidere.constant.IntentConstants.EXTRA_ACCOUNT_KEY
import org.mariotaku.twidere.constant.IntentConstants.EXTRA_ACCOUNT_TYPES
2017-03-19 06:07:45 +01:00
import org.mariotaku.twidere.constant.nameFirstKey
2017-02-09 14:43:51 +01:00
import org.mariotaku.twidere.constant.newDocumentApiKey
2017-03-19 06:07:45 +01:00
import org.mariotaku.twidere.extension.model.getTitle
2017-02-09 14:43:51 +01:00
import org.mariotaku.twidere.extension.model.user
2017-02-15 09:11:11 +01:00
import org.mariotaku.twidere.fragment.AbsContentListRecyclerViewFragment
2017-02-15 07:48:10 +01:00
import org.mariotaku.twidere.fragment.iface.IFloatingActionButtonFragment
import org.mariotaku.twidere.fragment.iface.IFloatingActionButtonFragment.ActionInfo
2017-02-09 14:43:51 +01:00
import org.mariotaku.twidere.loader.ObjectCursorLoader
import org.mariotaku.twidere.model.ParcelableMessageConversation
import org.mariotaku.twidere.model.UserKey
2017-02-10 18:03:40 +01:00
import org.mariotaku.twidere.model.event.GetMessagesTaskEvent
import org.mariotaku.twidere.provider.TwidereDataStore.Messages
2017-02-09 14:43:51 +01:00
import org.mariotaku.twidere.provider.TwidereDataStore.Messages.Conversations
2017-02-16 07:50:25 +01:00
import org.mariotaku.twidere.task.twitter.message.GetMessagesTask
2017-03-19 06:07:45 +01:00
import org.mariotaku.twidere.task.twitter.message.MarkMessageReadTask
import org.mariotaku.twidere.util.*
2017-02-09 14:43:51 +01:00
import org.mariotaku.twidere.util.Utils
2017-03-19 06:07:45 +01:00
import org.mariotaku.twidere.view.ExtendedRecyclerView
2016-07-08 09:52:32 +02:00
/**
* Created by mariotaku on 16/3/28.
*/
2017-02-10 07:17:48 +01:00
class MessagesEntriesFragment : AbsContentListRecyclerViewFragment<MessagesEntriesAdapter>(),
2017-02-15 07:48:10 +01:00
LoaderCallbacks<List<ParcelableMessageConversation>?>, MessageConversationClickListener,
IFloatingActionButtonFragment {
2017-02-09 14:43:51 +01:00
2017-02-10 19:01:31 +01:00
private val accountKeys: Array<UserKey> by lazy {
Utils.getAccountKeys(requireContext(), arguments) ?: DataStoreUtils.getActivatedAccountKeys(requireContext())
2017-02-10 19:01:31 +01:00
}
2017-02-09 14:43:51 +01:00
private val errorInfoKey: String = ErrorInfoStore.KEY_DIRECT_MESSAGES
2017-04-13 09:53:22 +02:00
private val loaderId: Int
get() = tabId.toInt().coerceIn(0..Int.MAX_VALUE)
2017-02-09 14:43:51 +01:00
override fun onActivityCreated(savedInstanceState: Bundle?) {
super.onActivityCreated(savedInstanceState)
adapter.listener = this
2017-02-10 12:55:00 +01:00
adapter.loadMoreSupportedPosition = ILoadMoreSupportAdapter.END
LoaderManager.getInstance(this).initLoader(loaderId, null, this)
2017-03-19 06:07:45 +01:00
registerForContextMenu(recyclerView)
2017-02-09 14:43:51 +01:00
}
2017-02-10 18:03:40 +01:00
override fun onStart() {
super.onStart()
bus.register(this)
}
override fun onStop() {
bus.unregister(this)
super.onStop()
}
2017-10-03 11:06:04 +02:00
override fun onActivityResult(requestCode: Int, resultCode: Int, data: Intent?) {
when (requestCode) {
REQUEST_SELECT_ACCOUNT -> {
if (resultCode != Activity.RESULT_OK) return
val accountKey = data?.getParcelableExtra<UserKey>(EXTRA_ACCOUNT_KEY) ?: return
2017-10-03 11:06:04 +02:00
startActivity(IntentUtils.newMessageConversation(accountKey))
}
else -> {
super.onActivityResult(requestCode, resultCode, data)
}
}
}
2017-02-09 14:43:51 +01:00
override fun onCreateLoader(id: Int, args: Bundle?): Loader<List<ParcelableMessageConversation>?> {
2017-03-05 09:08:09 +01:00
val loader = ObjectCursorLoader(context, ParcelableMessageConversation::class.java)
val projection = (Conversations.COLUMNS + Conversations.UNREAD_COUNT).map {
2017-02-17 14:21:07 +01:00
TwidereQueryBuilder.mapConversationsProjection(it)
}.toTypedArray()
val qb = SQLQueryBuilder.select(Columns(*projection))
qb.from(Table(Conversations.TABLE_NAME))
qb.join(Join(false, Join.Operation.LEFT_OUTER, Table(Messages.TABLE_NAME),
2017-04-10 11:35:35 +02:00
Expression.and(
Expression.equals(
Column(Table(Conversations.TABLE_NAME), Conversations.CONVERSATION_ID),
Column(Table(Messages.TABLE_NAME), Messages.CONVERSATION_ID)
),
Expression.equals(
Column(Table(Conversations.TABLE_NAME), Conversations.ACCOUNT_KEY),
Column(Table(Messages.TABLE_NAME), Messages.ACCOUNT_KEY))
)
))
qb.where(Expression.inArgs(Column(Table(Conversations.TABLE_NAME), Conversations.ACCOUNT_KEY), accountKeys.size))
qb.groupBy(Column(Table(Messages.TABLE_NAME), Messages.CONVERSATION_ID))
qb.orderBy(OrderBy(arrayOf(Conversations.LOCAL_TIMESTAMP, Conversations.SORT_ID), booleanArrayOf(false, false)))
2017-02-17 11:42:39 +01:00
loader.uri = TwidereQueryBuilder.rawQuery(qb.buildSQL(), Conversations.CONTENT_URI)
2017-02-10 19:01:31 +01:00
loader.selectionArgs = accountKeys.toStringArray()
2017-04-28 15:44:45 +02:00
loader.isUseCache = false
2017-02-09 14:43:51 +01:00
return loader
}
2020-01-26 08:35:15 +01:00
override fun onLoaderReset(loader: Loader<List<ParcelableMessageConversation>?>) {
2017-02-09 14:43:51 +01:00
adapter.conversations = null
}
2020-01-26 08:35:15 +01:00
override fun onLoadFinished(loader: Loader<List<ParcelableMessageConversation>?>, data: List<ParcelableMessageConversation>?) {
2017-02-09 14:43:51 +01:00
adapter.conversations = data
adapter.drawAccountColors = accountKeys.size > 1
2017-02-10 19:01:31 +01:00
setLoadMoreIndicatorPosition(ILoadMoreSupportAdapter.NONE)
2017-02-09 14:43:51 +01:00
showContentOrError()
}
override fun onCreateAdapter(context: Context, requestManager: RequestManager): MessagesEntriesAdapter {
return MessagesEntriesAdapter(context, this.requestManager)
2017-02-09 14:43:51 +01:00
}
override fun triggerRefresh(): Boolean {
super.triggerRefresh()
twitterWrapper.getMessagesAsync(object : GetMessagesTask.RefreshNewTaskParam(requireContext()) {
2017-02-14 17:26:48 +01:00
override val accountKeys: Array<UserKey> = this@MessagesEntriesFragment.accountKeys
2017-02-09 14:43:51 +01:00
})
return true
}
2017-02-10 12:55:00 +01:00
override fun onLoadMoreContents(position: Long) {
2017-02-10 19:01:31 +01:00
if (position != ILoadMoreSupportAdapter.END) {
return
}
2020-01-26 08:35:15 +01:00
val context = context ?: return
2017-02-10 19:01:31 +01:00
setLoadMoreIndicatorPosition(ILoadMoreSupportAdapter.END)
2017-02-14 17:26:48 +01:00
twitterWrapper.getMessagesAsync(object : GetMessagesTask.LoadMoreEntriesTaskParam(context) {
override val accountKeys: Array<UserKey> = this@MessagesEntriesFragment.accountKeys
2017-02-10 19:01:31 +01:00
})
2017-02-10 12:55:00 +01:00
}
2017-02-09 14:43:51 +01:00
override fun onConversationClick(position: Int) {
2020-01-26 08:35:15 +01:00
val context = context ?: return
2017-04-28 15:44:45 +02:00
val conversation = adapter.getConversation(position)
2017-02-09 14:43:51 +01:00
IntentUtils.openMessageConversation(context, conversation.account_key, conversation.id)
}
2017-03-19 06:07:45 +01:00
override fun onConversationLongClick(position: Int): Boolean {
2020-01-26 08:35:15 +01:00
val view = recyclerView.layoutManager?.findViewByPosition(position) ?: return false
2017-03-19 06:07:45 +01:00
recyclerView.showContextMenuForChild(view)
return true
}
2017-02-09 14:43:51 +01:00
override fun onProfileImageClick(position: Int) {
2020-01-26 08:35:15 +01:00
val context = context ?: return
2017-04-28 15:44:45 +02:00
val conversation = adapter.getConversation(position)
2017-02-09 14:43:51 +01:00
val user = conversation.user ?: return
IntentUtils.openUserProfile(context, user, preferences[newDocumentApiKey])
}
2017-02-15 07:48:10 +01:00
override fun getActionInfo(tag: String): ActionInfo? {
return ActionInfo(R.drawable.ic_action_add, getString(R.string.new_direct_message))
}
2017-02-15 09:11:11 +01:00
override fun onActionClick(tag: String): Boolean {
2017-02-15 07:48:10 +01:00
val accountKey = accountKeys.singleOrNull() ?: run {
2017-02-15 07:55:18 +01:00
val selectIntent = Intent(context, AccountSelectorActivity::class.java)
selectIntent.putExtra(EXTRA_ACCOUNT_KEYS, accountKeys)
2017-10-03 11:06:04 +02:00
selectIntent.putExtra(EXTRA_ACCOUNT_TYPES, arrayOf(AccountType.TWITTER,
AccountType.FANFOU))
2017-02-15 07:55:18 +01:00
startActivityForResult(selectIntent, REQUEST_SELECT_ACCOUNT)
2017-02-15 09:11:11 +01:00
return true
2017-02-15 07:48:10 +01:00
}
startActivity(IntentUtils.newMessageConversation(accountKey))
2017-02-15 09:11:11 +01:00
return true
2017-02-15 07:48:10 +01:00
}
2017-03-19 06:07:45 +01:00
override fun onCreateContextMenu(menu: ContextMenu, v: View, menuInfo: ContextMenu.ContextMenuInfo?) {
if (!userVisibleHint || menuInfo == null) return
2020-01-26 08:35:15 +01:00
val context = context ?: return
2017-03-19 06:07:45 +01:00
val info = menuInfo as? ExtendedRecyclerView.ContextMenuInfo ?: return
2017-04-28 15:44:45 +02:00
val conversation = adapter.getConversation(info.position)
2017-03-19 06:07:45 +01:00
val inflater = MenuInflater(context)
inflater.inflate(R.menu.context_message_entry, menu)
menu.setHeaderTitle(conversation.getTitle(context, userColorNameManager,
preferences[nameFirstKey]).first)
}
override fun onContextItemSelected(item: MenuItem): Boolean {
if (!userVisibleHint) return false
2020-01-26 08:35:15 +01:00
val context = context ?: return false
2017-03-19 06:07:45 +01:00
val menuInfo = item.menuInfo as? ExtendedRecyclerView.ContextMenuInfo ?: return false
when (item.itemId) {
R.id.mark_read -> {
2017-04-28 15:44:45 +02:00
val conversation = adapter.getConversation(menuInfo.position)
2017-03-19 06:07:45 +01:00
TaskStarter.execute(MarkMessageReadTask(context, conversation.account_key,
conversation.id))
return true
}
}
return super.onContextItemSelected(item)
}
2017-02-10 18:03:40 +01:00
@Subscribe
fun onGetMessagesTaskEvent(event: GetMessagesTaskEvent) {
if (!event.running) {
refreshing = false
}
}
2017-02-09 14:43:51 +01:00
private fun showContentOrError() {
val accountKeys = this.accountKeys
2020-01-26 08:35:15 +01:00
val context = context ?: return
2017-02-09 14:43:51 +01:00
if (adapter.itemCount > 0) {
showContent()
} else if (accountKeys.isNotEmpty()) {
val errorInfo = ErrorInfoStore.getErrorInfo(context,
errorInfoStore[errorInfoKey, accountKeys[0]])
if (errorInfo != null) {
showEmpty(errorInfo.icon, errorInfo.message)
} else {
showEmpty(R.drawable.ic_info_refresh, getString(R.string.swipe_down_to_refresh))
}
} else {
showError(R.drawable.ic_info_accounts, getString(R.string.message_toast_no_account_selected))
}
}
2017-02-10 18:03:40 +01:00
2017-02-09 14:43:51 +01:00
}