package de.danoeh.antennapod.dialog; import android.app.AlertDialog; import android.content.res.TypedArray; import android.graphics.drawable.Drawable; import android.os.Bundle; import android.support.annotation.IdRes; import android.support.annotation.NonNull; import android.support.annotation.PluralsRes; import android.support.annotation.StringRes; import android.support.design.widget.Snackbar; import android.support.v4.app.ActivityCompat; import android.support.v4.app.Fragment; import android.support.v4.util.ArrayMap; import android.support.v7.app.ActionBar; import android.support.v7.app.AppCompatActivity; import android.util.Log; import android.view.LayoutInflater; import android.view.Menu; import android.view.MenuInflater; import android.view.MenuItem; import android.view.View; import android.view.ViewGroup; import android.widget.ArrayAdapter; import android.widget.ListView; import com.leinardi.android.speeddial.SpeedDialView; import java.util.ArrayList; import java.util.Arrays; import java.util.Collections; import java.util.List; import java.util.Map; import de.danoeh.antennapod.R; import de.danoeh.antennapod.core.dialog.DownloadRequestErrorDialogCreator; import de.danoeh.antennapod.core.feed.FeedItem; import de.danoeh.antennapod.core.storage.DBTasks; import de.danoeh.antennapod.core.storage.DBWriter; import de.danoeh.antennapod.core.storage.DownloadRequestException; import de.danoeh.antennapod.core.util.LongList; public class EpisodesApplyActionFragment extends Fragment { public static final String TAG = "EpisodeActionFragment"; public static final int ACTION_ADD_TO_QUEUE = 1; public static final int ACTION_REMOVE_FROM_QUEUE = 2; private static final int ACTION_MARK_PLAYED = 4; private static final int ACTION_MARK_UNPLAYED = 8; private static final int ACTION_DOWNLOAD = 16; public static final int ACTION_DELETE = 32; private static final int ACTION_ALL = ACTION_ADD_TO_QUEUE | ACTION_REMOVE_FROM_QUEUE | ACTION_MARK_PLAYED | ACTION_MARK_UNPLAYED | ACTION_DOWNLOAD | ACTION_DELETE; /** * Specify an action (defined by #flag) 's UI bindings. * * Includes: the menu / action item and the actual logic */ private static class ActionBinding { int flag; @IdRes final int actionItemId; @NonNull final Runnable action; ActionBinding(int flag, @IdRes int actionItemId, @NonNull Runnable action) { this.flag = flag; this.actionItemId = actionItemId; this.action = action; } } private final List actionBindings; private ListView mListView; private ArrayAdapter mAdapter; private SpeedDialView mSpeedDialView; @NonNull private CharSequence actionBarTitleOriginal = ""; private final Map idMap = new ArrayMap<>(); private final List episodes = new ArrayList<>(); private int actions; private final List titles = new ArrayList<>(); private final LongList checkedIds = new LongList(); private MenuItem mSelectToggle; public EpisodesApplyActionFragment() { actionBindings = Arrays.asList( new ActionBinding(ACTION_ADD_TO_QUEUE, R.id.add_to_queue_batch, this::queueChecked), new ActionBinding(ACTION_REMOVE_FROM_QUEUE, R.id.remove_from_queue_batch, this::removeFromQueueChecked), new ActionBinding(ACTION_MARK_PLAYED, R.id.mark_read_batch, this::markedCheckedPlayed), new ActionBinding(ACTION_MARK_UNPLAYED, R.id.mark_unread_batch, this::markedCheckedUnplayed), new ActionBinding(ACTION_DOWNLOAD, R.id.download_batch, this::downloadChecked), new ActionBinding(ACTION_DELETE, R.id.delete_batch, this::deleteChecked) ); } public static EpisodesApplyActionFragment newInstance(List items) { return newInstance(items, ACTION_ALL); } public static EpisodesApplyActionFragment newInstance(List items, int actions) { EpisodesApplyActionFragment f = new EpisodesApplyActionFragment(); f.episodes.addAll(items); for(FeedItem episode : items) { f.idMap.put(episode.getId(), episode); } f.actions = actions; return f; } @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setRetainInstance(true); setHasOptionsMenu(true); } @Override public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) { View view = inflater.inflate(R.layout.episodes_apply_action_fragment, container, false); mListView = view.findViewById(android.R.id.list); mListView.setChoiceMode(ListView.CHOICE_MODE_MULTIPLE); mListView.setOnItemClickListener((ListView, view1, position, rowId) -> { long id = episodes.get(position).getId(); if (checkedIds.contains(id)) { checkedIds.remove(id); } else { checkedIds.add(id); } refreshCheckboxes(); }); mListView.setOnItemLongClickListener((adapterView, view12, position, id) -> { new AlertDialog.Builder(getActivity()) .setItems(R.array.batch_long_press_options, (dialogInterface, item) -> { int direction; if (item == 0) { direction = -1; } else { direction = 1; } int currentPosition = position + direction; while (currentPosition >= 0 && currentPosition < episodes.size()) { long id1 = episodes.get(currentPosition).getId(); if (!checkedIds.contains(id1)) { checkedIds.add(id1); } currentPosition += direction; } refreshCheckboxes(); }).show(); return true; }); titles.clear(); for (FeedItem episode : episodes) { titles.add(episode.getTitle()); } mAdapter = new ArrayAdapter<>(getActivity(), R.layout.simple_list_item_multiple_choice_on_start, titles); mListView.setAdapter(mAdapter); saveActionBarTitle(); // needed when we dynamically change the title based on selection // Init action UI (via a FAB Speed Dial) mSpeedDialView = view.findViewById(R.id.fabSD); mSpeedDialView.inflate(R.menu.episodes_apply_action_speeddial); // show only specified actions, and bind speed dial UIs to the actual logic for (ActionBinding binding : actionBindings) { if ((actions & binding.flag) == 0) { mSpeedDialView.removeActionItemById(binding.actionItemId); } } mSpeedDialView.setOnActionSelectedListener(actionItem -> { ActionBinding selectedBinding = null; for (ActionBinding binding : actionBindings) { if (actionItem.getId() == binding.actionItemId) { selectedBinding = binding; break; } } if (selectedBinding != null) { selectedBinding.action.run(); } else { Log.e(TAG, "Unrecognized speed dial action item. Do nothing. id=" + actionItem.getId()); } return true; }); showSpeedDialIfAnyChecked(); return view; } @Override public void onStop() { restoreActionBarTitle(); // it might have been changed to "N selected". Restore original. super.onStop(); } private void showSpeedDialIfAnyChecked() { if (checkedIds.size() > 0) { if (!mSpeedDialView.isShown()) { mSpeedDialView.show(); } } else { mSpeedDialView.hide(); // hide() also handles UI, e.g., overlay properly. } } @Override public void onCreateOptionsMenu(Menu menu, MenuInflater inflater) { super.onCreateOptionsMenu(menu, inflater); inflater.inflate(R.menu.episodes_apply_action_options, menu); mSelectToggle = menu.findItem(R.id.select_toggle); mSelectToggle.setOnMenuItemClickListener(item -> { if (checkedIds.size() == episodes.size()) { checkNone(); } else { checkAll(); } return true; }); } @Override public void onPrepareOptionsMenu (Menu menu) { // Prepare icon for select toggle button int[] icon = new int[1]; @StringRes int titleResId; if (checkedIds.size() == episodes.size()) { icon[0] = R.attr.ic_select_none; titleResId = R.string.deselect_all_label; } else { icon[0] = R.attr.ic_select_all; titleResId = R.string.select_all_label; } TypedArray a = getActivity().obtainStyledAttributes(icon); Drawable iconDrawable = a.getDrawable(0); a.recycle(); mSelectToggle.setIcon(iconDrawable); mSelectToggle.setTitle(titleResId); } @Override public boolean onOptionsItemSelected(MenuItem item) { @StringRes int resId = 0; switch(item.getItemId()) { case R.id.select_options: return true; case R.id.check_all: checkAll(); resId = R.string.selected_all_label; break; case R.id.check_none: checkNone(); resId = R.string.deselected_all_label; break; case R.id.check_played: checkPlayed(true); resId = R.string.selected_played_label; break; case R.id.check_unplayed: checkPlayed(false); resId = R.string.selected_unplayed_label; break; case R.id.check_downloaded: checkDownloaded(true); resId = R.string.selected_downloaded_label; break; case R.id.check_not_downloaded: checkDownloaded(false); resId = R.string.selected_not_downloaded_label; break; case R.id.check_queued: checkQueued(true); resId = R.string.selected_queued_label; break; case R.id.check_not_queued: checkQueued(false); resId = R.string.selected_not_queued_label; break; case R.id.check_has_media: checkWithMedia(); resId = R.string.selected_has_media_label; break; case R.id.sort_title_a_z: sortByTitle(false); return true; case R.id.sort_title_z_a: sortByTitle(true); return true; case R.id.sort_date_new_old: sortByDate(true); return true; case R.id.sort_date_old_new: sortByDate(false); return true; case R.id.sort_duration_long_short: sortByDuration(true); return true; case R.id.sort_duration_short_long: sortByDuration(false); return true; } if(resId != 0) { Snackbar.make(getActivity().findViewById(R.id.content), resId, Snackbar.LENGTH_SHORT) .show(); return true; } else { return false; } } private void sortByTitle(final boolean reverse) { Collections.sort(episodes, (lhs, rhs) -> { if (reverse) { return -1 * lhs.getTitle().compareTo(rhs.getTitle()); } else { return lhs.getTitle().compareTo(rhs.getTitle()); } }); refreshTitles(); refreshCheckboxes(); } private void sortByDate(final boolean reverse) { Collections.sort(episodes, (lhs, rhs) -> { if (lhs.getPubDate() == null) { return -1; } else if (rhs.getPubDate() == null) { return 1; } int code = lhs.getPubDate().compareTo(rhs.getPubDate()); if (reverse) { return -1 * code; } else { return code; } }); refreshTitles(); refreshCheckboxes(); } private void sortByDuration(final boolean reverse) { Collections.sort(episodes, (lhs, rhs) -> { int ordering; if (!lhs.hasMedia()) { ordering = 1; } else if (!rhs.hasMedia()) { ordering = -1; } else { ordering = lhs.getMedia().getDuration() - rhs.getMedia().getDuration(); } if(reverse) { return -1 * ordering; } else { return ordering; } }); refreshTitles(); refreshCheckboxes(); } private void checkAll() { for (FeedItem episode : episodes) { if(!checkedIds.contains(episode.getId())) { checkedIds.add(episode.getId()); } } refreshCheckboxes(); } private void checkNone() { checkedIds.clear(); refreshCheckboxes(); } private void checkPlayed(boolean isPlayed) { for (FeedItem episode : episodes) { if(episode.isPlayed() == isPlayed) { if(!checkedIds.contains(episode.getId())) { checkedIds.add(episode.getId()); } } else { if(checkedIds.contains(episode.getId())) { checkedIds.remove(episode.getId()); } } } refreshCheckboxes(); } private void checkDownloaded(boolean isDownloaded) { for (FeedItem episode : episodes) { if(episode.hasMedia() && episode.getMedia().isDownloaded() == isDownloaded) { if(!checkedIds.contains(episode.getId())) { checkedIds.add(episode.getId()); } } else { if(checkedIds.contains(episode.getId())) { checkedIds.remove(episode.getId()); } } } refreshCheckboxes(); } private void checkQueued(boolean isQueued) { for (FeedItem episode : episodes) { if(episode.isTagged(FeedItem.TAG_QUEUE) == isQueued) { checkedIds.add(episode.getId()); } else { checkedIds.remove(episode.getId()); } } refreshCheckboxes(); } private void checkWithMedia() { for (FeedItem episode : episodes) { if(episode.hasMedia()) { checkedIds.add(episode.getId()); } else { checkedIds.remove(episode.getId()); } } refreshCheckboxes(); } private void refreshTitles() { titles.clear(); for(FeedItem episode : episodes) { titles.add(episode.getTitle()); } mAdapter.notifyDataSetChanged(); } private void refreshCheckboxes() { for (int i = 0; i < episodes.size(); i++) { FeedItem episode = episodes.get(i); boolean checked = checkedIds.contains(episode.getId()); mListView.setItemChecked(i, checked); } ActivityCompat.invalidateOptionsMenu(EpisodesApplyActionFragment.this.getActivity()); showSpeedDialIfAnyChecked(); updateActionBarTitle(); } private void saveActionBarTitle() { ActionBar actionBar = ((AppCompatActivity) getActivity()).getSupportActionBar(); if (actionBar != null) { CharSequence title = actionBar.getTitle(); if (title == null) { title = ""; } actionBarTitleOriginal = title; } } private void restoreActionBarTitle() { ActionBar actionBar = ((AppCompatActivity) getActivity()).getSupportActionBar(); if (actionBar != null) { actionBar.setTitle(actionBarTitleOriginal); } } private void updateActionBarTitle() { ActionBar actionBar = ((AppCompatActivity) getActivity()).getSupportActionBar(); if (actionBar != null) { CharSequence title = checkedIds.size() > 0 ? getString(R.string.num_selected_label, checkedIds.size()) : actionBarTitleOriginal; actionBar.setTitle(title); } } private void queueChecked() { DBWriter.addQueueItem(getActivity(), true, checkedIds.toArray()); close(R.plurals.added_to_queue_batch_label, checkedIds.size()); } private void removeFromQueueChecked() { DBWriter.removeQueueItem(getActivity(), true, checkedIds.toArray()); close(R.plurals.removed_from_queue_batch_label, checkedIds.size()); } private void markedCheckedPlayed() { DBWriter.markItemPlayed(FeedItem.PLAYED, checkedIds.toArray()); close(R.plurals.marked_read_batch_label, checkedIds.size()); } private void markedCheckedUnplayed() { DBWriter.markItemPlayed(FeedItem.UNPLAYED, checkedIds.toArray()); close(R.plurals.marked_unread_batch_label, checkedIds.size()); } private void downloadChecked() { // download the check episodes in the same order as they are currently displayed List toDownload = new ArrayList<>(checkedIds.size()); for(FeedItem episode : episodes) { if(checkedIds.contains(episode.getId())) { toDownload.add(episode); } } try { DBTasks.downloadFeedItems(getActivity(), toDownload.toArray(new FeedItem[toDownload.size()])); } catch (DownloadRequestException e) { e.printStackTrace(); DownloadRequestErrorDialogCreator.newRequestErrorDialog(getActivity(), e.getMessage()); } close(R.plurals.downloading_batch_label, checkedIds.size()); } private void deleteChecked() { for(long id : checkedIds.toArray()) { FeedItem episode = idMap.get(id); if(episode.hasMedia()) { DBWriter.deleteFeedMediaOfItem(getActivity(), episode.getMedia().getId()); } } close(R.plurals.deleted_episode_batch_label, checkedIds.size()); } private void close(@PluralsRes int msgId, int numItems) { if (numItems > 0) { Snackbar.make(getActivity().findViewById(R.id.content), getResources().getQuantityString(msgId, numItems, numItems), Snackbar.LENGTH_LONG ) .setAction(android.R.string.ok, v -> {}) .show(); } getActivity().getSupportFragmentManager().popBackStack(); } }