/* Copyright 2017 Andrew Dawson * * 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.graphics.drawable.Drawable; import android.os.Bundle; import android.support.annotation.Nullable; import android.support.design.widget.Snackbar; import android.support.design.widget.TabLayout; import android.support.v7.widget.DividerItemDecoration; import android.support.v7.widget.LinearLayoutManager; import android.support.v7.widget.RecyclerView; import android.view.LayoutInflater; import android.view.View; import android.view.ViewGroup; import com.keylesspalace.tusky.entity.Account; import com.keylesspalace.tusky.entity.Relationship; import java.util.List; import retrofit2.Call; import retrofit2.Callback; import retrofit2.Response; public class AccountListFragment extends BaseFragment implements AccountActionListener { private static final String TAG = "AccountList"; // logging tag public enum Type { FOLLOWS, FOLLOWERS, BLOCKS, MUTES, FOLLOW_REQUESTS, } private Type type; private String accountId; private LinearLayoutManager layoutManager; private RecyclerView recyclerView; private EndlessOnScrollListener scrollListener; private AccountAdapter adapter; private TabLayout.OnTabSelectedListener onTabSelectedListener; private MastodonAPI api; public static AccountListFragment newInstance(Type type) { Bundle arguments = new Bundle(); AccountListFragment fragment = new AccountListFragment(); arguments.putSerializable("type", type); fragment.setArguments(arguments); return fragment; } public static AccountListFragment newInstance(Type type, String accountId) { Bundle arguments = new Bundle(); AccountListFragment fragment = new AccountListFragment(); arguments.putSerializable("type", type); arguments.putString("accountId", accountId); fragment.setArguments(arguments); return fragment; } @Override public void onCreate(@Nullable Bundle savedInstanceState) { super.onCreate(savedInstanceState); Bundle arguments = getArguments(); type = (Type) arguments.getSerializable("type"); accountId = arguments.getString("accountId"); api = null; } @Nullable @Override public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) { View rootView = inflater.inflate(R.layout.fragment_account_list, container, false); Context context = getContext(); recyclerView = (RecyclerView) rootView.findViewById(R.id.recycler_view); recyclerView.setHasFixedSize(true); layoutManager = new LinearLayoutManager(context); recyclerView.setLayoutManager(layoutManager); DividerItemDecoration divider = new DividerItemDecoration( context, layoutManager.getOrientation()); Drawable drawable = ThemeUtils.getDrawable(context, R.attr.status_divider_drawable, R.drawable.status_divider_dark); divider.setDrawable(drawable); recyclerView.addItemDecoration(divider); scrollListener = null; if (type == Type.BLOCKS) { adapter = new BlocksAdapter(this); } else if (type == Type.MUTES) { adapter = new MutesAdapter(this); } else if (type == Type.FOLLOW_REQUESTS) { adapter = new FollowRequestsAdapter(this); } else { adapter = new FollowAdapter(this); } recyclerView.setAdapter(adapter); if (jumpToTopAllowed()) { TabLayout layout = (TabLayout) getActivity().findViewById(R.id.tab_layout); onTabSelectedListener = new TabLayout.OnTabSelectedListener() { @Override public void onTabSelected(TabLayout.Tab tab) { } @Override public void onTabUnselected(TabLayout.Tab tab) { } @Override public void onTabReselected(TabLayout.Tab tab) { jumpToTop(); } }; layout.addOnTabSelectedListener(onTabSelectedListener); } return rootView; } @Override public void onActivityCreated(@Nullable Bundle savedInstanceState) { super.onActivityCreated(savedInstanceState); /* MastodonAPI on the base activity is only guaranteed to be initialised after the parent * activity is created, so everything needing to access the api object has to be delayed * until here. */ api = ((BaseActivity) getActivity()).mastodonAPI; scrollListener = new EndlessOnScrollListener(layoutManager) { @Override public void onLoadMore(int page, int totalItemsCount, RecyclerView view) { AccountAdapter adapter = (AccountAdapter) view.getAdapter(); Account account = adapter.getItem(adapter.getItemCount() - 2); if (account != null) { fetchAccounts(account.id, null); } else { fetchAccounts(); } } }; recyclerView.addOnScrollListener(scrollListener); } @Override public void onDestroyView() { if (jumpToTopAllowed()) { TabLayout tabLayout = (TabLayout) getActivity().findViewById(R.id.tab_layout); tabLayout.removeOnTabSelectedListener(onTabSelectedListener); } super.onDestroyView(); } private void fetchAccounts(final String fromId, String uptoId) { Callback> cb = new Callback>() { @Override public void onResponse(Call> call, Response> response) { if (response.isSuccessful()) { onFetchAccountsSuccess(response.body(), fromId); } else { onFetchAccountsFailure(new Exception(response.message())); } } @Override public void onFailure(Call> call, Throwable t) { onFetchAccountsFailure((Exception) t); } }; Call> listCall; switch (type) { default: case FOLLOWS: { listCall = api.accountFollowing(accountId, fromId, uptoId, null); break; } case FOLLOWERS: { listCall = api.accountFollowers(accountId, fromId, uptoId, null); break; } case BLOCKS: { listCall = api.blocks(fromId, uptoId, null); break; } case MUTES: { listCall = api.mutes(fromId, uptoId, null); break; } case FOLLOW_REQUESTS: { listCall = api.followRequests(fromId, uptoId, null); break; } } callList.add(listCall); listCall.enqueue(cb); } private void fetchAccounts() { fetchAccounts(null, null); } private static boolean findAccount(List accounts, String id) { for (Account account : accounts) { if (account.id.equals(id)) { return true; } } return false; } private void onFetchAccountsSuccess(List accounts, String fromId) { if (fromId != null) { if (accounts.size() > 0 && !findAccount(accounts, fromId)) { adapter.addItems(accounts); } } else { adapter.update(accounts); } } private void onFetchAccountsFailure(Exception exception) { Log.e(TAG, "Fetch failure: " + exception.getMessage()); } @Override public void onViewAccount(String id) { Intent intent = new Intent(getContext(), AccountActivity.class); intent.putExtra("id", id); startActivity(intent); } @Override public void onMute(final boolean mute, final String id, final int position) { if (api == null) { /* If somehow an unmute button is clicked after onCreateView but before * onActivityCreated, then this would get called with a null api object, so this eats * that input. */ Log.d(TAG, "MastodonAPI isn't initialised so this mute can't occur."); return; } Callback callback = new Callback() { @Override public void onResponse(Call call, Response response) { if (response.isSuccessful()) { onMuteSuccess(mute, id, position); } else { onMuteFailure(mute, id); } } @Override public void onFailure(Call call, Throwable t) { onMuteFailure(mute, id); } }; Call call; if (!mute) { call = api.unmuteAccount(id); } else { call = api.muteAccount(id); } callList.add(call); call.enqueue(callback); } private void onMuteSuccess(boolean muted, final String id, final int position) { if (muted) { return; } final MutesAdapter mutesAdapter = (MutesAdapter) adapter; final Account unmutedUser = mutesAdapter.removeItem(position); View.OnClickListener listener = new View.OnClickListener() { @Override public void onClick(View v) { mutesAdapter.addItem(unmutedUser, position); onMute(true, id, position); } }; Snackbar.make(recyclerView, R.string.confirmation_unmuted, Snackbar.LENGTH_LONG) .setAction(R.string.action_undo, listener) .show(); } private void onMuteFailure(boolean mute, String id) { String verb; if (mute) { verb = "mute"; } else { verb = "unmute"; } Log.e(TAG, String.format("Failed to %s account id %s", verb, id)); } @Override public void onBlock(final boolean block, final String id, final int position) { if (api == null) { /* If somehow an unblock button is clicked after onCreateView but before * onActivityCreated, then this would get called with a null api object, so this eats * that input. */ Log.d(TAG, "MastodonAPI isn't initialised so this block can't occur."); return; } Callback cb = new Callback() { @Override public void onResponse(Call call, Response response) { if (response.isSuccessful()) { onBlockSuccess(block, id, position); } else { onBlockFailure(block, id); } } @Override public void onFailure(Call call, Throwable t) { onBlockFailure(block, id); } }; Call call; if (!block) { call = api.unblockAccount(id); } else { call = api.blockAccount(id); } callList.add(call); call.enqueue(cb); } private void onBlockSuccess(boolean blocked, final String id, final int position) { if (blocked) { return; } final BlocksAdapter blocksAdapter = (BlocksAdapter) adapter; final Account unblockedUser = blocksAdapter.removeItem(position); View.OnClickListener listener = new View.OnClickListener() { @Override public void onClick(View v) { blocksAdapter.addItem(unblockedUser, position); onBlock(true, id, position); } }; Snackbar.make(recyclerView, R.string.confirmation_unblocked, Snackbar.LENGTH_LONG) .setAction(R.string.action_undo, listener) .show(); } private void onBlockFailure(boolean block, String id) { String verb; if (block) { verb = "block"; } else { verb = "unblock"; } Log.e(TAG, String.format("Failed to %s account id %s", verb, id)); } @Override public void onRespondToFollowRequest(final boolean accept, final String accountId, final int position) { if (api == null) { /* If somehow an response button is clicked after onCreateView but before * onActivityCreated, then this would get called with a null api object, so this eats * that input. */ Log.d(TAG, "MastodonAPI isn't initialised, so follow requests can't be responded to."); return; } Callback callback = new Callback() { @Override public void onResponse(Call call, Response response) { if (response.isSuccessful()) { onRespondToFollowRequestSuccess(position); } else { onRespondToFollowRequestFailure(accept, accountId); } } @Override public void onFailure(Call call, Throwable t) { onRespondToFollowRequestFailure(accept, accountId); } }; Call call; if (accept) { call = api.authorizeFollowRequest(accountId); } else { call = api.rejectFollowRequest(accountId); } callList.add(call); call.enqueue(callback); } private void onRespondToFollowRequestSuccess(int position) { FollowRequestsAdapter followRequestsAdapter = (FollowRequestsAdapter) adapter; followRequestsAdapter.removeItem(position); } private void onRespondToFollowRequestFailure(boolean accept, String accountId) { String verb = (accept) ? "accept" : "reject"; String message = String.format("Failed to %s account id %s.", verb, accountId); Log.e(TAG, message); } private boolean jumpToTopAllowed() { return type == Type.FOLLOWS || type == Type.FOLLOWERS; } private void jumpToTop() { layoutManager.scrollToPositionWithOffset(0, 0); scrollListener.reset(); } }