/* Copyright 2019 Tusky Contributors * * This file is a part of Tusky. * * 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. * * Tusky 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 Tusky; if not, * see . */ package com.keylesspalace.tusky import android.content.Context import android.content.Intent import android.os.Bundle import android.util.Log import android.view.Menu import android.view.MenuItem import androidx.fragment.app.commit import androidx.lifecycle.lifecycleScope import at.connyduck.calladapter.networkresult.fold import com.google.android.material.snackbar.Snackbar import com.keylesspalace.tusky.appstore.EventHub import com.keylesspalace.tusky.appstore.PreferenceChangedEvent import com.keylesspalace.tusky.components.timeline.TimelineFragment import com.keylesspalace.tusky.components.timeline.viewmodel.TimelineViewModel.Kind import com.keylesspalace.tusky.databinding.ActivityStatuslistBinding import com.keylesspalace.tusky.entity.Filter import com.keylesspalace.tusky.util.viewBinding import dagger.android.DispatchingAndroidInjector import dagger.android.HasAndroidInjector import kotlinx.coroutines.launch import javax.inject.Inject class StatusListActivity : BottomSheetActivity(), HasAndroidInjector { @Inject lateinit var dispatchingAndroidInjector: DispatchingAndroidInjector @Inject lateinit var eventHub: EventHub private val binding: ActivityStatuslistBinding by viewBinding(ActivityStatuslistBinding::inflate) private lateinit var kind: Kind private var hashtag: String? = null private var followTagItem: MenuItem? = null private var unfollowTagItem: MenuItem? = null private var muteTagItem: MenuItem? = null private var unmuteTagItem: MenuItem? = null /** The filter muting hashtag, null if unknown or hashtag is not filtered */ private var mutedFilter: Filter? = null override fun onCreate(savedInstanceState: Bundle?) { Log.d("StatusListActivity", "onCreate") super.onCreate(savedInstanceState) setContentView(binding.root) setSupportActionBar(binding.includedToolbar.toolbar) kind = Kind.valueOf(intent.getStringExtra(EXTRA_KIND)!!) val listId = intent.getStringExtra(EXTRA_LIST_ID) hashtag = intent.getStringExtra(EXTRA_HASHTAG) val title = when (kind) { Kind.FAVOURITES -> getString(R.string.title_favourites) Kind.BOOKMARKS -> getString(R.string.title_bookmarks) Kind.TAG -> getString(R.string.title_tag).format(hashtag) else -> intent.getStringExtra(EXTRA_LIST_TITLE) } supportActionBar?.run { setTitle(title) setDisplayHomeAsUpEnabled(true) setDisplayShowHomeEnabled(true) } if (supportFragmentManager.findFragmentById(R.id.fragmentContainer) == null) { supportFragmentManager.commit { val fragment = if (kind == Kind.TAG) { TimelineFragment.newHashtagInstance(listOf(hashtag!!)) } else { TimelineFragment.newInstance(kind, listId) } replace(R.id.fragmentContainer, fragment) } } } override fun onCreateOptionsMenu(menu: Menu): Boolean { val tag = hashtag if (kind == Kind.TAG && tag != null) { lifecycleScope.launch { mastodonApi.tag(tag).fold( { tagEntity -> menuInflater.inflate(R.menu.view_hashtag_toolbar, menu) followTagItem = menu.findItem(R.id.action_follow_hashtag) unfollowTagItem = menu.findItem(R.id.action_unfollow_hashtag) muteTagItem = menu.findItem(R.id.action_mute_hashtag) unmuteTagItem = menu.findItem(R.id.action_unmute_hashtag) followTagItem?.isVisible = tagEntity.following == false unfollowTagItem?.isVisible = tagEntity.following == true followTagItem?.setOnMenuItemClickListener { followTag() } unfollowTagItem?.setOnMenuItemClickListener { unfollowTag() } muteTagItem?.setOnMenuItemClickListener { muteTag() } unmuteTagItem?.setOnMenuItemClickListener { unmuteTag() } updateMuteTagMenuItems() }, { Log.w(TAG, "Failed to query tag #$tag", it) } ) } } return super.onCreateOptionsMenu(menu) } private fun followTag(): Boolean { val tag = hashtag if (tag != null) { lifecycleScope.launch { mastodonApi.followTag(tag).fold( { followTagItem?.isVisible = false unfollowTagItem?.isVisible = true }, { Snackbar.make(binding.root, getString(R.string.error_following_hashtag_format, tag), Snackbar.LENGTH_SHORT).show() Log.e(TAG, "Failed to follow #$tag", it) } ) } } return true } private fun unfollowTag(): Boolean { val tag = hashtag if (tag != null) { lifecycleScope.launch { mastodonApi.unfollowTag(tag).fold( { followTagItem?.isVisible = true unfollowTagItem?.isVisible = false }, { Snackbar.make(binding.root, getString(R.string.error_unfollowing_hashtag_format, tag), Snackbar.LENGTH_SHORT).show() Log.e(TAG, "Failed to unfollow #$tag", it) } ) } } return true } /** * Determine if the current hashtag is muted, and update the UI state accordingly. */ private fun updateMuteTagMenuItems() { val tag = hashtag ?: return muteTagItem?.isVisible = true muteTagItem?.isEnabled = false unmuteTagItem?.isVisible = false lifecycleScope.launch { mastodonApi.getFilters().fold( { filters -> for (filter in filters) { if ((tag == filter.phrase) and filter.context.contains(Filter.HOME)) { Log.d(TAG, "Tag $hashtag is filtered") muteTagItem?.isVisible = false unmuteTagItem?.isVisible = true mutedFilter = filter return@fold } } Log.d(TAG, "Tag $hashtag is not filtered") mutedFilter = null muteTagItem?.isEnabled = true muteTagItem?.isVisible = true muteTagItem?.isVisible = true }, { throwable -> Log.e(TAG, "Error getting filters: $throwable") } ) } } private fun muteTag(): Boolean { val tag = hashtag ?: return true lifecycleScope.launch { mastodonApi.createFilter( tag, listOf(Filter.HOME), irreversible = false, wholeWord = true, expiresInSeconds = null ).fold( { filter -> mutedFilter = filter muteTagItem?.isVisible = false unmuteTagItem?.isVisible = true eventHub.dispatch(PreferenceChangedEvent(filter.context[0])) }, { Snackbar.make(binding.root, getString(R.string.error_muting_hashtag_format, tag), Snackbar.LENGTH_SHORT).show() Log.e(TAG, "Failed to mute #$tag", it) } ) } return true } private fun unmuteTag(): Boolean { val filter = mutedFilter ?: return true lifecycleScope.launch { mastodonApi.deleteFilter(filter.id).fold( { muteTagItem?.isVisible = true unmuteTagItem?.isVisible = false eventHub.dispatch(PreferenceChangedEvent(filter.context[0])) mutedFilter = null }, { Snackbar.make(binding.root, getString(R.string.error_unmuting_hashtag_format, filter.phrase), Snackbar.LENGTH_SHORT).show() Log.e(TAG, "Failed to unmute #${filter.phrase}", it) } ) } return true } override fun androidInjector() = dispatchingAndroidInjector companion object { private const val EXTRA_KIND = "kind" private const val EXTRA_LIST_ID = "id" private const val EXTRA_LIST_TITLE = "title" private const val EXTRA_HASHTAG = "tag" const val TAG = "StatusListActivity" fun newFavouritesIntent(context: Context) = Intent(context, StatusListActivity::class.java).apply { putExtra(EXTRA_KIND, Kind.FAVOURITES.name) } fun newBookmarksIntent(context: Context) = Intent(context, StatusListActivity::class.java).apply { putExtra(EXTRA_KIND, Kind.BOOKMARKS.name) } fun newListIntent(context: Context, listId: String, listTitle: String) = Intent(context, StatusListActivity::class.java).apply { putExtra(EXTRA_KIND, Kind.LIST.name) putExtra(EXTRA_LIST_ID, listId) putExtra(EXTRA_LIST_TITLE, listTitle) } @JvmStatic fun newHashtagIntent(context: Context, hashtag: String) = Intent(context, StatusListActivity::class.java).apply { putExtra(EXTRA_KIND, Kind.TAG.name) putExtra(EXTRA_HASHTAG, hashtag) } } }