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

314 lines
13 KiB
Kotlin
Raw Normal View History

2016-06-29 15:47:52 +02:00
/*
* Twidere - Twitter client for Android
*
* Copyright (C) 2012-2014 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
2017-04-22 09:18:44 +02:00
import android.accounts.AccountManager
2016-06-29 15:47:52 +02:00
import android.content.Context
import android.os.Bundle
2020-01-26 08:35:15 +01:00
import androidx.loader.app.LoaderManager.LoaderCallbacks
import androidx.loader.app.hasRunningLoadersSafe
import androidx.loader.content.Loader
import androidx.recyclerview.widget.LinearLayoutManager
import androidx.recyclerview.widget.RecyclerView
2016-06-29 15:47:52 +02:00
import android.view.KeyEvent
import androidx.loader.app.LoaderManager
import com.bumptech.glide.RequestManager
2016-06-29 15:47:52 +02:00
import com.squareup.otto.Subscribe
import kotlinx.android.synthetic.main.fragment_content_recyclerview.*
import org.mariotaku.commons.parcel.ParcelUtils
2017-01-20 15:08:42 +01:00
import org.mariotaku.kpreferences.get
2016-06-29 15:47:52 +02:00
import org.mariotaku.twidere.R
import org.mariotaku.twidere.adapter.ParcelableUsersAdapter
import org.mariotaku.twidere.adapter.decorator.ExtendedDividerItemDecoration
2016-06-29 15:47:52 +02:00
import org.mariotaku.twidere.adapter.iface.ILoadMoreSupportAdapter
import org.mariotaku.twidere.adapter.iface.IUsersAdapter
import org.mariotaku.twidere.adapter.iface.IUsersAdapter.UserClickListener
2017-04-22 09:18:44 +02:00
import org.mariotaku.twidere.annotation.AccountType
2017-04-21 11:23:55 +02:00
import org.mariotaku.twidere.constant.IntentConstants.*
2017-01-20 15:08:42 +01:00
import org.mariotaku.twidere.constant.newDocumentApiKey
2017-04-22 09:18:44 +02:00
import org.mariotaku.twidere.extension.model.getAccountType
2016-06-29 15:47:52 +02:00
import org.mariotaku.twidere.loader.iface.IExtendedLoader
2017-04-21 11:23:55 +02:00
import org.mariotaku.twidere.loader.iface.IPaginationLoader
import org.mariotaku.twidere.loader.users.AbsRequestUsersLoader
2017-04-21 11:40:33 +02:00
import org.mariotaku.twidere.model.ListResponse
2016-06-29 15:47:52 +02:00
import org.mariotaku.twidere.model.ParcelableUser
import org.mariotaku.twidere.model.UserKey
2017-02-09 09:00:12 +01:00
import org.mariotaku.twidere.model.event.FriendshipTaskEvent
2017-04-21 11:23:55 +02:00
import org.mariotaku.twidere.model.pagination.Pagination
2017-04-22 09:18:44 +02:00
import org.mariotaku.twidere.model.util.AccountUtils
2016-06-29 15:47:52 +02:00
import org.mariotaku.twidere.util.IntentUtils
import org.mariotaku.twidere.util.KeyboardShortcutsHandler
import org.mariotaku.twidere.util.KeyboardShortcutsHandler.KeyboardShortcutCallback
import org.mariotaku.twidere.util.RecyclerViewNavigationHelper
import org.mariotaku.twidere.view.holder.UserViewHolder
2016-12-06 06:15:22 +01:00
abstract class ParcelableUsersFragment : AbsContentListRecyclerViewFragment<ParcelableUsersAdapter>(),
2016-08-13 16:04:31 +02:00
LoaderCallbacks<List<ParcelableUser>?>, UserClickListener, KeyboardShortcutCallback,
IUsersAdapter.FriendshipClickListener {
2016-06-29 15:47:52 +02:00
2017-04-21 11:23:55 +02:00
override var refreshing: Boolean
get() {
if (context == null || isDetached) return false
return LoaderManager.getInstance(this).hasRunningLoadersSafe()
2017-04-21 11:23:55 +02:00
}
set(value) {
super.refreshing = value
}
protected open val simpleLayout: Boolean
2020-01-26 08:35:15 +01:00
get() = arguments?.getBoolean(EXTRA_SIMPLE_LAYOUT) ?: false
2016-06-29 15:47:52 +02:00
2017-04-21 11:23:55 +02:00
protected open val showFollow: Boolean
get() = true
protected var nextPagination: Pagination? = null
private set
protected var prevPagination: Pagination? = null
private set
private lateinit var navigationHelper: RecyclerViewNavigationHelper
private val usersBusCallback: Any
2016-06-29 15:47:52 +02:00
2016-12-06 06:15:22 +01:00
init {
2016-06-29 15:47:52 +02:00
usersBusCallback = createMessageBusCallback()
}
override fun onActivityCreated(savedInstanceState: Bundle?) {
super.onActivityCreated(savedInstanceState)
2017-04-21 11:23:55 +02:00
if (savedInstanceState != null) {
nextPagination = savedInstanceState.getParcelable(EXTRA_NEXT_PAGINATION)
prevPagination = savedInstanceState.getParcelable(EXTRA_PREV_PAGINATION)
}
2016-06-29 15:47:52 +02:00
adapter.userClickListener = this
navigationHelper = RecyclerViewNavigationHelper(recyclerView, layoutManager, adapter,
2016-06-29 15:47:52 +02:00
this)
val loaderArgs = Bundle(arguments)
2017-04-21 11:23:55 +02:00
loaderArgs.putBoolean(EXTRA_FROM_USER, true)
LoaderManager.getInstance(this).initLoader(0, loaderArgs, this)
2016-06-29 15:47:52 +02:00
}
override fun onStart() {
super.onStart()
bus.register(usersBusCallback)
}
override fun onStop() {
bus.unregister(usersBusCallback)
super.onStop()
}
2017-04-21 11:23:55 +02:00
override fun onSaveInstanceState(outState: Bundle) {
super.onSaveInstanceState(outState)
outState.putParcelable(EXTRA_NEXT_PAGINATION, nextPagination)
outState.putParcelable(EXTRA_PREV_PAGINATION, prevPagination)
}
2016-06-29 15:47:52 +02:00
2020-01-26 08:35:15 +01:00
override fun onCreateLoader(id: Int, args: Bundle?): Loader<List<ParcelableUser>?> {
val fromUser = args?.getBoolean(EXTRA_FROM_USER)
args?.remove(EXTRA_FROM_USER)
return onCreateUsersLoader(requireActivity(), args!!, fromUser!!).apply {
2017-04-21 11:23:55 +02:00
if (this is AbsRequestUsersLoader) {
pagination = args.getParcelable(EXTRA_PAGINATION)
}
}
2016-06-29 15:47:52 +02:00
}
override fun onLoadFinished(loader: Loader<List<ParcelableUser>?>, data: List<ParcelableUser>?) {
adapter.setData(data)
2016-08-13 16:04:31 +02:00
if (loader !is IExtendedLoader || loader.fromUser) {
2016-06-29 15:47:52 +02:00
adapter.loadMoreSupportedPosition = if (hasMoreData(data)) ILoadMoreSupportAdapter.END else ILoadMoreSupportAdapter.NONE
2016-07-03 08:42:28 +02:00
refreshEnabled = true
2016-06-29 15:47:52 +02:00
}
if (loader is IExtendedLoader) {
2016-08-13 16:04:31 +02:00
loader.fromUser = false
2016-06-29 15:47:52 +02:00
}
2019-10-24 17:52:11 +02:00
if (loader is IPaginationLoader && data?.loadSuccess() == true) {
2017-04-21 11:23:55 +02:00
nextPagination = loader.nextPagination
prevPagination = loader.prevPagination
}
2016-06-29 15:47:52 +02:00
showContent()
2016-07-03 08:42:28 +02:00
refreshEnabled = true
2016-06-29 15:47:52 +02:00
refreshing = false
setLoadMoreIndicatorPosition(ILoadMoreSupportAdapter.NONE)
}
2017-04-21 11:23:55 +02:00
override fun onLoaderReset(loader: Loader<List<ParcelableUser>?>) {
if (loader is IExtendedLoader) {
loader.fromUser = false
}
}
override fun onLoadMoreContents(@ILoadMoreSupportAdapter.IndicatorPosition position: Long) {
// Only supports load from end, skip START flag
if (position and ILoadMoreSupportAdapter.START != 0L) return
super.onLoadMoreContents(position)
if (position == 0L) return
val loaderArgs = Bundle(arguments)
loaderArgs.putBoolean(EXTRA_FROM_USER, true)
loaderArgs.putParcelable(EXTRA_PAGINATION, nextPagination)
LoaderManager.getInstance(this).restartLoader(0, loaderArgs, this)
2017-04-21 11:23:55 +02:00
}
override fun onCreateAdapter(context: Context, requestManager: RequestManager): ParcelableUsersAdapter {
val adapter = ParcelableUsersAdapter(context, this.requestManager)
2017-04-21 11:23:55 +02:00
adapter.simpleLayout = simpleLayout
adapter.showFollow = showFollow
2020-01-26 08:35:15 +01:00
val accountType = arguments?.getParcelable<UserKey?>(EXTRA_ACCOUNT_KEY)?.let { key ->
2017-04-22 09:18:44 +02:00
val am = AccountManager.get(context)
return@let AccountUtils.findByAccountKey(am, key)?.getAccountType(am)
}
when (accountType) {
AccountType.TWITTER, AccountType.FANFOU, AccountType.STATUSNET -> {
adapter.friendshipClickListener = this
}
else -> {
adapter.friendshipClickListener = null
}
}
2017-04-21 11:23:55 +02:00
return adapter
}
2016-08-13 16:04:31 +02:00
override fun handleKeyboardShortcutSingle(handler: KeyboardShortcutsHandler, keyCode: Int,
2017-03-01 15:12:25 +01:00
event: KeyEvent, metaState: Int): Boolean {
return navigationHelper.handleKeyboardShortcutSingle(handler, keyCode, event, metaState)
2016-06-29 15:47:52 +02:00
}
2016-08-13 16:04:31 +02:00
override fun handleKeyboardShortcutRepeat(handler: KeyboardShortcutsHandler, keyCode: Int,
2017-03-01 15:12:25 +01:00
repeatCount: Int, event: KeyEvent, metaState: Int): Boolean {
return navigationHelper.handleKeyboardShortcutRepeat(handler, keyCode, repeatCount, event, metaState)
2016-06-29 15:47:52 +02:00
}
2017-04-21 11:23:55 +02:00
2016-08-13 16:04:31 +02:00
override fun isKeyboardShortcutHandled(handler: KeyboardShortcutsHandler, keyCode: Int,
2017-03-01 15:12:25 +01:00
event: KeyEvent, metaState: Int): Boolean {
return navigationHelper.isKeyboardShortcutHandled(handler, keyCode, event, metaState)
2016-06-29 15:47:52 +02:00
}
override fun onUserClick(holder: UserViewHolder, position: Int) {
val user = adapter.getUser(position) ?: return
2020-01-26 08:35:15 +01:00
activity?.let { IntentUtils.openUserProfile(it, user, preferences[newDocumentApiKey]) }
2016-06-29 15:47:52 +02:00
}
override fun onFollowClicked(holder: UserViewHolder, position: Int) {
val user = adapter.getUser(position) ?: return
2017-09-08 13:40:12 +02:00
val accountKey = user.account_key ?: return
if (twitterWrapper.isUpdatingRelationship(accountKey, user.key)) return
2016-06-29 15:47:52 +02:00
if (user.is_following) {
parentFragmentManager.let { DestroyFriendshipDialogFragment.show(it, user) }
2016-06-29 15:47:52 +02:00
} else {
2017-09-08 13:40:12 +02:00
twitterWrapper.createFriendshipAsync(accountKey, user.key, user.screen_name)
2016-06-29 15:47:52 +02:00
}
}
2016-08-10 08:48:10 +02:00
override fun onUnblockClicked(holder: UserViewHolder, position: Int) {
val user = adapter.getUser(position) ?: return
2017-09-08 13:40:12 +02:00
val accountKey = user.account_key ?: return
if (twitterWrapper.isUpdatingRelationship(accountKey, user.key)) return
twitterWrapper.destroyBlockAsync(accountKey, user.key)
2016-08-10 08:48:10 +02:00
}
override fun onUnmuteClicked(holder: UserViewHolder, position: Int) {
val user = adapter.getUser(position) ?: return
2017-09-08 13:40:12 +02:00
val accountKey = user.account_key ?: return
if (twitterWrapper.isUpdatingRelationship(accountKey, user.key)) return
twitterWrapper.destroyMuteAsync(accountKey, user.key)
2016-08-10 08:48:10 +02:00
}
2017-04-07 08:09:00 +02:00
2016-06-29 15:47:52 +02:00
override fun onUserLongClick(holder: UserViewHolder, position: Int): Boolean {
return true
}
2017-04-21 11:23:55 +02:00
override fun onCreateItemDecoration(context: Context, recyclerView: RecyclerView,
2020-01-26 08:35:15 +01:00
layoutManager: LinearLayoutManager): RecyclerView.ItemDecoration? {
val itemDecoration = ExtendedDividerItemDecoration(context,
2016-06-29 15:47:52 +02:00
(recyclerView.layoutManager as LinearLayoutManager).orientation)
val res = context.resources
if (adapter.profileImageEnabled) {
val decorPaddingLeft = res.getDimensionPixelSize(R.dimen.element_spacing_normal) * 2 + res.getDimensionPixelSize(R.dimen.icon_size_status_profile_image)
itemDecoration.setPadding { position, rect ->
val itemViewType = adapter.getItemViewType(position)
var nextItemIsUser = false
if (position < adapter.itemCount - 1) {
nextItemIsUser = adapter.getItemViewType(position + 1) == ParcelableUsersAdapter.ITEM_VIEW_TYPE_USER
}
if (nextItemIsUser && itemViewType == ParcelableUsersAdapter.ITEM_VIEW_TYPE_USER) {
rect.left = decorPaddingLeft
} else {
rect.left = 0
}
true
}
}
itemDecoration.setDecorationEndOffset(1)
return itemDecoration
}
2017-04-21 11:23:55 +02:00
abstract fun onCreateUsersLoader(context: Context, args: Bundle, fromUser: Boolean):
Loader<List<ParcelableUser>?>
2016-06-29 15:47:52 +02:00
protected open fun shouldRemoveUser(position: Int, event: FriendshipTaskEvent): Boolean {
return false
}
protected fun hasMoreData(data: List<ParcelableUser>?): Boolean {
2020-06-08 23:07:20 +02:00
return data == null || data.isNotEmpty()
2016-06-29 15:47:52 +02:00
}
protected fun createMessageBusCallback(): Any {
return UsersBusCallback()
}
2017-04-21 11:23:55 +02:00
private fun findPosition(accountKey: UserKey, userKey: UserKey): Int {
return adapter.findPosition(accountKey, userKey)
}
2017-04-21 11:40:33 +02:00
private fun List<ParcelableUser>.loadSuccess(): Boolean {
return this !is ListResponse<*> || !this.hasException()
}
2016-06-29 15:47:52 +02:00
protected inner class UsersBusCallback {
@Subscribe
fun onFriendshipTaskEvent(event: FriendshipTaskEvent) {
val position = findPosition(event.accountKey, event.userKey)
val data = adapter.getData() ?: return
if (position < 0 || position >= data.size) return
if (shouldRemoveUser(position, event)) {
adapter.removeUserAt(position)
} else {
val adapterUser = data[position]
val eventUser = event.user
if (eventUser != null) {
if (adapterUser.account_key == eventUser.account_key) {
val clone = ParcelUtils.clone(eventUser)
clone.position = adapterUser.position
adapter.setUserAt(position, clone)
}
}
adapter.notifyItemChanged(position)
}
}
}
}